Converted Rico3 icons to CSS sprites. Moved minsrc and baselibs directories out of...
authorMatt Brown <dowdybrown@yahoo.com>
Sat, 7 May 2011 13:35:30 +0000 (13:35 +0000)
committerMatt Brown <dowdybrown@yahoo.com>
Sat, 7 May 2011 13:35:30 +0000 (13:35 +0000)
git-svn-id: svn://svn.code.sf.net/p/openrico/code/trunk/rico3@76 53df2df2-7ab5-4331-af62-ea79255fa4e2

87 files changed:
examples/asp/ex2editfilter.asp
examples/asp/ex2editfilterKW.asp
examples/asp/ex2nosession.asp
examples/dotnet/ex2editfilter.aspx
examples/dotnet/ex2nosession.aspx
examples/html/LoadRicoClient.js
examples/php/LoadRicoClient.php
examples/php/ex2editfilter.php
examples/php/ex2nosession.php
minsrc/baselibs/README.TXT [new file with mode: 0644]
minsrc/baselibs/glow.core-1.7.js [new file with mode: 0644]
minsrc/baselibs/prototype.js [new file with mode: 0644]
minsrc/ricoCalendar.js [new file with mode: 0644]
minsrc/ricoColorPicker.js [new file with mode: 0644]
minsrc/ricoDragDrop.js [new file with mode: 0644]
minsrc/ricoGridCommon.js [new file with mode: 0644]
minsrc/ricoLiveGrid.js [new file with mode: 0644]
minsrc/ricoLiveGridAjax.js [new file with mode: 0644]
minsrc/ricoLiveGridControls.js [new file with mode: 0644]
minsrc/ricoLiveGridForms.js [new file with mode: 0644]
minsrc/ricoLiveGridMenu.js [new file with mode: 0644]
minsrc/ricoSearch.js [new file with mode: 0644]
minsrc/ricoSimpleGrid.js [new file with mode: 0644]
minsrc/ricoTree.js [new file with mode: 0644]
minsrc/ricoUI.js [new file with mode: 0644]
plugins/asp/LoadRicoClient.asp
plugins/asp/ricoLiveGridForms.vbs
plugins/php/ricoLiveGridForms.php
ricoClient/css/coffee-with-milk.css
ricoClient/css/grayedout.css
ricoClient/css/rico.css
ricoClient/css/seaglass.css
ricoClient/css/warmfall.css
ricoClient/images/calarrow.png [deleted file]
ricoClient/images/calendaricon.gif [deleted file]
ricoClient/images/checkmark.gif [deleted file]
ricoClient/images/close_b.gif [deleted file]
ricoClient/images/close_w.gif [deleted file]
ricoClient/images/delete.gif [deleted file]
ricoClient/images/delete_w.gif [deleted file]
ricoClient/images/divider.gif [deleted file]
ricoClient/images/doc.gif [deleted file]
ricoClient/images/dotbutton.gif [deleted file]
ricoClient/images/drop.gif [deleted file]
ricoClient/images/filtercol.gif [deleted file]
ricoClient/images/folderclosed.gif [deleted file]
ricoClient/images/folderopen.gif [deleted file]
ricoClient/images/info_icon.gif [deleted file]
ricoClient/images/left_b.gif [deleted file]
ricoClient/images/left_w.gif [deleted file]
ricoClient/images/link.gif [deleted file]
ricoClient/images/m.gif [deleted file]
ricoClient/images/node.gif [deleted file]
ricoClient/images/nodeblank.gif [deleted file]
ricoClient/images/nodelast.gif [deleted file]
ricoClient/images/nodeline.gif [deleted file]
ricoClient/images/nodem.gif [deleted file]
ricoClient/images/nodemlast.gif [deleted file]
ricoClient/images/nodep.gif [deleted file]
ricoClient/images/nodeplast.gif [deleted file]
ricoClient/images/p.gif [deleted file]
ricoClient/images/removeFilter.gif [deleted file]
ricoClient/images/ricoIcons.gif [new file with mode: 0644]
ricoClient/images/right_b.gif [deleted file]
ricoClient/images/right_w.gif [deleted file]
ricoClient/images/sort_asc.gif [deleted file]
ricoClient/images/sort_desc.gif [deleted file]
ricoClient/images/tableFilterCollapse.gif [deleted file]
ricoClient/images/tableFilterExpand.gif [deleted file]
ricoClient/js/baselibs/README.TXT [deleted file]
ricoClient/js/baselibs/glow.core-1.7.js [deleted file]
ricoClient/js/baselibs/prototype.js [deleted file]
ricoClient/js/minsrc/ricoCalendar.js [deleted file]
ricoClient/js/minsrc/ricoColorPicker.js [deleted file]
ricoClient/js/minsrc/ricoDragDrop.js [deleted file]
ricoClient/js/minsrc/ricoGridCommon.js [deleted file]
ricoClient/js/minsrc/ricoLiveGrid.js [deleted file]
ricoClient/js/minsrc/ricoLiveGridAjax.js [deleted file]
ricoClient/js/minsrc/ricoLiveGridControls.js [deleted file]
ricoClient/js/minsrc/ricoLiveGridForms.js [deleted file]
ricoClient/js/minsrc/ricoLiveGridMenu.js [deleted file]
ricoClient/js/minsrc/ricoSearch.js [deleted file]
ricoClient/js/minsrc/ricoSimpleGrid.js [deleted file]
ricoClient/js/minsrc/ricoTree.js [deleted file]
ricoClient/js/minsrc/ricoUI.js [deleted file]
ricoClient/js/rico.js
ricoClient/js/rico_min.js

index e4f1600..3e39f11 100644 (file)
@@ -27,16 +27,13 @@ set RicoClient = Nothing
 \r
 function orders_FormInit() {\r
   var cal=new Rico.CalendarControl("Cal");\r
-  Rico.EditControls.register(cal, Rico.imgDir+'calarrow.png');\r
+  Rico.EditControls.register(cal, 'rico-icon rico-calarrow');\r
   \r
   var CustTree=new Rico.TreeControl("CustomerTree","CustTree.asp");\r
-  Rico.EditControls.register(CustTree, Rico.imgDir+'dotbutton.gif');\r
+  Rico.EditControls.register(CustTree, 'rico-icon rico-dotbutton');\r
 }\r
 </script>\r
 <style type="text/css">\r
-div.ricoLG_outerDiv thead .ricoLG_cell, div.ricoLG_outerDiv thead td, div.ricoLG_outerDiv thead th {\r
-       height:1.5em;\r
-}\r
 div.ricoLG_cell {\r
   white-space:nowrap;\r
 }\r
index 6f74943..9e254b9 100644 (file)
@@ -34,9 +34,6 @@ function orders_FormInit() {
 }\r
 </script>\r
 <style type="text/css">\r
-div.ricoLG_outerDiv thead .ricoLG_cell, div.ricoLG_outerDiv thead td, div.ricoLG_outerDiv thead th {\r
-       height:1.5em;\r
-}\r
 div.ricoLG_cell {\r
   white-space:nowrap;\r
 }\r
index 9200846..8cb71a4 100644 (file)
@@ -61,9 +61,6 @@ function orders_FormInit() {
 }\r
 </script>\r
 <style type="text/css">\r
-div.ricoLG_outerDiv thead .ricoLG_cell, div.ricoLG_outerDiv thead td, div.ricoLG_outerDiv thead th {\r
-       height:1.5em;\r
-}\r
 div.ricoLG_cell {\r
   white-space:nowrap;\r
 }\r
index 137777a..acca634 100644 (file)
@@ -51,9 +51,6 @@ function ex8_FormInit() {
 </script>\r
 \r
 <style type="text/css">\r
-div.ricoLG_outerDiv thead .ricoLG_cell, div.ricoLG_outerDiv thead td, div.ricoLG_outerDiv thead th {\r
-       height:1.5em;\r
-}\r
 div.ricoLG_cell {\r
   white-space:nowrap;\r
 }\r
index f0ad536..becd1a5 100644 (file)
@@ -62,9 +62,6 @@ function ex8_FormInit() {
 </script>\r
 \r
 <style type="text/css">\r
-div.ricoLG_outerDiv thead .ricoLG_cell, div.ricoLG_outerDiv thead td, div.ricoLG_outerDiv thead th {\r
-       height:1.5em;\r
-}\r
 div.ricoLG_cell {\r
   white-space:nowrap;\r
 }\r
index ef0f8c5..54833ff 100644 (file)
@@ -1,7 +1,8 @@
 Rico_CONFIG = {\r
   jsDir: "../../ricoClient/js/",       // directory containing Rico's javascript files\r
   cssDir: "../../ricoClient/css/",     // directory containing Rico's css files\r
-  imgDir: "../../ricoClient/images/",  // directory containing Rico's image files\r
+  imgResize: "../../ricoClient/images/resize.gif",\r
+  imgIcons: "../../ricoClient/images/ricoIcons.gif",\r
   enableLogging: false,    // enable console logging\r
   grid_striping: true,     // apply row striping to LiveGrids?\r
   LoadBaseLib: true,       // load base Javascript library (prototype, jQuery, etc)?\r
index 6d21f00..c24db98 100644 (file)
@@ -33,8 +33,8 @@ function SetConfig() {
   echo "Rico_CONFIG = {\n";\r
   if ($ricoLogging) echo "enableLogging: true,\n";\r
   echo "jsDir: '" . $jsDir . "',\n";\r
-  echo "cssDir: '" . $cssDir . "',\n";\r
-  echo "imgDir: '" . $imgDir . "'\n";\r
+  echo "imgResize: '" . $imgDir . "resize.gif',\n";\r
+  echo "imgIcons: '" . $imgDir . "ricoIcons.gif'\n";\r
   echo "};";\r
   echo "</script>\n";\r
 }\r
@@ -86,7 +86,6 @@ function LoadTheme($theme) {
   switch ($prefix) {\r
     case 'j':\r
       requireRicoJS("Themeroller");\r
-      //echo "<link type='text/css' rel='stylesheet' href='".$cssDir."jquery-base/ui.base.css' />";\r
       echo "<link type='text/css' rel='Stylesheet' href='" . $jQuery_theme_path . $theme."/jquery-ui.css' />";\r
       break;\r
     case 'r':\r
index dee9e32..50e6e5d 100644 (file)
@@ -31,9 +31,6 @@ function orders_FormInit() {
 
 <link href="../demo.css" type="text/css" rel="stylesheet" />
 <style type="text/css">
-div.ricoLG_outerDiv thead .ricoLG_cell, div.ricoLG_outerDiv thead td, div.ricoLG_outerDiv thead th {
-       height:1.5em;
-}
 div.ricoLG_cell {
   white-space:nowrap;
 }
index 64b7631..b212ba1 100644 (file)
@@ -61,9 +61,6 @@ function orders_FormInit() {
 
 <link href="../demo.css" type="text/css" rel="stylesheet" />
 <style type="text/css">
-div.ricoLG_outerDiv thead .ricoLG_cell, div.ricoLG_outerDiv thead td, div.ricoLG_outerDiv thead th {
-       height:1.5em;
-}
 div.ricoLG_cell {
   white-space:nowrap;
 }
diff --git a/minsrc/baselibs/README.TXT b/minsrc/baselibs/README.TXT
new file mode 100644 (file)
index 0000000..32689cb
--- /dev/null
@@ -0,0 +1,5 @@
+The files in this directory are included in the Rico distribution\r
+for the purpose of demonstrating the examples. Each of these libraries \r
+has licensing terms that are different from Rico itself. Make sure you\r
+understand the terms before deploying any library into a production \r
+environment.
\ No newline at end of file
diff --git a/minsrc/baselibs/glow.core-1.7.js b/minsrc/baselibs/glow.core-1.7.js
new file mode 100644 (file)
index 0000000..e6d6e0e
--- /dev/null
@@ -0,0 +1,20 @@
+/*     
+       Copyright 2009 British Broadcasting Corporation
+
+       Licensed under the Apache License, Version 2.0 (the "License");
+       you may not use this file except in compliance with the License.
+       You may obtain a copy of the License at
+
+          http://www.apache.org/licenses/LICENSE-2.0
+
+       Unless required by applicable law or agreed to in writing, software
+       distributed under the License is distributed on an "AS IS" BASIS,
+       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+       See the License for the specific language governing permissions and
+       limitations under the License.
+*/
+(function(){var f={glow:true},b=/([$^\\\/()|?+*\[\]{}.-])/g,a=navigator.userAgent.toLowerCase(),k="1.7.3",r=0,g=[],m=0,q=[],o=0,s=false,n={VERSION:k,UID:"glow"+Math.floor(Math.random()*(1<<30)),isDomReady:window.gloader&&gloader.isReady,isReady:window.gloader&&gloader.isReady,env:function(){var u=[0,NaN],d=(/opera[\s\/]([\w\.]+)/.exec(a)||u)[1],v=d?NaN:(/msie ([\w\.]+)/.exec(a)||u)[1],x=(/rv:([\w\.]+).*gecko\//.exec(a)||u)[1],e=(/applewebkit\/([\w\.]+)/.exec(a)||u)[1],t=(/khtml\/([\w\.]+)/.exec(a)||u)[1],w=parseFloat;return{gecko:w(x),ie:w(v),opera:w(d),webkit:w(e),khtml:w(t),version:v||x||e||d||t,standardsMode:document.compatMode!="BackCompat"&&(!v||v>=6)};}(),module:function(u){var t=2,w=u.depends[0]||[],d=w.length,e=u.name,v=window.glow;if(u.library[1]!=n.VERSION){throw new Error("Cannot register "+e+": Version mismatch");}if(w[2]){for(;t<d;t++){if(!f[w[t]]){throw new Error("Module "+w[t]+" required before "+e);}}}u.builder(n);f[e]=true;return n;},ready:function(d){if(this.isReady){d();}else{q[o++]=d;}return this;},_readyBlockers:{},_addReadyBlock:function(d){if(!n._readyBlockers[d]){n._readyBlockers[d]=true;n.isReady=false;r++;}return n;},_removeReadyBlock:function(d){if(n._readyBlockers[d]){n._readyBlockers[d]=false;r--;if(!r){n.isReady=true;p();}}return n;},onDomReady:function(d){if(this.isDomReady){d();}else{g[m++]=d;}},lang:{trim:function(d){return d.replace(/^\s*((?:[\S\s]*\S)?)\s*$/,"$1");},toArray:function(e){if(e.constructor==Array){return e;}var u=[],t=0,d=e.length;for(;t<d;t++){u[t]=e[t];}return u;},apply:function(d,t){for(var e in t){d[e]=t[e];}return d;},map:function(e,x,w){if(Array.prototype.map){return Array.prototype.map.call(e,x,w||e);}if(!x.call){throw new TypeError();}var d=e.length,v=[],u=w||e,t=0;for(;t<d;t++){if(t in e){v[t]=x.call(u,e[t],t,e);}}return v;},replace:(function(){var d="g".replace(/g/,function(){return"l";})!="l",e=String.prototype.replace;return function(y,w,t){var z,v,x,u;if(!d||typeof(t)!="function"){return e.call(y,w,t);}if(!(w instanceof RegExp)){z=y.indexOf(w);return z==-1?y:e.call(y,w,t.call(null,w,z,y));}u=[];x=w.lastIndex=0;while((v=w.exec(y))!=null){z=v.index;u[u.length]=y.slice(x,z);u[u.length]=t.apply(null,v);if(w.global){x=w.lastIndex;}else{x=z+v[0].length;break;}}u[u.length]=y.slice(x);return u.join("");};})(),interpolate:function(t,w,u){var d,v,e,x;u=u||{};if(u.escapeHtml){if(!n.dom){throw new Error("glow.lang.interpolate - glow.dom is needed for escapeHtml");}x=n.dom.create("<div></div>");}if(u.delimiter==undefined){d=/\{[^{}]+\}/g;}else{v=u.delimiter.substr(0,1).replace(b,"\\$1");e=u.delimiter.substr(1,1).replace(b,"\\$1")||v;d=new RegExp(v+"[^"+v+e+"]+"+e,"g");}return t.replace(d,function(D){var A=D.slice(1,-1),C=A.split("."),B,z=0,y=C.length;if(A in w){B=w[A];}else{B=w;for(;z<y;z++){if(C[z] in B){B=B[C[z]];}else{return D;}}}if(u.escapeHtml){B=x.text(B).html();}return B;});},hasOwnProperty:{}.hasOwnProperty?function(d,e){return d.hasOwnProperty(e);}:function(x,y){var u=x[y],w=x.__proto__,t=w?w[y]:{};if(u!==t){return true;}var e=n.lang.hasOwnProperty(w,y),d=w[y]={},v=(x[y]!==d);delete w[y];if(e){w[name]=d;}return v;},extend:function(e,u,d){var t=function(){},v;t.prototype=u.prototype;v=new t();e.prototype=v;v.constructor=e;e.base=u;if(d){n.lang.apply(e.prototype,d);}},clone:function(t){var d,u,e;t=t.valueOf();if(typeof t!=="object"){return t;}else{if(t[0]||t.concat){e=[];d=t.length;while(d--){e[d]=arguments.callee(t[d]);}}else{e={};for(d in t){e[d]=arguments.callee(t[d]);}}return e;}}}},h=n.env,l=document;function c(){n.isDomReady=true;for(var d=0;d<m;d++){g[d]();}}function p(){if(s){return;}s=true;for(var d=0;d<o;){q[d]();d++;if(r){break;}}q=q.slice(d);o=o-d;s=false;}(function(){if(n.isDomReady){return;}n._addReadyBlock("glow_domReady");if(h.ie){if(typeof window.frameElement!="undefined"){l.attachEvent("onreadystatechange",function(){if(l.readyState=="complete"){l.detachEvent("onreadystatechange",arguments.callee);c();n._removeReadyBlock("glow_domReady");}});}else{(function(){try{l.documentElement.doScroll("left");}catch(u){setTimeout(arguments.callee,0);return;}c();n._removeReadyBlock("glow_domReady");})();}}else{if(n.env.webkit<525.13&&typeof l.readyState!="undefined"){var e=function(){if(/loaded|complete/.test(l.readyState)){c();n._removeReadyBlock("glow_domReady");}else{setTimeout(e,0);}};e();}else{var t=function(){if(t.fired){return;}t.fired=true;c();n._removeReadyBlock("glow_domReady");};if(l.addEventListener){l.addEventListener("DOMContentLoaded",t,false);}var d=window.onload;window.onload=function(){if(d){d();}t();};}}})();n.isSupported=!(h.ie<6||(h.gecko<1.9&&!/^1\.8\.1/.test(h.version))||h.opera<9||h.webkit<412);if(!n.isSupported){n._addReadyBlock("glow_browserSupport");}if(window.gloader){gloader.library({name:"glow",version:"1.7.3",builder:function(){return n;}});}else{if(window.glow){throw new Error("Glow global object already exists");}else{window.glow=n;}}if(n.ie){try{document.execCommand("BackgroundImageCache",false,true);}catch(j){}}})();
+/*@cc_on @*/
+/*@if (@_jscript_version > 5.5)@*/
+(window.gloader||glow).module({name:"glow.i18n",library:["glow","1.7.3"],depends:[["glow","1.7.3"]],builder:function(r){var A;var t={l:/^[a-z]$/,lv:/^[a-z]{2,3}$/,s:/^[A-Z][a-z]{3}$/,r:/^[A-Z]{2}|[0-9]{3}$/,v:/^[a-z0-9]{4,}$/};var s=1,k=2,l=4,j=8,p=s+k+l+j,f=s+l+j,x=s+k+j,c=s+j,y=s+k+l,e=s+l,d=s+k;var m={l:s,s:k,r:l,v:j},I=["l","s","r","v"],F={l:0,s:1,r:2,v:3};var C={};var b={};var o=w(document.documentElement.lang||"en")||w("en");function D(K){for(var J in t){if(t[J].test(K)){return J;}}return"";}function w(V){if(!V.split){V="";}var N=V.split("-"),Q=N.length,R=[],K={l:"",s:"",r:"",v:""},J=0,O=J,U=0,P,S;for(var M=0,T=I.length;M<T;M++){O=J;P=I[M];S=F[P];while((D(N[O]).indexOf(P)==-1)&&(O<Q)){O++;}if(O<Q){R[S]=N[O];U+=m[P];K[P]=N[O];N[O]="*";J=O;}}var L=R.join("-").replace(/-+/g,"-");if((L=="")||(L.substring(0,1)=="-")){return false;}else{return{canonical:L,mask:U,subtags:K};}}function g(K,M,J){var L;if((J&~K.mask)==0){L=K.subtags.l;if(k&J){L=L+"-"+K.subtags.s;}if(l&J){L=L+"-"+K.subtags.r;}if(j&J){L=L+"-"+K.subtags.v;}if(M(L)){return L;}}return false;}function a(K,N,J,L){var M;switch(K.mask){case f:if((M=g(K,N,f))){break;}case e:if((M=g(K,N,e))){break;}case p:if((M=g(K,N,p))){break;}case y:if((M=g(K,N,y))){break;}case x:if((M=g(K,N,x))){break;}case d:if((M=g(K,N,d))){break;}case c:if((M=g(K,N,c))){break;}case s:if((M=g(K,N,s))){break;}default:if(N("en")){M="en";}else{M=null;}}if(M==null){L();}else{J(M);}}function v(L){var J=o,K=w(L);if(K){o=K;o.next=J;}return A;}function H(){o=o.next||o;return A;}function B(){return o.canonical;}function h(M,L,Q){var J=w(L),P,O,K;if(J){P=C[J.canonical]=C[J.canonical]||{};O=P[M]=P[M]||{};K=b[M]=b[M]||{};for(var N in Q){O[N]=Q[N];K[N]=1;}}return A;}function E(K,J){var L={},S=J||{},N=b[K]||{},O=o,T,Q;function P(U){if(C[U]&&C[U][K]&&C[U][K][Q]){return true;}else{return false;}}function R(U){L[Q]=C[U][K][Q];}function M(){L[Q]="[Error! No "+K+"."+Q+" on "+O.canonical+"]";}if(S.locale!=undefined){T=w(S.locale);if(T){O=T;}}for(Q in N){a(O,P,R,M);}return L;}function G(K,L){for(var J in L){h(J,K,L[J]);}return A;}function u(L,M){var K=M||{},J=w(L);if(K.module){if(K.label){return q(J,K.module,K.label);}else{return n(J,K.module);}}else{return z(J);}return null;}function q(L,N,M){var J;function P(Q){if(C[Q]&&C[Q][N]&&C[Q][N][M]){return true;}else{return false;}}function O(Q){J=Q;}function K(){J="**error** - no negotiated value exists";}a(L,P,O,K);return J;}function n(L,O){var K=b[O]||{},N={},M;function Q(R){if(C[R]&&C[R][O]&&C[R][O][M]){return true;}else{return false;}}function P(R){N[M]=R;}function J(){N[M]="**error** - no negotiated value exists";}for(M in K){a(L,Q,P,J);}return N;}function z(K){var N={},M,L;function P(Q){if(C[Q]&&C[Q][M]&&C[Q][M][L]){return true;}else{return false;}}function O(Q){N[M][L]=Q;}function J(){N[M][L]="**error** - no negotiated value exists";}for(M in b){N[M]={};for(L in b[M]){a(K,P,O,J);}}return N;}r.i18n=A={setLocale:v,revertLocale:H,getLocale:B,addLocaleModule:h,getLocaleModule:E,addLocalePack:G,checkLocale:u};G("en",{PROPERTIES:{LANGUAGE:"English",DIR:"ltr"}});}});(window.gloader||glow).module({name:"glow.dom",library:["glow","1.7.3"],depends:[],builder:function(u){var n=u.env,k=u.lang,s={tagName:/^(\w+|\*)/,combinator:/^\s*([>]?)\s*/,classNameOrId:(n.webkit<417)?new RegExp("^([\\.#])((?:(?![\\.#\\[:\\s\\\\]).|\\\\.)+)"):/^([\.#])((?:[^\.#\[:\\\s]+|\\.)+)/},X=/([$^\\\/()|?+*\[\]{}.-])/g,B={},R={checked:"checked","class":"className",disabled:"disabled","for":"htmlFor",maxlength:"maxLength"},c={checked:true,disabled:true},ah={maxlength:function(r){return r.toString()=="2147483647"?undefined:r;}},ad=1,x="_unique"+u.UID,ai="_uniqueData"+u.UID,ag=1,L=[],I={black:0,silver:12632256,gray:8421504,white:16777215,maroon:8388608,red:16711680,purple:8388736,fuchsia:16711935,green:32768,lime:65280,olive:8421376,yellow:16776960,navy:128,blue:255,teal:32896,aqua:65535,orange:16753920},D=/height|top/,t=/^rgb\(([\d\.]+)(%?),\s*([\d\.]+)(%?),\s*([\d\.]+)(%?)/i,A=/^(?:(width|height)|(border-(top|bottom|left|right)-width))$/,C=/width|height|top$|bottom$|left$|right$|spacing$|indent$|font-size/,T,d,K,H,aa=window,l=document,V,G,w,P=l.createElement("div"),y=[1,"<table>","</table>"],ab=[0,"",""],O=n.webkit<526?[0,"","</div>",true]:[1,"b<div>","</div>"],a=[3,"<table><tbody><tr>","</tr></tbody></table>"],E={caption:y,thead:y,th:a,colgroup:y,tbody:y,tr:[2,"<table><tbody>","</tbody></table>"],td:a,tfoot:y,option:[1,"<select>","</select>"],legend:[1,"<fieldset>","</fieldset>"],link:O,script:O,style:O};if(n.ie){window.attachEvent("onunload",function(){P=null;});}u.ready(function(){V=l.body;G=l.documentElement;});(function(){var r=l.createElement("div");r.a=1;w=!!r.cloneNode(true).a;})();function af(r){for(var aj=r.firstChild;aj;aj=aj.nextSibling){if(aj.nodeType==1){return aj;}}return null;}function v(r){return new RegExp(["(^|\\s)",r.replace(X,"\\$1"),"($|\\s)"].join(""),"g");}function N(ap){var ao=[],al=(/^\s*<([^\s>]+)/.exec(ap)||[,"div"])[1],aj=E[al]||ab,am,ak,an=0;P.innerHTML=(aj[1]+ap+aj[2]);ak=P;am=aj[0];while(am--){ak=ak.lastChild;}while(ak.firstChild){ao[an++]=ak.removeChild(ak.firstChild);}ak=null;return ao;}function p(al){var ak=[],aj=0;for(;al[aj];aj++){ak[aj]=al[aj];}return ak;}function e(am,aj){for(var al=this,r=0,ak=al.length;r<ak;r++){aj.call(al[r],am.call?am.call(al[r],r):am);}return al;}if(document.all){T=function(ak,aj){var al=0,r=ak.length,am=aj.length;if(typeof aj.length=="number"){for(;al<am;al++){ak[r++]=aj[al];}}else{for(;aj[al];al++){ak[r++]=aj[al];}}};}else{T=function(ak,aj){var al=0,r=ak.length;for(;aj[al];al++){ak[r++]=aj[al];}};}function M(r){return(r.ownerDocument&&!r.ownerDocument.body)||(r.documentElement&&!r.documentElement.body);}if(n.ie){d=function(ak){if(ak.length==1){return ak;}var am=[],aj=0,al=0;for(;ak[al];al++){if(ak[al].getAttribute(x)!=ad&&ak[al].nodeType==1){am[aj++]=ak[al];}ak[al].setAttribute(x,ad);}for(al=0;ak[al];al++){ak[al].removeAttribute(x);}ad++;return am;};}else{d=function(ak){if(ak.length==1){return ak;}var am=[],aj=0,al=0;for(;ak[al];al++){if(ak[al][x]!=ad&&ak[al].nodeType==1){am[aj++]=ak[al];}ak[al][x]=ad;}ad++;return am;};}if(document.all){H=function(aj,al){var am=[],ak=0;for(;al[ak];ak++){if(aj=="*"&&al[ak].all&&!M(al[ak])){T(am,al[ak].all);}else{T(am,al[ak].getElementsByTagName(aj));}}return am;};}else{H=function(ak,am){var an=[],al=0,aj=am.length;for(;al<aj;al++){T(an,am[al].getElementsByTagName(ak));}return an;};}function b(am){var al=[],an=am.childNodes,ak=0,aj=0;for(;an[ak];ak++){if(an[ak].nodeType==1&&an[ak].nodeName!="!"){al[aj++]=an[ak];}}return al;}var U=["border-left-width","border-right-width","padding-left","padding-right"],Q=["border-top-width","border-bottom-width","padding-top","padding-bottom"];function f(ap,an){var am,aj=n.standardsMode?G:V,ao=(an=="width"),ak=ao?"Width":"Height",al;if(ap.window){am=n.webkit<522.11?(ao?ap.innerWidth:ap.innerHeight):n.webkit?(ao?V.clientWidth:ap.innerHeight):n.opera<9.5?(ao?V.clientWidth:V.clientHeight):(ao?aj.clientWidth:aj.clientHeight);}else{if(ap.getElementById){am=Math.max(V["scroll"+ak],G["scroll"+ak]);}else{al=ao?U:Q;am=ap["offset"+ak]-parseInt(Z(ap,al));}}return am;}function ae(r){if(n.ie<6){return r.document.body;}else{return r.ownerDocument.body;}}function j(al,am,ak){if(typeof am=="number"||/\d$/.test(am)){am+="px";}for(var aj=0,r=al.length;aj<r;aj++){al[aj].style[ak]=am;}}function W(r){if(r=="float"){return n.ie?"styleFloat":"cssFloat";}return k.replace(r,/-(\w)/g,function(aj,ak){return ak.toUpperCase();});}function z(ap,an){var am,ao=ap.style,al=ao.display,aj=ao.visibility,ak=ao.position;ao.visibility="hidden";ao.position="absolute";ao.display="block";if(!h(ap)){ao.position=ak;am=z(ap.parentNode,an);ao.display=al;ao.visibility=aj;}else{am=an();ao.display=al;ao.position=ak;ao.visibility=aj;}return am;}function h(r){return r.offsetWidth||r.offsetHeight;}function Z(ar,al){var ak,au=0,ap=0,ao=al.length,an=l.defaultView&&(l.defaultView.getComputedStyle(ar,null)||l.defaultView.getComputedStyle),am=ar.currentStyle,at,aq,aj=al.push||A.exec(al)||[];if(al.push){for(;ap<ao;ap++){au+=parseInt(Z(ar,al[ap]),10)||0;}return au+"px";}if(aj[1]){if(!h(ar)){return z(ar,function(){return f(ar,aj[1])+"px";});}return f(ar,aj[1])+"px";}else{if(aj[2]&&u.env.ie&&Z(ar,"border-"+aj[3]+"-style")=="none"){return"0";}else{if(an){if(typeof an=="function"){at=ar.style.display;ak=z(ar,function(){if(al=="display"){ar.style.display=at;if(!l.defaultView.getComputedStyle(ar,null)){return"none";}ar.style.display="block";}return Z(ar,al);});}else{if(n.webkit>500&&n.webkit<526&&al=="margin-right"&&an.getPropertyValue("position")!="absolute"){al="margin-left";}ak=an.getPropertyValue(al);}}else{if(am){if(al=="opacity"){aq=/alpha\(opacity=([^\)]+)\)/.exec(am.filter);return aq?String(parseInt(aq[1],10)/100):"1";}ak=String(am[W(al)]);if(/^-?[\d\.]+(?!px)[%a-z]+$/i.test(ak)&&al!="font-size"){ak=J(ar,ak,D.test(al))+"px";}}}}}if(al.indexOf("color")!=-1){ak=S(ak).toString();}else{if(ak.indexOf("url")==0){ak=ak.replace(/\"/g,"");}}return ak;}function J(ao,aq,am){var ak=am?"top":"left",an=am?"Top":"Left",ar=ao.style,al=ar[ak],ap=ao.runtimeStyle[ak],aj;ao.runtimeStyle[ak]=ao.currentStyle[ak];ar[ak]=aq;aj=ar["pixel"+an];ar[ak]=al;ao.runtimeStyle[ak]=ap;return aj;}function S(ak){if(/^(transparent|rgba\(0, ?0, ?0, ?0\))$/.test(ak)){return"transparent";}var ao,aj,ap,aq,al,an=Math.round,ar=parseInt,am=parseFloat;if(ao=t.exec(ak)){aj=ao[2]?an(((am(ao[1])/100)*255)):ar(ao[1]);ap=ao[4]?an(((am(ao[3])/100)*255)):ar(ao[3]);aq=ao[6]?an(((am(ao[5])/100)*255)):ar(ao[5]);}else{if(typeof ak=="number"){al=ak;}else{if(ak.charAt(0)=="#"){if(ak.length=="4"){ak="#"+ak.charAt(1)+ak.charAt(1)+ak.charAt(2)+ak.charAt(2)+ak.charAt(3)+ak.charAt(3);}al=ar(ak.slice(1),16);}else{al=I[ak];}}aj=(al)>>16;ap=(al&65280)>>8;aq=(al&255);}ak=new String("rgb("+aj+", "+ap+", "+aq+")");ak.r=aj;ak.g=ap;ak.b=aq;return ak;}function m(an){var am="",ak=an.childNodes,al=0,aj=ak.length;for(;al<aj;al++){if(ak[al].nodeType==3){am+=ak[al].nodeValue;}else{if(ak[al].nodeType==1){am+=m(ak[al]);}}}return am;}function q(ao,al){var ak=[],aj=0,r,am=0,an=ao.length;for(;am<an;am++){r=ao[am];while(r=r[al+"Sibling"]){if(r.nodeType==1&&r.nodeName!="!"){ak[aj++]=r;break;}}}return Y.get(ak);}function F(aj){var r=aj.offsetParent;while(r&&Y.get(r).css("position")=="static"){r=r.offsetParent;}if(!r&&Y.get(G).css("position")!="static"){r=G;}return r||null;}function ac(am,al){var ak,aj="scroll"+(al?"Left":"Top");if(am.window){ak=am.document.documentElement[aj]||(al?am.pageXOffset:am.pageYOffset)||0;}else{ak=am[aj];}return ak;}function o(ak,aj,r){if(ak.window){ak.scrollTo(aj?r:ac(ak,true),!aj?r:ac(ak,false));}else{ak["scroll"+(aj?"Left":"Top")]=r;}}function g(r,al,ak){var aj=r.length;if(ak!==undefined){while(aj--){o(r[aj],al,ak);}return r;}else{return ac(r[0],al);}}var Y={};Y.get=function(){var am=new u.dom.NodeList(),al=0,ak=arguments,aj=ak.length;for(;al<aj;al++){if(typeof ak[al]=="string"){am.push(new u.dom.NodeList().push(l).get(ak[al]));}else{am.push(ak[al]);}}return am;};Y.create=function(an,am){var r=[],aj=0,al=0,ak;am=u.lang.apply({interpolate:null,escapeHtml:false},am||{});if(am.interpolate){an=k.interpolate(an,am.interpolate,{escapeHtml:am.escapeHtml});}ak=N(an);for(;ak[aj];aj++){if(ak[aj].nodeType==1&&ak[aj].nodeName!="!"){r[al++]=ak[aj];}else{if(ak[aj].nodeType==3&&k.trim(ak[aj].nodeValue)!==""){throw new Error("glow.dom.create - Text must be wrapped in an element");}}}return new Y.NodeList().push(r);};Y.parseCssColor=function(r){var aj=S(r);return{r:aj.r,g:aj.g,b:aj.b};};Y.NodeList=function(){this.length=0;};Y.NodeList.prototype={item:function(r){return this[r];},push:function(){var ak=arguments,aj=ak.length,al=0,ao,r,am=this,an=Array.prototype.push;for(;al<aj;al++){if(!ak[al]){continue;}else{if(ak[al].nodeType==1||ak[al].nodeType==9||ak[al].document){an.call(am,ak[al]);}else{if(ak[al][0]){for(ao=0,r=ak[al].length;ao<r;ao++){an.call(am,ak[al][ao]);}}}}}return am;},each:function(al){for(var r=0,ak=this,aj=ak.length;r<aj;r++){al.call(ak[r],r,ak);}return ak;},eq:function(al){var ak=this,r=0,aj=ak.length;if(!al.push){al=[al];}if(al.length!=ak.length){return false;}for(;r<aj;r++){if(ak[r]!=al[r]){return false;}}return true;},isWithin:function(al){if(al.push){al=al[0];}if(!al||!this.length){return false;}var ak=this,r=0,aj=ak.length,am;if(al.contains&&n.webkit>=521){for(;r<aj;r++){if(!(al.contains(ak[r])&&ak[r]!=al)){return false;}}}else{if(ak[0].compareDocumentPosition){for(;r<aj;r++){if(!(ak[r].compareDocumentPosition(al)&8)){return false;}}}else{for(;r<aj;r++){am=ak[r];while(am=am.parentNode){if(am==al){break;}}if(!am){return false;}}}}return true;},attr:function(ak){var am=this,aj=arguments,r=aj.length,al,an;if(am.length===0){return r>1?am:undefined;}if(typeof ak=="object"){for(al in ak){if(k.hasOwnProperty(ak,al)){am.attr(al,ak[al]);}}return am;}if(n.ie&&R[ak]){if(r>1){e.call(am,aj[1],function(ao){this[R[ak]]=ao;});return am;}an=am[0][R[ak]];if(c[ak]){return an?ak:undefined;}else{if(ah[ak]){return ah[ak](an);}}return an;}if(r>1){e.call(am,aj[1],function(ao){this.setAttribute(ak,ao);});return am;}return M(am[0])?am[0].getAttribute(ak):am[0].getAttribute(ak,2);},removeAttr:function(aj){var r=n.ie&&R[aj],am=this,ak=0,al=am.length;for(;ak<al;ak++){if(r){am[ak][r]="";}else{am[ak].removeAttribute(aj);}}return am;},hasAttr:function(al){var an=this[0],ak=an.attributes;if(M(an)&&n.ie){var ak=an.attributes,am=0,aj=ak.length;for(;am<aj;am++){if(ak[am].nodeName==al){return ak[am].specified;}}return false;}else{if(this[0].getAttributeNode){var r=this[0].getAttributeNode(al);return r?r.specified:false;}}return typeof ak[r]!="undefined";},prop:function(r,am){if(r.constructor===Object){var al=r,ak;for(ak in al){this.prop(ak,al[ak]);}return this;}if(am!==undefined){var aj=this.length;while(aj--){this[aj][r]=am;}return this;}if(!this[0]){return undefined;}return this[0][r];},hasClass:function(r){for(var aj=0,ak=this.length;aj<ak;aj++){if((" "+this[aj].className+" ").indexOf(" "+r+" ")!=-1){return true;}}return false;},addClass:function(r){for(var aj=0,ak=this.length;aj<ak;aj++){if((" "+this[aj].className+" ").indexOf(" "+r+" ")==-1){this[aj].className+=((this[aj].className)?" ":"")+r;}}return this;},removeClass:function(r){var ak=v(r),am=this,aj=0,al=am.length;for(;aj<al;aj++){am[aj].className=am[aj].className.replace(ak," ");}return am;},toggleClass:function(aj){var al=this.length,ak,r=" "+aj+" ";while(al--){ak=" "+this[al].className+" ";if(ak.indexOf(r)!=-1){this[al].className=ak.replace(r," ");}else{this[al].className+=" "+aj;}}return this;},val:function(){function al(ap){var am=ap.type,an=ap.checked,ar=ap.value,at=[],ao=0;if(am=="radio"){return an?ar:"";}else{if(am=="checkbox"){return an?ar:"";}else{if(am=="select-one"){return ap.selectedIndex>-1?ap.options[ap.selectedIndex].value:"";}else{if(am=="select-multiple"){for(var aq=ap.options.length;ao<aq;ao++){if(ap.options[ao].selected){at[at.length]=ap.options[ao].value;}}return at;}else{return ar;}}}}}function aj(an){var av={},au={},aq=an.elements,at=0,ap=aq.length,am,ax,ar,ao,aw;for(;at<ap;at++){ax=aq[at];aw=ax.nodeName.toLowerCase();am=ax.name;if(aw=="fieldset"||aw=="object"||!am){continue;}if(ax.type=="checkbox"&&!ax.checked){if(!am in av){av[am]=undefined;}}else{if(ax.type=="radio"){if(au[am]){au[am][au[am].length]=ax;}else{au[am]=[ax];}}else{var ay=al(ax);if(am in av){if(av[am].push){av[am][av[am].length]=ay;}else{av[am]=[av[am],ay];}}else{av[am]=ay;}}}}for(at in au){ar=0;for(ap=au[at].length;ar<ap;ar++){ao=au[at][ar];am=ao.name;if(ao.checked){av[ao.name]=ao.value;break;}}if(!am in av){av[am]=undefined;}}return av;}function r(an,aw){var am,ax,at={},aq,ar=0,ao,au,av,ap;for(am in aw){ax=an[am];if(ax&&ax[0]&&!ax.options){aw[am]=aw[am]&&aw[am].push?aw[am]:[aw[am]];at.radios=[];at.checkboxesSelects=[];at.multiSelects=[];at.other=[];for(ar=0;ax[ar];ar++){ap=ax[ar].type;if(ap=="radio"){aq="radios";}else{if(ap=="select-one"||ap=="checkbox"){aq="checkboxesSelects";}else{if(ap=="select-multiple"){aq="multiSelects";}else{aq="other";}}}at[aq][at[aq].length]=ax[ar];}for(ar=0;at.multiSelects[ar];ar++){aw[am]=ak(at.multiSelects[ar],aw[am]);}for(ar=0;at.checkboxesSelects[ar];ar++){ak(at.checkboxesSelects[ar],"");for(ao=0,au=aw[am].length;ao<au;ao++){if(ak(at.checkboxesSelects[ar],aw[am][ao])){aw[am].slice(ao,1);break;}}}for(ar=0;at.radios[ar];ar++){at.radios[ar].checked=false;av=false;for(ao=0,au=aw[am].length;ao<au;ao++){if(ak(at.radios[ar],aw[am][ao])){aw[am].slice(ao,1);av=true;break;}if(av){break;}}}for(ar=0;at.other[ar]&&aw[am][ar]!==undefined;ar++){ak(at.other[ar],aw[am][ar]);}}else{if(ax&&ax.nodeName){ak(ax,aw[am]);}}}}function ak(ao,aq){var ar=0,an,ap=0,av,am,au;if(ao.type=="select-one"){for(an=ao.options.length;ar<an;ar++){if(ao.options[ar].value==aq){ao.selectedIndex=ar;return true;}}return false;}else{if(ao.type=="select-multiple"){var at=!!aq.push;for(ar=0,an=ao.options.length;ar<an;ar++){am=ao.options[ar];au=am.value;if(at){am.selected=false;for(av=aq.length;ap<av;ap++){if(au==aq[ap]){am.selected=true;aq.splice(ap,1);break;}}}else{return am.selected=aq==au;}}return false;}else{if(ao.type=="radio"||ao.type=="checkbox"){ao.checked=aq==ao.value;return aq==ao.value;}else{ao.value=aq;return true;}}}}return function(){var am=arguments,aq=am[0],ap=this,an=0,ao=ap.length;if(am.length===0){return ap[0].nodeName=="FORM"?aj(ap[0]):al(ap[0]);}if(ap[0].nodeName=="FORM"){if(!typeof aq=="object"){throw"value for FORM must be object";}r(ap[0],aq);}else{for(;an<ao;an++){ak(ap[an],aq);}}return ap;};}(),slice:function(){return new Y.NodeList().push(Array.prototype.slice.apply(this,arguments));},sort:function(al){var ak=this,aj=0,r;if(!ak.length){return ak;}if(!al){if(typeof ak[0].sourceIndex=="number"){al=function(an,am){return an.sourceIndex-am.sourceIndex;};}else{if(ak[0].compareDocumentPosition){al=function(an,am){return 3-(an.compareDocumentPosition(am)&6);};}else{r=H("*",[l]);for(;r[aj];aj++){r[aj]._sourceIndex=aj;}al=function(an,am){return an._sourceIndex-am._sourceIndex;};}}}return Y.get([].sort.call(ak,al));},filter:function(am){var aj=[],r=0,ak=0,al=this.length;for(;ak<al;ak++){if(am.apply(this[ak],[ak])){aj[r++]=this[ak];}}return Y.get(aj);},children:function(){var aj=[],r=0,ak=0,an=0,am=this.length,al;for(;ak<am;ak++){aj=aj.concat(b(this[ak]));}return Y.get(aj);},parent:function(){var aj=[],r=0,ak=0,al=this.length;for(;ak<al;ak++){aj[r++]=this[ak].parentNode;}return Y.get(d(aj));},ancestors:function(){var aj=[],r=0,ak=0,al=this.length,am;while(ak<al){am=this[ak].parentNode;while(am&&am.nodeType==1){aj[r++]=am;am=am.parentNode;}ak++;}return Y.get(d(aj));},wrap:function(am){var al=this.length,r,ak,aj;if(typeof am=="string"){aj=Y.create(am);}else{aj=Y.get(am);}for(i=0;i<al;i++){ak=aj[0];while(ak){r=af(ak);if(r){ak=r;}else{break;}}if(this[i].parentNode){aj.insertBefore(this[i]);}if(i!=al-1){aj=aj.clone();}ak.appendChild(this[i]);}return this;},unwrap:function(){var aj,r=this.parent(),ak=r.length;for(i=0;i<ak;i++){aj=r.slice(i,i+1);if(!aj[0].parentNode){aj.children().remove();aj.destroy();}else{aj.children().insertBefore(aj);aj.destroy();}}return this;},next:function(){return q(this,"next");},prev:function(){return q(this,"previous");},is:function(aj){var ak=u.dom.get(aj),am=0,r=this.length,al,an;node:for(;am<r;am++){for(al=0,an=ak.length;al<an;al++){if(this[am]==ak[al]){continue node;}}return false;}return true;},text:function(){var r=arguments,aj=0,al=this,ak=al.length;if(r.length>0){for(;aj<ak;aj++){al[aj].innerHTML="";al[aj].appendChild(l.createTextNode(r[0]));}return al;}return al[0].innerText||al[0].textContent==undefined?m(al[0]):al[0].textContent;},empty:function(){var aj=0,r=this.length;for(;aj<r;aj++){while(this[aj].firstChild){this[aj].removeChild(this[aj].firstChild);}}return this;},remove:function(){for(var al=this,aj=0,ak=al.length,r;aj<ak;aj++){if(r=al[aj].parentNode){r.removeChild(al[aj]);}}return al;},destroy:function(){this.get("*").push(this).removeData();this.appendTo(P);P.innerHTML="";Array.prototype.splice.call(this,0,this.length);return this;},clone:function(am){var aj=[],al=this.length,r,ak;eventIdProp="__eventId"+u.UID;if(al===0){return new Y.NodeList();}while(al--){aj[al]=this[al].cloneNode(true);}r=Y.get(aj).get("*").push(aj);if(w&&!M(aj[0])){al=r.length;while(al--){r[al][eventIdProp]=null;}}ak=this.get("*").push(this);al=r.length;while(al--){r[al].removeAttribute(ai);u.dom.get(r[al]).data(u.dom.get(ak[al]).data());}if(am){if(!u.events){throw"glow.events required to clone event listeners";}u.events._copyListeners(this.get("*").push(this),r||Y.get(aj).get("*").push(aj));}return Y.get(aj);},html:function(r){var aj=0,ak=this.length;if(r!==undefined){return this.empty().append(r.toString());}return this[0]?this[0].innerHTML:"";},width:function(r){if(r==undefined){return f(this[0],"width");}j(this,r,"width");return this;},height:function(r){if(r==undefined){return f(this[0],"height");}j(this,r,"height");return this;},scrollLeft:function(r){return g(this,true,r);},scrollTop:function(r){return g(this,false,r);},show:function(){var ak=0,aj=this.length,r,al;for(;ak<aj;ak++){r=Y.get(this[ak]);al=r[0].style;if(r.css("display")=="none"){al.display="";al.visibility="visible";if(r.css("display")=="none"){al.display="block";}}}return this;},hide:function(){return this.css("display","none").css("visibility","hidden");},css:function(ao,al){var ak=this,an,aj=0,r=ak.length,am=ao;if(ao.constructor===Object){for(style in ao){this.css(style,ao[style]);}return ak;}else{if(al!=undefined){ao=W(ao);for(;aj<r;aj++){an=ak[aj].style;if(typeof al=="number"&&C.test(am)){al=al.toString()+"px";}if(ao=="opacity"&&n.ie){an.zoom="1";if(al===""){an.filter="";}else{an.filter="alpha(opacity="+Math.round(Number(al,10)*100)+")";}}else{an[ao]=al;}}return ak;}else{if(!r){return;}return Z(ak[0],ao);}}},offset:function(){var an=this[0],al={x:ac(window,true),y:ac(window,false)};if(!u.env.webkit&&an.getBoundingClientRect){var ap=an.getBoundingClientRect();return{top:ap.top+al.y-G.clientTop,left:ap.left+al.x-G.clientLeft};}else{var ao=an.offsetTop,r=an.offsetLeft,aj=an,aq,am=false,ak=an;while(an=an.offsetParent){r+=an.offsetLeft;ao+=an.offsetTop;if(Z(an,"position")=="fixed"){am=true;}if(n.gecko||n.webkit>500){r+=parseInt(Z(an,"border-left-width"))||0;ao+=parseInt(Z(an,"border-top-width"))||0;}if(an.nodeName.toLowerCase()!="body"){ak=an;}}an=aj;while((an=an.parentNode)&&(an!=V)&&(an!=G)){r-=an.scrollLeft;ao-=an.scrollTop;if(n.gecko&&Z(an,"overflow")!="visible"){r+=parseInt(Z(an,"border-left-width"));ao+=parseInt(Z(an,"border-top-width"));}}if(am){r+=al.x;ao+=al.y;}if((n.webkit<500&&(am||Z(ak,"position")=="absolute"))||(n.gecko&&Z(ak,"position")!="absolute")){r-=V.offsetLeft;ao-=V.offsetTop;}return{left:r,top:ao};}},position:function(){var aj=Y.get(F(this[0])),ao=!!aj[0],an=parseInt(this.css("margin-left"))||0,am=parseInt(this.css("margin-top"))||0,al=(ao&&parseInt(aj.css("border-left-width")))||0,r=(ao&&parseInt(aj.css("border-top-width")))||0,ap=this.offset(),ak=ao?aj.offset():{top:0,left:0};return{left:ap.left-ak.left-an-al,top:ap.top-ak.top-am-r};},append:function(an){var am=this,aj=0,ak=1,al=am.length,r;if(al==0){return am;}r=typeof an=="string"?p(N(an)):an.nodeType?[an]:p(an);for(;r[aj];aj++){am[0].appendChild(r[aj]);}for(;ak<al;ak++){for(aj=0;r[aj];aj++){am[ak].appendChild(r[aj].cloneNode(true));}}return am;},prepend:function(ao){var am=this,aj=0,ak=1,al=am.length,r,an;if(al==0){return am;}r=typeof ao=="string"?p(N(ao)):ao.nodeType?[ao]:p(ao);an=am[0].firstChild;for(;r[aj];aj++){am[0].insertBefore(r[aj],an);}for(;ak<al;ak++){an=am[ak].firstChild;for(aj=0;r[aj];aj++){am[ak].insertBefore(r[aj].cloneNode(true),an);}}return am;},appendTo:function(r){if(!(r instanceof Y.NodeList)){r=Y.get(r);}r.append(this);return this;},prependTo:function(r){if(!(r instanceof Y.NodeList)){r=Y.get(r);}r.prepend(this);return this;},after:function(ap){var ao=this,an=ao.length,ak,aj,al,am=1,r;if(an==0){return ao;}ak=typeof ap=="string"?Y.create(ap):ap instanceof Y.NodeList?ap:Y.get(ap);aj=ak.length;for(al=aj-1;al>=0;al--){ao[0].parentNode.insertBefore(ak[al],ao[0].nextSibling);}for(;am<an;am++){r=ak.clone();for(al=aj-1;al>=0;al--){ao[am].parentNode.insertBefore(r[al],ao[am].nextSibling);}}return ao;},before:function(ap){var ao=this,an=ao.length,al=0,am=1,ak,aj,r;if(an==0){return ao;}ak=typeof ap=="string"?Y.create(ap):ap instanceof Y.NodeList?ap:Y.get(ap);aj=ak.length;for(;al<aj;al++){ao[0].parentNode.insertBefore(ak[al],ao[0]);}for(;am<an;am++){r=ak.clone();for(al=0;al<aj;al++){ao[am].parentNode.insertBefore(r[al],ao[am]);}}return ao;},insertAfter:function(r){if(!(r instanceof Y.NodeList)){r=Y.get(r);}r.after(this);return this;},insertBefore:function(r){if(!(r instanceof Y.NodeList)){r=Y.get(r);}r.before(this);return this;},replaceWith:function(r){if(n.webkit<500){this.after(K).remove();Y.get("u.glow-placeholder").after(r).remove();}else{this.after(r).remove();}return this;},data:function(ak,al){if(typeof ak==="object"){for(var an in ak){this.data(an,ak[an]);}return this;}var r,am;switch(arguments.length){case 0:if(this[0]===undefined){return undefined;}r=this[0][ai]||ag++;return L[r]||(L[r]={});case 1:if(this[0]===undefined){return undefined;}r=this[0][ai];return r?L[r][ak]:undefined;case 2:for(var aj=this.length;aj--;){am=this[aj];if(!(r=am[ai])){r=ag++;am[ai]=r;L[r]={};}L[r][ak]=al;}return this;default:throw new Error("glow.dom.NodeList#data expects 2 or less arguments, not "+arguments.length+".");}},removeData:function(ak){var am,aj=this.length,r;while(aj--){am=this[aj];r=am[ai];if(r!==undefined){switch(arguments.length){case 0:L[r]=undefined;am[ai]=undefined;try{delete am[ai];}catch(al){am.removeAttribute&&am.removeAttribute(ai);}break;case 1:L[r][ak]=undefined;delete L[r][ak];break;default:throw new Error("glow.dom.NodeList#removeData expects 1 or less arguments, not "+arguments.length+".");}}}return this;},get:function(){function ak(az){if(B[az]){return B[az];}var ar=[],aC=0,at,aA,av,ax,aB,ay,au=true,aw=az;while(az&&az!=ay){aA="";av="";ay=az;if(ax=s.combinator.exec(az)){at=ax[1];az=az.slice(ax[0].length);}if(ax=s.tagName.exec(az)){aA=ax[1];az=az.slice(ax[0].length);}if(ax=s.classNameOrId.exec(az)){if(ax[1]=="#"){av=ax[2];az=az.slice(ax[0].length);}}if(!at){if(av&&au){ar[aC++]=[am,[av.replace(/\\/g,""),aA||"*",null]];}else{ar[aC++]=[H,[aA||"*",null]];if(av){ar[aC++]=[ap,[av.replace(/\\/g,""),null]];}}}else{if(at==">"){ar[aC++]=[al,[null]];if(av){ar[aC++]=[ap,[av.replace(/\\/g,""),null]];}if(aA&&aA!="*"){ar[aC++]=[aq,[aA,null]];}}}aB=true;while(aB){if(az.charAt(0)=="#"||az.charAt(0)=="."){if(ax=s.classNameOrId.exec(az)){if(az.charAt(0)=="#"){ar[aC++]=[ap,[ax[2].replace(/\\/g,""),null]];}else{ar[aC++]=[ao,[ax[2].replace(/\\/g,""),null]];}az=az.slice(ax[0].length);}else{throw new Error("Invalid Selector "+aw);}}else{aB=false;}}au=false;}if(az!==""){throw new Error("Invalid Selector "+aw);}return B[az]=ar;}function an(ar,av){var au=av;for(var at=0,aw=ar.length;at<aw;at++){ar[at][1][ar[at][1].length-1]=au;au=ar[at][0].apply(this,ar[at][1]);}return au;}function am(at,ax,au){var ar=[],aB=0,aw=[],aA=0,ay;for(var az=0,av=au.length;az<av;az++){if(au[az].getElementById){ay=au[az].getElementById(at);if(ay&&(ay.tagName==ax.toUpperCase()||ax=="*"||ay.tagName==ax)){ar[aB++]=ay;}}else{aw[aA++]=au[az];}}if(aw[0]){aw=H(ax,aw);aw=ap(at,aw);}return ar.concat(aw);}function al(au){var av=[],at=0,ar=au.length;for(;at<ar;at++){T(av,b(au[at]));}return av;}function ap(av,at){for(var ar=0,au=at.length;ar<au;ar++){if(at[ar].id==av){return[at[ar]];}}return[];}function aq(av,au){var ax=[],ar=0;for(var at=0,aw=au.length;at<aw;at++){if(au[at].tagName==av.toUpperCase()||au[at].tagName==av){ax[ar++]=au[at];}}return ax;}function ao(av,au){var ax=[],ar=0;for(var at=0,aw=au.length;at<aw;at++){if((" "+au[at].className+" ").indexOf(" "+av+" ")!=-1){ax[ar++]=au[at];}}return ax;}function aj(ay,au){var av;var ax=[];var aw=ay.split(",");for(var at=0,ar=aw.length;at<ar;at++){av=ak(u.lang.trim(aw[at]));ax=ax.concat(an(av,au));}return ax;}function r(at,av){at=at.length?at:[at];var aw=[];var ar;for(var au=0;at[au];au++){ar=u.dom.get(at[au]);for(var ax=0;av[ax];ax++){if(ar.isWithin(av[ax])){aw[aw.length]=ar[0];break;}}}return aw;}return function(av){if(!this.length){return this;}var au=[];for(var at=0,ar=arguments.length;at<ar;at++){if(typeof arguments[at]=="string"){au=au.concat(aj(arguments[at],this));}else{au=au.concat(r(arguments[at],this));}}return u.dom.get(d(au));};}()};K=Y.create('<u class="glow-placeholder"></u>');u.dom=Y;}});(window.gloader||glow).module({name:"glow.events",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.dom"]],builder:function(o){var k=o.dom.get;var C={};var x=1;var m=1;var n={};var b={};var p={};var l="__eventId"+o.UID;var g=l+"PreventDefault";var u=l+"StopPropagation";var E={};var f=1;var B={};var z={};var G=1;var d=2;var s=4;var h={TAB:"\t",SPACE:" ",ENTER:"\n",BACKTICK:"`"};var K={"96":223};var t={CAPSLOCK:20,NUMLOCK:144,SCROLLLOCK:145,BREAK:19,BACKTICK:223,BACKSPACE:8,PRINTSCREEN:44,MENU:93,SPACE:32,SHIFT:16,CTRL:17,ALT:18,ESC:27,TAB:9,META:91,RIGHTMETA:92,ENTER:13,F1:112,F2:113,F3:114,F4:115,F5:116,F6:117,F7:118,F8:119,F9:120,F10:121,F11:122,F12:123,INS:45,HOME:36,PAGEUP:33,DEL:46,END:35,PAGEDOWN:34,LEFT:37,UP:38,RIGHT:39,DOWN:40};var I={};for(var H in t){I[""+t[H]]=H;}var y="0123456789=;'\\/#,.-";function D(O){var N=z[O];if(!N){return false;}var M=B[N];if(!M){return false;}for(var L=0,r=M.length;L<r;L++){if(M[L][0]==O){M.splice(L,1);return true;}}return false;}function A(r){E[r]=C.addListener(document,"key"+r,function(R){var P=0;if(R.ctrlKey){P+=G;}if(R.altKey){P+=d;}if(R.shiftKey){P+=s;}var Q=R.chr?R.chr.toLowerCase():R.key?R.key.toLowerCase():R.keyCode;var N=P+":"+Q+":"+r;var O=B[N]?B[N].slice(0):[];if(R.shiftKey){var S=(P&~s)+":"+Q+":"+r;if(B[S]){for(var M=0,L=B[S].length;M<L;M++){O[O.length]=B[S][M];}}}if(!O){return;}for(var M=0,L=O.length;M<L;M++){if(O[M][2].call(O[M][3]||this,R)===false){R.preventDefault();}}return !R.defaultPrevented();});}function w(){var r;for(r in b){C.removeListener(r);}}var q;var J,e;function F(O,N,L){var r;L=!!L;if(o.env.opera){if(N.toLowerCase()=="resize"&&!J&&O==window){J=C.addListener(window.document.body,"resize",function(R){C.fire(window,"resize",R);});}else{if(N.toLowerCase()=="scroll"&&!e&&O==window){e=C.addListener(window.document,"scroll",function(R){C.fire(window,"scroll",R);});}}}var Q=function(T){if(!T){T=window.event;}var S=new C.Event(),R=N.toLowerCase();S.nativeEvent=T;S.source=T.target||T.srcElement;S.relatedTarget=T.relatedTarget||(R=="mouseover"?T.fromElement:T.toElement);S.button=o.env.ie?(T.button&1?0:T.button&2?2:1):T.button;if(T.pageX||T.pageY){S.pageX=T.pageX;S.pageY=T.pageY;}else{if(T.clientX||T.clientY){S.pageX=T.clientX+document.body.scrollLeft+document.documentElement.scrollLeft;S.pageY=T.clientY+document.body.scrollTop+document.documentElement.scrollTop;}}if(R=="mousewheel"){S.wheelDelta=T.wheelDelta?T.wheelDelta/120:T.detail?-T.detail/3:0;if(S.wheelDelta==0){return;}}if(R.indexOf("key")!=-1){S.altKey=!!T.altKey;S.ctrlKey=!!T.ctrlKey;S.shiftKey=!!T.shiftKey;if(N=="keydown"){q=T.keyCode;}S.charCode=T.keyCode&&T.charCode!==0?undefined:T.charCode;if(R=="keypress"){if(typeof(S.charCode)=="undefined"){S.charCode=T.keyCode;}if(o.env.opera&&S.charCode&&S.charCode==q&&y.indexOf(String.fromCharCode(S.charCode))==-1){S.charCode=undefined;S.keyCode=q;}}if(S.charCode&&S.charCode<=49){S.charCode=undefined;}if(S.charCode){S.chr=String.fromCharCode(S.charCode);}else{if(T.keyCode){S.charCode=undefined;S.keyCode=K[T.keyCode.toString()]||T.keyCode;S.key=I[S.keyCode];if(h[S.key]){S.chr=h[S.key];S.charCode=S.chr.charCodeAt(0);}}}if(S.chr){S.capsLock=S.chr.toUpperCase()!=S.chr?S.shiftKey:S.chr.toLowerCase()!=S.chr?!S.shiftKey:undefined;}}C.fire(this,N,S);if(S.defaultPrevented()){return false;}};if(O.addEventListener&&(!o.env.webkit||o.env.webkit>418)){if((N=="focus"||N=="blur")&&(o.env.opera)){O.parentNode.addEventListener(N,function(){},true);}O.addEventListener(N.toLowerCase()=="mousewheel"&&o.env.gecko?"DOMMouseScroll":N,Q,L);}else{var M="on"+N;var P=O[M];if(P){O[M]=function(){var R=P.apply(this,arguments),S=Q.apply(this,arguments);return(R!==false)&&(S!==false);};}else{O[M]=Q;}}O=null;}function a(M,N){var O=k(M),r=N?"mouseout":"mouseover",L=N?"mouseleave":"mouseenter";C.addListener(M,r,function(Q){var P=k(Q.relatedTarget);if(!P.eq(O)&&!P.isWithin(O)){return !C.fire(O[0],L,Q).defaultPrevented();}});}C._copyListeners=function(R,Q){var M=R.length,P,r,L,O,N;while(M--){if(R[M][l]){P=n[R[M][l]];for(r in P){L=0;O=P[r].length;for(;L<O;L++){N=P[r][L];C.addListener(Q[M],r,N[2],N[3]);}}}}};C.addListener=function(S,M,V,O){var T=false;if(!S){throw"no attachTo paramter passed to addListener";}if(typeof S=="string"){if(!o.dom){throw"glow.dom must be loaded to use a selector as the first argument to glow.events.addListener";}S=k(S);}if(o.dom&&S instanceof o.dom.NodeList){var U=[],R=S.length;while(R--){U[R]=C.addListener(S[R],M,V,O);}return U;}var N;if(!(N=S[l])){N=S[l]=m++;}var Q=x++;var P=[N,M,V,O,Q];b[Q]=P;var r=n[N];if(!r){r=n[N]={};}var L=r[M];if(!L){L=r[M]=[];}L[L.length]=P;if((S.addEventListener||S.attachEvent)&&!p[N+":"+M]){switch(M){case"mouseenter":a(S,false);return Q;case"mouseleave":a(S,true);return Q;case"focus":if(o.env.ie){c(S,true);return Q;}else{T=true;}break;case"blur":if(o.env.ie){c(S,false);return Q;}else{T=true;}break;}F(S,M,T);p[N+":"+M]=true;}return Q;};function c(M,N){var r=N?"focusin":"focusout",L=N?"focus":"blur";C.addListener(M,r,function(O){return !C.fire(M,L,O).defaultPrevented();});}C.removeListener=function(O){if(O&&O.toString().indexOf("k:")!=-1){return D(O);}if(O instanceof Array){var L=O.length;while(L--){C.removeListener(O[L]);}return true;}var N=b[O];if(!N){return false;}delete b[O];var M=n[N[0]][N[1]];for(var L=0,r=M.length;L<r;L++){if(M[L]==N){M.splice(L,1);break;}}if(!M.length){delete n[N[0]][N[1]];}var P=false;for(var L in n[N[0]]){P=true;break;}if(!P){delete n[N[0]];}return true;};C.removeAllListeners=function(Q){var N,M,P=[],O=0,r,L;if(typeof Q=="string"){Q=k(Q);}if(Q instanceof Array||Q instanceof o.dom.NodeList){N=Q.length;while(N--){C.removeAllListeners(Q[N]);}return C;}M=Q[l];if(!M){return C;}L=n[M];for(r in L){N=L[r].length;while(N--){P[O++]=L[r][N][4];}}if(P.length){C.removeListener(P);}return C;};C.fire=function(r,L,M){if(!r){throw"glow.events.fire: required parameter attachedTo not passed (name: "+L+")";}if(!L){throw"glow.events.fire: required parameter name not passed";}if(!M){M=new C.Event();}if(M.constructor===Object){M=new C.Event(M);}if(typeof r=="string"){if(!o.dom){throw"glow.dom must be loaded to use a selector as the first argument to glow.events.addListener";}r=k(r);}M.type=L;M.attachedTo=r;if(!M.source){M.source=r;}if(r instanceof o.dom.NodeList){r.each(function(N){v(r[N],M);});}else{v(r,M);}return M;};function v(R,Q){var L,r,M=r&&r[Q.type];(L=R[l])&&(r=n[L])&&(M=r[Q.type]);if(!M){return Q;}var N;var S=M.slice(0);for(var O=0,P=S.length;O<P;O++){N=S[O];if(N[2].call(N[3]||R,Q)===false){Q.preventDefault();}}}var j=/^((?:(?:ctrl|alt|shift)\+)*)(?:(\w+|.)|[\n\r])$/i;C.addKeyListener=function(Q,P,R,L){P.replace(/^key/i,"");P=P.toLowerCase();if(!(P=="press"||P=="down"||P=="up")){throw"event type must be press, down or up";}if(!E[P]){A(P);}var N=Q.match(j),S=0,T;if(!N){throw"key format not recognised";}if(N[1].toLowerCase().indexOf("ctrl")!=-1){S+=G;}if(N[1].toLowerCase().indexOf("alt")!=-1){S+=d;}if(N[1].toLowerCase().indexOf("shift")!=-1){S+=s;}var r=S+":"+(N[2]?N[2].toLowerCase():"\n")+":"+P;var M="k:"+f++;z[M]=r;var O=B[r];if(!O){O=B[r]=[];}O[O.length]=[M,P,R,L];return M;};C.Event=function(r){if(r){o.lang.apply(this,r);}};C.Event.prototype.preventDefault=function(){if(this[g]){return;}this[g]=true;if(this.nativeEvent&&this.nativeEvent.preventDefault){this.nativeEvent.preventDefault();this.nativeEvent.returnValue=false;}};C.Event.prototype.defaultPrevented=function(){return !!this[g];};C.Event.prototype.stopPropagation=function(){if(this[u]){return;}this[u]=true;var r=this.nativeEvent;if(r){r.cancelBubble=true;if(r.stopPropagation){r.stopPropagation();}}};C.Event.prototype.propagationStopped=function(){return !!this[u];};if(o.env.ie<8||o.env.webkit<500){C.addListener(window,"unload",w);}o.events=C;o.events.listenersByObjId=n;}});(window.gloader||glow).module({name:"glow.data",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.dom"]],builder:function(glow){var TYPES={UNDEFINED:"undefined",OBJECT:"object",NUMBER:"number",BOOLEAN:"boolean",STRING:"string",ARRAY:"array",FUNCTION:"function",NULL:"null"};var TEXT={AT:"@",EQ:"=",DOT:".",EMPTY:"",AND:"&",OPEN:"(",CLOSE:")"};var JSON={HASH:{START:"{",END:"}",SHOW_KEYS:true},ARRAY:{START:"[",END:"]",SHOW_KEYS:false},DATA_SEPARATOR:",",KEY_SEPARATOR:":",KEY_DELIMITER:'"',STRING_DELIMITER:'"',SAFE_PT1:/^[\],:{}\s]*$/,SAFE_PT2:/\\./g,SAFE_PT3:/\"[^\"\\\n\r]*\"|true|false|null|-?\d+(?:\.\d*)?(:?[eE][+\-]?\d+)?/g,SAFE_PT4:/(?:^|:|,)(?:\s*\[)+/g};var SLASHES={TEST:/[\b\n\r\t\\\f\"]/g,B:{PLAIN:"\b",ESC:"\\b"},N:{PLAIN:"\n",ESC:"\\n"},R:{PLAIN:"\r",ESC:"\\r"},T:{PLAIN:"\t",ESC:"\\t"},F:{PLAIN:"\f",ESC:"\\f"},SL:{PLAIN:"\\",ESC:"\\\\"},QU:{PLAIN:'"',ESC:'\\"'}};function _replaceSlashes(s){switch(s){case SLASHES.B.PLAIN:return SLASHES.B.ESC;case SLASHES.N.PLAIN:return SLASHES.N.ESC;case SLASHES.R.PLAIN:return SLASHES.R.ESC;case SLASHES.T.PLAIN:return SLASHES.T.ESC;case SLASHES.F.PLAIN:return SLASHES.F.ESC;case SLASHES.SL.PLAIN:return SLASHES.SL.ESC;case SLASHES.QU.PLAIN:return SLASHES.QU.ESC;default:return s;}}function _getType(object){if((typeof object)==TYPES.OBJECT){if(object==null){return TYPES.NULL;}else{return(object instanceof Array)?TYPES.ARRAY:TYPES.OBJECT;}}else{return(typeof object);}}glow.data={encodeUrl:function(object){var objectType=_getType(object);var paramsList=[];var listLength=0;if(objectType!=TYPES.OBJECT){throw new Error("glow.data.encodeUrl: cannot encode item");}else{for(var key in object){switch(_getType(object[key])){case TYPES.FUNCTION:case TYPES.OBJECT:throw new Error("glow.data.encodeUrl: cannot encode item");break;case TYPES.ARRAY:for(var i=0,l=object[key].length;i<l;i++){switch(_getType(object[key])[i]){case TYPES.FUNCTION:case TYPES.OBJECT:case TYPES.ARRAY:throw new Error("glow.data.encodeUrl: cannot encode item");break;default:paramsList[listLength++]=key+TEXT.EQ+encodeURIComponent(object[key][i]);}}break;default:paramsList[listLength++]=key+TEXT.EQ+encodeURIComponent(object[key]);}}return paramsList.join(TEXT.AND);}},decodeUrl:function(text){if(_getType(text)!=TYPES.STRING){throw new Error("glow.data.decodeUrl: cannot decode item");}else{if(text===""){return{};}}var result={};var keyValues=text.split(/[&;]/);var thisPair,key,value;for(var i=0,l=keyValues.length;i<l;i++){thisPair=keyValues[i].split(TEXT.EQ);if(thisPair.length!=2){throw new Error("glow.data.decodeUrl: cannot decode item");}else{key=glow.lang.trim(decodeURIComponent(thisPair[0]));value=glow.lang.trim(decodeURIComponent(thisPair[1]));switch(_getType(result[key])){case TYPES.ARRAY:result[key][result[key].length]=value;break;case TYPES.UNDEFINED:result[key]=value;break;default:result[key]=[result[key],value];}}}return result;},encodeJson:function(object,options){function _encode(object,options){if(_getType(object)==TYPES.ARRAY){var type=JSON.ARRAY;}else{var type=JSON.HASH;}var serial=[type.START];var len=1;var dataType;var notFirst=false;for(var key in object){dataType=_getType(object[key]);if(dataType!=TYPES.UNDEFINED){if(notFirst){serial[len++]=JSON.DATA_SEPARATOR;}notFirst=true;if(type.SHOW_KEYS){serial[len++]=JSON.KEY_DELIMITER;serial[len++]=key;serial[len++]=JSON.KEY_DELIMITER;serial[len++]=JSON.KEY_SEPARATOR;}switch(dataType){case TYPES.FUNCTION:throw new Error("glow.data.encodeJson: cannot encode item");break;case TYPES.STRING:default:serial[len++]=JSON.STRING_DELIMITER;serial[len++]=glow.lang.replace(object[key],SLASHES.TEST,_replaceSlashes);serial[len++]=JSON.STRING_DELIMITER;break;case TYPES.NUMBER:case TYPES.BOOLEAN:serial[len++]=object[key];break;case TYPES.OBJECT:case TYPES.ARRAY:serial[len++]=_encode(object[key],options);break;case TYPES.NULL:serial[len++]=TYPES.NULL;break;}}}serial[len++]=type.END;return serial.join(TEXT.EMPTY);}options=options||{};var type=_getType(object);if((type==TYPES.OBJECT)||(type==TYPES.ARRAY)){return _encode(object,options);}else{throw new Error("glow.data.encodeJson: cannot encode item");}},decodeJson:function(text,options){if(_getType(text)!=TYPES.STRING){throw new Error("glow.data.decodeJson: cannot decode item");}options=options||{};options.safeMode=options.safeMode||false;var canEval=true;if(options.safeMode){canEval=(JSON.SAFE_PT1.test(text.replace(JSON.SAFE_PT2,TEXT.AT).replace(JSON.SAFE_PT3,JSON.ARRAY.END).replace(JSON.SAFE_PT4,TEXT.EMPTY)));}if(canEval){try{return eval(TEXT.OPEN+text+TEXT.CLOSE);}catch(e){}}throw new Error("glow.data.decodeJson: cannot decode item");},escapeHTML:function(html){return glow.dom.create("<div></div>").text(html).html();}};}});(window.gloader||glow).module({name:"glow.net",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.data","glow.events"]],builder:function(h){var t={XML_ERR:"Cannot get response as XML, check the mime type of the data",POST_DEFAULT_CONTENT_TYPE:"application/x-www-form-urlencoded;"},u=/\+xml$/,p=[],e="c",o="_"+h.UID+"loadScriptCbs",g=h.dom.get,c=h.events,f=function(){},k=1;function s(){if(window.ActiveXObject){return(s=function(){return new ActiveXObject("Microsoft.XMLHTTP");})();}else{return(s=function(){return new XMLHttpRequest();})();}}function d(w){var r=h.lang.apply({onLoad:f,onError:f,onAbort:f,headers:{},async:true,useCache:false,data:null,defer:false,forceXml:false},w||{});if(!("X-Requested-With" in r.headers)){r.headers["X-Requested-With"]="XMLHttpRequest";}return r;}function v(r){return[r,(/\?/.test(r)?"&":"?"),"a",new Date().getTime(),parseInt(Math.random()*100000)].join("");}function j(C,r,z){var y=s(),A=z.data&&(typeof z.data=="string"?z.data:h.data.encodeUrl(z.data)),w,x=new q(y,z);if(!z.useCache){r=v(r);}y.open(C,r,z.async);for(w in z.headers){y.setRequestHeader(w,z.headers[w]);}function B(){x.send=f;if(z.async){if(z.timeout){x._timeout=setTimeout(function(){a(x);var E=new b(y,true,x);c.fire(x,"error",E);},z.timeout*1000);}y.onreadystatechange=function(){if(y.readyState==4){x._timeout&&clearTimeout(x._timeout);x.completed=true;var E=new b(y,false,x);if(E.wasSuccessful){c.fire(x,"load",E);}else{c.fire(x,"error",E);}y.onreadystatechange=new Function();}};y.send(A);return x;}else{y.send(A);x.completed=true;var D=new b(y,false,x);if(D.wasSuccessful){c.fire(x,"load",D);}else{c.fire(x,"error",D);}return D;}}x.send=B;return z.defer?x:B();}var m={};m.get=function(r,w){w=d(w);return j("GET",r,w);};m.post=function(r,w,x){x=d(x);x.data=w;if(!x.headers["Content-Type"]){x.headers["Content-Type"]=t.POST_DEFAULT_CONTENT_TYPE;}return j("POST",r,x);};m.send=function(y,r,w,x){w=w||"";x=d(x);x.data=w;return j(y,r,x);};m.put=function(r,w,x){x=d(x);x.data=w;if(!x.headers["Content-Type"]){x.headers["Content-Type"]=t.POST_DEFAULT_CONTENT_TYPE;}return j("PUT",r,x);};m.del=function(r,w){w=d(w);return j("DELETE",r,w);};m.loadScript=function(w,z){var x=p.length,r,A=e+x,z=d(z),y=new q(x,z),w=z.useCache?w:v(w),B=window[o]||(window[o]={});if(z.onLoad!=f){B[A]=function(){y._timeout&&clearTimeout(y._timeout);y.completed=true;z.onLoad.apply(this,arguments);y.destroy();r=B[A]=undefined;delete B[A];};w=h.lang.interpolate(w,{callback:o+"."+A});}r=p[x]=document.createElement("script");if(z.charset){r.charset=z.charset;}c.addListener(y,"abort",z.onAbort);h.ready(function(){if(z.timeout){y._timeout=setTimeout(function(){a(y);z.onError();},z.timeout*1000);}if(h.env.opera){setTimeout(function(){if(r){r.src=w;}},0);}else{r.src=w;}document.body.appendChild(r);});return y;};function a(w){var r=w.nativeRequest,x=w._callbackIndex;w._timeout&&clearTimeout(w._timeout);if(r){r.onreadystatechange=new Function();r.abort();}else{if(x){window[o][e+x]=f;h.dom.get(p[x]).destroy();}}}function q(y,w){this._timeout=null;this._forceXml=w.forceXml;if(w.forceXml&&y.overrideMimeType){y.overrideMimeType("application/xml");}this.complete=false;if(typeof y=="number"){this._callbackIndex=y;}else{this.nativeRequest=y;}var x=["Load","Error","Abort"],r=0;for(;r<3;r++){c.addListener(this,x[r].toLowerCase(),w["on"+x[r]]);}}q.prototype={send:function(){},abort:function(){if(!this.completed&&!c.fire(this,"abort").defaultPrevented()){a(this);}return this;},destroy:function(){var r=this;if(this._callbackIndex!==undefined){setTimeout(function(){g(p[r._callbackIndex]).destroy();p[r._callbackIndex]=undefined;delete p[r._callbackIndex];},0);}return this;}};function b(x,r,w){c.Event.call(this);this._request=w;this.nativeResponse=x;this.status=r?408:x.status==1223?204:x.status;this.timedOut=!!r;this.wasSuccessful=(this.status>=200&&this.status<300)||this.status==304||(this.status==0&&x.responseText);}function l(){var r=this.header("Content-Type");return u.test(r)||r==="";}h.lang.extend(b,c.Event,{text:function(){return this.nativeResponse.responseText;},xml:function(){var w=this.nativeResponse;if((h.env.ie&&l.call(this))||(this._request._forceXml&&!this._request.nativeRequest.overrideMimeType&&window.ActiveXObject)){var r=new ActiveXObject("Microsoft.XMLDOM");r.loadXML(w.responseText);return r;}else{if(!w.responseXML){throw new Error(t.XML_ERR);}return w.responseXML;}},json:function(r){return h.data.decodeJson(this.text(),{safeMode:r});},header:function(r){return this.nativeResponse.getResponseHeader(r);},statusText:function(){return this.timedOut?"Request Timeout":this.nativeResponse.statusText;}});var n=function(r,x,y,w){this.url=r;this.data=x;this.isGet=y;this.opts=w;};n.prototype={_send:function(){this._addIframe();this._addForm();this._addTimeout();this.onLoad=this._handleResponse;this._submitForm();},_addIframe:function(){this.iframe=h.dom.create('<iframe style="visibility: hidden; position: absolute; height: 0;"></iframe>');var r=this.iframe[0],w=this,x=function(){if(w.onLoad){w.onLoad();}};if(r.attachEvent){r.attachEvent("onload",x);}else{r.onload=x;}g("body").append(this.iframe);},_addForm:function(){var x=this._window().document;if(h.env.ie){x.open();x.write("<html><body></body></html>");x.close();}var w=this.form=x.createElement("form");w.setAttribute("action",this.url);w.setAttribute("method",this.isGet?"GET":"POST");var r=x.getElementsByTagName("body")[0];r.appendChild(w);this._addFormData();},_addFormData:function(){for(var x in this.data){if(!this.data.hasOwnProperty(x)){continue;}if(this.data[x] instanceof Array){var r=this.data[x].length;for(var w=0;w<r;w++){this._addHiddenInput(x,this.data[x][w]);}}else{this._addHiddenInput(x,this.data[x]);}}},_addHiddenInput:function(w,x){var r=this._window().document.createElement("input");r.type="hidden";r.name=w;r.value=x;this.form.appendChild(r);},_window:function(){var r=this.iframe[0];if(r.contentWindow){return r.contentWindow;}throw new Error("could not get contentWindow from iframe");},_addTimeout:function(){var r=this;this.timeout=setTimeout(function(){var w;if(r.opts.hasOwnProperty("onTimeout")){try{r.opts.onTimeout();}catch(x){w=x;}}r._cleanup();if(w){throw new Error("error in xDomainPost onTimeout callback: "+w);}},(this.opts.timeout||10)*1000);},_handleResponse:function(){var w,r,y=this._window();try{r=y.location.href;}catch(x){w=x;}if(r!="about:blank"||w){clearTimeout(this.timeout);this.onLoad=this._readHandler;if("_fullBlankUrl" in this.opts){y.location=this.opts._fullBlankUrl;}else{y.location=window.location.protocol+"//"+window.location.host+(this.opts.blankUrl||"/favicon.ico");}}},_readHandler:function(){var r;if(this.opts.hasOwnProperty("onLoad")){try{this.opts.onLoad(this._window().name);}catch(w){r=w;}}this._cleanup();if(r){throw new Error("error in xDomainPost onLoad callback: "+r);}},_cleanup:function(){this.iframe.remove();},_submitForm:function(){var r=this;this._window().setTimeout(function(){r.form.submit();},0);}};m.xDomainPost=function(r,y,x){var w=new n(r,y,false,x);w._send();};m.xDomainGet=function(r,x){var w=new n(r,{},true,x);w._send();};h.net=m;}});(window.gloader||glow).module({name:"glow.tweens",library:["glow","1.7.3"],depends:[],builder:function(b){function a(c){return function(d){return 1-c(1-d);};}b.tweens={linear:function(){return function(c){return c;};},easeIn:function(c){c=c||2;return function(d){return Math.pow(1,c-1)*Math.pow(d,c);};},easeOut:function(c){return a(this.easeIn(c));},easeBoth:function(c){return this.combine(this.easeIn(c),this.easeOut(c));},overshootIn:function(c){return a(this.overshootOut(c));},overshootOut:function(c){c=c||1.70158;return function(d){if(d==0||d==1){return d;}return((d-=1)*d*((c+1)*d+c)+1);};},overshootBoth:function(c){return this.combine(this.overshootIn(c),this.overshootOut(c));},bounceIn:function(){return a(this.bounceOut());},bounceOut:function(){return function(c){if(c<(1/2.75)){return 7.5625*c*c;}else{if(c<(2/2.75)){return(7.5625*(c-=(1.5/2.75))*c+0.75);}else{if(c<(2.5/2.75)){return(7.5625*(c-=(2.25/2.75))*c+0.9375);}else{return(7.5625*(c-=(2.625/2.75))*c+0.984375);}}}};},bounceBoth:function(){return this.combine(this.bounceIn(),this.bounceOut());},elasticIn:function(c,d){return a(this.elasticOut(c,d));},elasticOut:function(c,d){return function(e){if(e==0||e==1){return e;}if(!d){d=0.3;}if(!c||c<1){c=1;var f=d/4;}else{var f=d/(2*Math.PI)*Math.asin(1/c);}return c*Math.pow(2,-10*e)*Math.sin((e-f)*(2*Math.PI)/d)+1;};},elasticBoth:function(c,d){d=d||0.45;return this.combine(this.elasticIn(c,d),this.elasticOut(c,d));},combine:function(d,c){return function(e){if(e<0.5){return d(e*2)/2;}else{return c((e-0.5)*2)/2+0.5;}};}};}});(window.gloader||glow).module({name:"glow.anim",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.tweens","glow.events","glow.dom"]],builder:function(glow){var $=glow.dom.get,manager,events=glow.events,dom=glow.dom,get=dom.get,hasUnits=/width|height|top$|bottom$|left$|right$|spacing$|indent$|font-size/,noNegatives=/width|height|padding|opacity/,usesYAxis=/height|top/,getUnit=/(\D+)$/,testElement=dom.create('<div style="position:absolute;visibility:hidden"></div>');function addEventsFromOpts(instance,opts,eventProps){for(var i=0,len=eventProps.length;i<len;i++){if(opts[eventProps[i]]){events.addListener(instance,eventProps[i].slice(2).toLowerCase(),opts[eventProps[i]]);}}}(function(){var queue=[],queueLen=0,intervalTime=13,interval;manager={addToQueue:function(anim){queue[queueLen++]=anim;anim._playing=true;anim._timeAnchor=anim._timeAnchor||new Date().valueOf();if(!interval){this.startInterval();}},removeFromQueue:function(anim){for(var i=0;i<queueLen;i++){if(queue[i]==anim){queue.splice(i,1);anim._timeAnchor=null;anim._playing=false;if(--queueLen==0){this.stopInterval();}return;}}},startInterval:function(){interval=window.setInterval(this.processQueue,intervalTime);},stopInterval:function(){window.clearInterval(interval);interval=null;},processQueue:function(){var anim,i,now=new Date().valueOf();for(i=0;i<queueLen;i++){anim=queue[i];if(anim.position==anim.duration){manager.removeFromQueue(anim);i--;events.fire(anim,"complete");if(anim._opts.destroyOnComplete){anim.destroy();}continue;}if(anim.useSeconds){anim.position=(now-anim._timeAnchor)/1000;if(anim.position>anim.duration){anim.position=anim.duration;}}else{anim.position++;}anim.value=anim.tween(anim.position/anim.duration);events.fire(anim,"frame");}}};})();function convertCssUnit(element,fromValue,toUnit,axis){var elmStyle=testElement[0].style,axisProp=(axis=="x")?"width":"height",startPixelValue,toUnitPixelValue;elmStyle.margin=elmStyle.padding=elmStyle.border="0";startPixelValue=testElement.css(axisProp,fromValue).insertAfter(element)[axisProp]();toUnitPixelValue=testElement.css(axisProp,10+toUnit)[axisProp]()/10;testElement.remove();return startPixelValue/toUnitPixelValue;}function keepWithinRange(num,start,end){if(start!==undefined&&num<start){return start;}if(end!==undefined&&num>end){return end;}return num;}function buildAnimFunction(element,spec){var cssProp,r=["a=(function(){"],rLen=1,fromUnit,unitDefault=[0,"px"],to,from,unit,a;for(cssProp in spec){r[rLen++]='element.css("'+cssProp+'", ';if(typeof spec[cssProp]!="object"){to=spec[cssProp];}else{to=spec[cssProp].to;}if((from=spec[cssProp].from)===undefined){if(cssProp=="font-size"||cssProp=="background-position"){throw new Error("From value must be set for "+cssProp);}from=element.css(cssProp);}if(hasUnits.test(cssProp)){unit=(getUnit.exec(to)||unitDefault)[1];fromUnit=(getUnit.exec(from)||unitDefault)[1];from=parseFloat(from)||0;to=parseFloat(to)||0;if(from&&unit!=fromUnit){if(cssProp=="font-size"){throw new Error("Units must be the same for font-size");}from=convertCssUnit(element,from+fromUnit,unit,usesYAxis.test(cssProp)?"y":"x");}if(noNegatives.test(cssProp)){r[rLen++]="keepWithinRange(("+(to-from)+" * this.value) + "+from+', 0) + "'+unit+'"';}else{r[rLen++]="("+(to-from)+" * this.value) + "+from+' + "'+unit+'"';}}else{if(!(isNaN(from)||isNaN(to))){from=Number(from);to=Number(to);r[rLen++]="("+(to-from)+" * this.value) + "+from;}else{if(cssProp.indexOf("color")!=-1){to=dom.parseCssColor(to);if(!glow.lang.hasOwnProperty(from,"r")){from=dom.parseCssColor(from);}r[rLen++]='"rgb(" + keepWithinRange(Math.round('+(to.r-from.r)+" * this.value + "+from.r+'), 0, 255) + "," + keepWithinRange(Math.round('+(to.g-from.g)+" * this.value + "+from.g+'), 0, 255) + "," + keepWithinRange(Math.round('+(to.b-from.b)+" * this.value + "+from.b+'), 0, 255) + ")"';}else{if(cssProp=="background-position"){var vals={},fromTo=["from","to"],unit=(getUnit.exec(from)||unitDefault)[1];vals.fromOrig=from.toString().split(/\s/);vals.toOrig=to.toString().split(/\s/);if(vals.fromOrig[1]===undefined){vals.fromOrig[1]="50%";}if(vals.toOrig[1]===undefined){vals.toOrig[1]="50%";}for(var i=0;i<2;i++){vals[fromTo[i]+"X"]=parseFloat(vals[fromTo[i]+"Orig"][0]);vals[fromTo[i]+"Y"]=parseFloat(vals[fromTo[i]+"Orig"][1]);vals[fromTo[i]+"XUnit"]=(getUnit.exec(vals[fromTo[i]+"Orig"][0])||unitDefault)[1];vals[fromTo[i]+"YUnit"]=(getUnit.exec(vals[fromTo[i]+"Orig"][1])||unitDefault)[1];}if((vals.fromXUnit!==vals.toXUnit)||(vals.fromYUnit!==vals.toYUnit)){throw new Error("Mismatched axis units cannot be used for "+cssProp);}r[rLen++]="("+(vals.toX-vals.fromX)+" * this.value + "+vals.fromX+') + "'+vals.fromXUnit+' " + ('+(vals.toY-vals.fromY)+" * this.value + "+vals.fromY+') + "'+vals.fromYUnit+'"';}}}}r[rLen++]=");";}r[rLen++]="})";return eval(r.join(""));}var r={};r.css=function(element,duration,spec,opts){element=get(element);var anim=new r.Animation(duration,opts);if(element[0]){events.addListener(anim,"frame",buildAnimFunction(element,spec));}return anim;};slideElement=function slideElement(element,duration,action,opts){duration=duration||0.5;element=$(element);opts=glow.lang.apply({tween:glow.tweens.easeBoth(),onStart:function(){},onComplete:function(){}},opts);var i=0,thatlength=element.length,completeHeight,fromHeight,channels=[],timeline;for(;i<thatlength;i++){if(action=="up"||(action=="toggle"&&element.slice(i,i+1).height()>0)){element[i].style.overflow="hidden";if(glow.env.ie<8){element[i].style.zoom=1;}completeHeight=0;fromHeight=element.slice(i,i+1).height();}else{if(action=="down"||(action=="toggle"&&element.slice(i,i+1).height()==0)){fromHeight=element.slice(i,i+1).height();element[i].style.height="";completeHeight=element.slice(i,i+1).height();if(completeHeight===0){element[i].style.height="auto";completeHeight=element.slice(i,i+1).height();}element[i].style.height=fromHeight+"px";}}channels[i]=[glow.anim.css(element[i],duration,{height:{from:fromHeight,to:completeHeight}},{tween:opts.tween})];}timeline=new glow.anim.Timeline(channels,{destroyOnComplete:true});events.addListener(timeline,"complete",function(){element.each(function(){if(this.style.height.slice(0,1)!="0"){this.style.height="";if(glow.dom.get(this).height()===0){this.style.height="auto";}}});});events.addListener(timeline,"start",opts.onStart);events.addListener(timeline,"complete",opts.onComplete);return timeline.start();};r.slideDown=function(element,duration,opts){return slideElement(element,duration,"down",opts);};r.slideUp=function(element,duration,opts){return slideElement(element,duration,"up",opts);};r.slideToggle=function(element,duration,opts){return slideElement(element,duration,"toggle",opts);};r.fadeOut=function(element,duration,opts){return r.fadeTo(element,0,duration,opts);};r.fadeIn=function(element,duration,opts){return r.fadeTo(element,1,duration,opts);};r.fadeTo=function(element,opacity,duration,opts){duration=duration||0.5;element=$(element);opts=glow.lang.apply({tween:glow.tweens.easeBoth(),onStart:function(){},onComplete:function(){}},opts);var i=0,thatlength=element.length,channels=[],timeline;for(;i<thatlength;i++){channels[i]=[glow.anim.css(element[i],duration,{opacity:opacity},{tween:opts.tween})];}timeline=new glow.anim.Timeline(channels,{destroyOnComplete:true});events.addListener(timeline,"start",opts.onStart);events.addListener(timeline,"complete",opts.onComplete);return timeline.start();};r.highlight=function(element,highlightColour,duration,opts){element=$(element);duration=duration||1;highlightColour=highlightColour||"#ffff99";opts=glow.lang.apply({tween:glow.tweens.easeBoth(),onStart:function(){},onComplete:function(){}},opts);var i=0,transArray=[],elmsLength=element.length,completeColour,channels=[],timeline;for(;i<elmsLength;i++){completeColour=opts.completeColour||element.slice(i,i+1).css("background-color");if(completeColour=="transparent"||completeColour==""){completeColour="#fff";}channels[i]=[r.css(element[i],duration,{"background-color":{from:highlightColour,to:completeColour}},{tween:opts.tween})];}timeline=new glow.anim.Timeline(channels,{destroyOnComplete:true});events.addListener(timeline,"start",opts.onStart);events.addListener(timeline,"complete",opts.onComplete);return timeline.start();};var animationEventConstructorNames=["onStart","onStop","onComplete","onResume","onFrame"];r.Animation=function(duration,opts){this._opts=opts=glow.lang.apply({useSeconds:true,tween:glow.tweens.linear(),destroyOnComplete:false,onStart:null,onStop:null,onComplete:null,onResume:null,onFrame:null},opts);this._playing=false;this._timeAnchor=null;this.duration=duration;this.useSeconds=opts.useSeconds;this.tween=opts.tween;this.position=0;this.value=0;addEventsFromOpts(this,opts,animationEventConstructorNames);};r.Animation.prototype={start:function(){if(this._playing){this.stop();}var e=events.fire(this,"start");if(e.defaultPrevented()){return this;}this._timeAnchor=null;this.position=0;manager.addToQueue(this);return this;},stop:function(){if(this._playing){var e=events.fire(this,"stop");if(e.defaultPrevented()){return this;}manager.removeFromQueue(this);}return this;},destroy:function(){this.stop();events.removeAllListeners(this);return this;},resume:function(){if(!this._playing){var e=events.fire(this,"resume");if(e.defaultPrevented()){return this;}this._timeAnchor=new Date().valueOf()-(this.position*1000);manager.addToQueue(this);}return this;},isPlaying:function(){return this._playing;},goTo:function(pos){this._timeAnchor=new Date().valueOf()-((this.position=pos)*1000);this.value=this.tween(this.duration&&this.position/this.duration);events.fire(this,"frame");return this;}};var timelineEventConstructorNames=["onStart","onStop","onComplete","onResume"];r.Timeline=function(channels,opts){this._opts=opts=glow.lang.apply({loop:false,destroyOnComplete:false,onStart:null,onStop:null,onComplete:null,onResume:null},opts);this._channels=(channels[0]&&channels[0].push)?channels:[channels];this._channelPos=[];this._playing=false;this.loop=opts.loop;var i,j,iLen,jLen,channel,allChannels=this._channels,totalDuration=0,channelDuration;for(i=0,iLen=allChannels.length;i<iLen;i++){channel=allChannels[i];channelDuration=0;for(j=0,jLen=channel.length;j<jLen;j++){if(typeof channel[j]=="number"){channel[j]=new r.Animation(channel[j]);}if(channel[j] instanceof r.Animation){if(!channel[j].useSeconds){throw new Error("Timelined animations must be timed in seconds");}channel[j]._timelineOffset=channelDuration*1000;channelDuration+=channel[j].duration;channel[j]._channelIndex=i;}}this.duration=totalDuration=Math.max(channelDuration,totalDuration);}this._controlAnim=new r.Animation(totalDuration);events.addListener(this._controlAnim,"frame",this._processFrame,this);events.addListener(this._controlAnim,"complete",this._complete,this);addEventsFromOpts(this,opts,timelineEventConstructorNames);};r.Timeline.prototype={_advanceChannel:function(i){var currentAnim=this._channels[i][this._channelPos[i]],nextAnim=this._channels[i][++this._channelPos[i]];if(currentAnim&&currentAnim._playing){currentAnim._playing=false;events.fire(currentAnim,"complete");if(currentAnim._opts.destroyOnComplete){currentAnim.destroy();}}if((nextAnim)!==undefined){if(typeof nextAnim=="function"){nextAnim();this._advanceChannel(i);}else{nextAnim.position=0;nextAnim._channelIndex=i;events.fire(nextAnim,"start");nextAnim._playing=true;}}},_complete:function(){if(this.loop){this.start();return;}this._playing=false;events.fire(this,"complete");if(this._opts.destroyOnComplete){this.destroy();}},_processFrame:function(){var i,len,anim,controlAnim=this._controlAnim,msFromStart=(new Date().valueOf())-controlAnim._timeAnchor;for(i=0,len=this._channels.length;i<len;i++){if(!(anim=this._channels[i][this._channelPos[i]])){continue;}anim.position=(msFromStart-anim._timelineOffset)/1000;if(anim.position>anim.duration){anim.position=anim.duration;}anim.value=anim.tween(anim.position/anim.duration);events.fire(anim,"frame");if(anim.position==anim.duration){this._advanceChannel(i);}}},start:function(){var e=events.fire(this,"start");if(e.defaultPrevented()){return this;}var i,iLen,j,jLen,anim;this._playing=true;for(i=0,iLen=this._channels.length;i<iLen;i++){this._channelPos[i]=-1;this._advanceChannel(i);for(j=this._channels[i].length;j;j--){anim=this._channels[i][j];if(anim instanceof r.Animation){anim.goTo(0);}}}this._controlAnim.start();return this;},stop:function(){if(this._playing){var e=events.fire(this,"stop");if(e.defaultPrevented()){return this;}this._playing=false;var anim;for(var i=0,len=this._channels.length;i<len;i++){anim=this._channels[i][this._channelPos[i]];if(anim instanceof r.Animation&&anim._playing){events.fire(anim,"stop");anim._playing=false;}}this._controlAnim.stop();}return this;},destroy:function(){var i,j;this.stop();events.removeAllListeners(this);this._controlAnim.destroy();i=this._channels.length;while(i--){j=this._channels[i].length;while(j--){if(this._channels[i][j].destroy){this._channels[i][j].destroy();}}}return this;},resume:function(){if(!this._playing){var e=events.fire(this,"resume");if(e.defaultPrevented()){return this;}this._playing=true;var anim;for(var i=0,len=this._channels.length;i<len;i++){anim=this._channels[i][this._channelPos[i]];if(anim instanceof r.Animation&&!anim._playing){events.fire(anim,"resume");anim._playing=true;}}this._controlAnim.resume();}return this;},isPlaying:function(){return this._playing;},goTo:function(pos){var i,j,k,channelsLen=this._channels.length,channelLen,anim,runningDuration;if(typeof pos=="number"){if(pos>this.duration){if(this.loop){pos=pos%this.duration;}else{pos=this.duration;}}this._controlAnim.goTo(pos);for(i=0;i<channelsLen;i++){runningDuration=0;for(j=0,channelLen=this._channels[i].length;j<channelLen;j++){anim=this._channels[i][j];if(anim instanceof r.Animation){if((runningDuration+anim.duration)>pos){this._channelPos[i]=j;anim.goTo(pos-runningDuration);break;}anim.goTo(anim.duration);runningDuration+=anim.duration;}}for(k=channelLen;k>j;k--){anim.goTo(0);}}}else{for(i=0;i<channelsLen;i++){runningDuration=0;for(j=0,channelLen=this._channels[i].length;j<channelLen;j++){anim=this._channels[i][j];if(anim===pos){return this.goTo(runningDuration);}if(anim instanceof r.Animation){runningDuration+=anim.duration;}}}throw"Animation not found in animation channels";}return this;}};glow.anim=r;}});(window.gloader||glow).module({name:"glow.forms",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.dom","glow.events","glow.anim","glow.net","glow.i18n"]],builder:function(g){var b=g.i18n,f=g.lang.interpolate;b.addLocaleModule("GLOW_FORMS","en",{TEST_MESSAGE_REQUIRED:"Value is required",TEST_MESSAGE_IS_NUMBER:"Must be a number.",TEST_MESSAGE_MIN:"The value must be at least {arg}.",TEST_MESSAGE_MAX:"The value must be less than {arg}.",TEST_MESSAGE_RANGE:"The value must be {min} or greater, and less than {max}.",TEST_MESSAGE_MIN_COUNT:"Must be have at least {arg} values.",TEST_MESSAGE_MAX_COUNT:"Must be have at most {arg} values.",TEST_MESSAGE_COUNT:"Must have {arg} values.",TEST_MESSAGE_REGEX:"Must be in the correct format.",TEST_MESSAGE_MIN_LEN:"Must be at least {arg} characters.",TEST_MESSAGE_MAX_LEN:"Must be at most {arg} characters.",TEST_MESSAGE_IS_EMAIL:"Must be a valid email address.",TEST_MESSAGE_SAME_AS:"Must be the same as: {arg}",TEST_MESSAGE_AJAX:"server responded",TEST_MESSAGE_IS:"Must be {arg}",TEST_MESSAGE_IS_NOT:"Must not be {arg}"});g.forms={};g.forms.Form=function(k,j){this.formNode=g.dom.get(k);if(!this.formNode[0]){throw"Could not find form. Possibly run before DOM ready.";}this._fields=[];this._result=null;this.opts=j||{};g.events.addListener(this,"validate",this.opts.onValidate||a.defaultFeedback);this._idleTimer=null;this._localeModule=b.getLocaleModule("GLOW_FORMS");var h=this;g.events.addListener(this.formNode,"submit",function(){h.validate("submit");return false;});};g.forms.Form.prototype.validate=function(h,j){this.eventName=h||"submit";this._result=new g.forms.ValidateResult(this.eventName);this._result.form=this;this._fieldCur=0;this._testCur=-1;this._fieldName=j;c.call(this);};var c=function(){this._testCur++;if(this._testCur>=this._fields[this._fieldCur]._tests.length){if(!d.call(this)){return;}}var j=this._fields[this._fieldCur]._tests[this._testCur];var h;if(j.opts.field){h=this.formNode.val()[j.opts.field]||"";j.isConditional=true;}else{h=this.formNode.val()[this._fields[this._fieldCur].name]||"";}if(!h.join){h=[h];}var k=function(l){return function(){e.apply(l,arguments);};}(this);j.opts.on=j.opts.on||"submit";if(this._result.eventName&&(" "+j.opts.on+" ").indexOf(" "+this._result.eventName+" ")!=-1){if(this._fieldName&&this._fieldName!=j.name){c.call(this);return;}if(typeof g.forms.tests[j.type]!="function"){throw"Unimplemented test: no test exists of type '"+j.type+"'.";}j.opts._localeModule=this._localeModule;g.forms.tests[j.type](h,j.opts,k,this.formNode.val());}else{c.call(this);}};var d=function(){this._fieldCur++;this._testCur=0;if(this._fieldCur>=this._fields.length){this._fieldCur=0;g.events.fire(this,"validate",this._result);if(this.eventName=="submit"&&this._result&&!this._result.defaultPrevented()){try{this.formNode[0].submit();}catch(h){throw new Error("Glow can't submit the form because the submit function can't be called. Perhaps that form's submit was replaced by an input element named 'submit'?");}}return false;}return true;};var e=function(h,j){if(typeof h=="boolean"){h=(h)?g.forms.PASS:g.forms.FAIL;}if(this._fields[this._fieldCur]._tests[this._testCur].isConditional&&h===g.forms.FAIL){h=g.forms.SKIP;}this._result.fields.push({name:this._fields[this._fieldCur].name,result:h,message:j});if(h!==g.forms.PASS){if(h===g.forms.FAIL){this._result.errorCount++;}this._testCur=this._fields[this._fieldCur]._tests.length;}c.call(this);};g.forms.Form.prototype.addTests=function(r){var q={name:r,_tests:[]};var j=function(s){return function(){s.validate.apply(s,["change",r]);};}(this);var n=function(s){return function(){s.validate.apply(s,["click",r]);};}(this);var o=function(s){return function(){s.validate.apply(s,["idle",r]);};}(this);for(var l=1;l<arguments.length;l++){var k=arguments[l][0];var p=(arguments[l].length>1)?arguments[l][1]:{};q._tests.push({name:r,type:k,opts:p});if(!j.added&&(" "+p.on+" ").indexOf(" change ")!=-1){var m=this.formNode.get("*").each(function(s){if(this.name==r){g.events.addListener(this,"change",j);j.added=true;}});}if(!n.added&&(" "+p.on+" ").indexOf(" click ")!=-1){var m=this.formNode.get("*").each(function(s){if(this.name==r){g.events.addListener(this,"click",n);n.added=true;}});}if(!o.added&&(" "+p.on+" ").indexOf(" idle ")!=-1){var h=(typeof p.delay!="undefined")?parseInt(p.delay):1000;var m=this.formNode.get("*").each(function(s){if(this.name==r){g.events.addListener(this,"keyup",function(u){return function(){window.clearTimeout(this.idleTimeoutID);if(this.value){this.idleTimeoutID=window.setTimeout(o,u);}};}(h));g.events.addListener(this,"blur",function(){window.clearTimeout(this.idleTimeoutID);});o.added=true;}});}}this._fields.push(q);return this;};g.forms.ValidateResult=function(h){g.events.Event.apply(this);this.eventName=h;this.errorCount=0;this.value=undefined;this.fields=[];};g.lang.extend(g.forms.ValidateResult,g.events.Event);g.forms.PASS=1;g.forms.FAIL=0;g.forms.SKIP=-1;g.forms.tests={required:function(j,m,n){var l=m.message||m._localeModule.TEST_MESSAGE_REQUIRED;for(var k=0,h=j.length;k<h;k++){if(/^\s*$/.test(j[k])){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},isNumber:function(j,m,n){var l=m.message||m._localeModule.TEST_MESSAGE_IS_NUMBER;for(var k=0,h=j.length;k<h;k++){if(j[k]==""||isNaN(j[k])){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},min:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MIN,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(Number(j[k])<Number(m.arg)){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},max:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MAX,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(Number(j[k])>Number(m.arg)){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},range:function(k,o,p){var n=o.arg.split("..");if(typeof n[0]=="undefined"||typeof n[1]=="undefined"){throw"Range test requires a parameter like 0..10.";}var m=o.message||f(o._localeModule.TEST_MESSAGE_RANGE,{min:n[0],max:n[1]});n[0]*=1;n[1]*=1;if(n[0]>n[1]){var j=n[0];n[0]=n[1];n[1]=j;}for(var l=0,h=k.length;l<h;l++){if(k[l]<n[0]||k[l]>n[1]){p(g.forms.FAIL,m);return;}}p(g.forms.PASS,m);},minCount:function(h,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MIN_COUNT,{arg:m.arg});var k=0;for(var j=0;j<h.length;j++){if(h[j]!=""){k++;}}if(k<m.arg){n(g.forms.FAIL,l);return;}n(g.forms.PASS,l);},maxCount:function(h,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MAX_COUNT,{arg:m.arg});var k=0;for(var j=0;j<h.length;j++){if(h[j]!=""){k++;}}if(k>m.arg){n(g.forms.FAIL,l);return;}n(g.forms.PASS,l);},count:function(h,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_COUNT,{arg:m.arg});var k=0;for(var j=0;j<h.length;j++){if(h[j]!=""){k++;}}if(k!=m.arg){n(g.forms.FAIL,l);return;}n(g.forms.PASS,l);},regex:function(j,n,o){var m=n.message||n._localeModule.TEST_MESSAGE_REGEX;var l=(typeof n.arg=="string")?new RegExp(n.arg):n.arg;for(var k=0,h=j.length;k<h;k++){if(!l.test(j[k])){o(g.forms.FAIL,m);return;}}o(g.forms.PASS,m);},minLen:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MIN_LEN,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(j[k].length<m.arg){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},maxLen:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MAX_LEN,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(j[k].length>m.arg){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},isEmail:function(j,m,n){var l=m.message||m._localeModule.TEST_MESSAGE_IS_EMAIL;for(var k=0,h=j.length;k<h;k++){if(!/^\s*[a-z0-9._%+-]+@[a-z0-9.-]+\.[a-z]{2,}\s*$/i.test(j[k])){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},sameAs:function(j,n,p,o){var m=n.message||f(n._localeModule.TEST_MESSAGE_SAME_AS,{arg:n.arg});var k=o[n.arg];for(var l=0,h=j.length;l<h;l++){if(j[l]!=k){p(g.forms.FAIL,m);return;}}p(g.forms.PASS,m);},ajax:function(o,h,q,n){var m={},r=(h.message||h._localeModule.TEST_MESSAGE_AJAX);for(var k in n){if(typeof n[k]=="string"){m[k]=escape(n[k]);}else{if(typeof n[k].push!="undefined"){m[k]=g.lang.map(n[k],function(p){return escape(p);}).join(",");}}}var j=g.lang.interpolate(h.url,m);var l=g.net.get(j,{onLoad:function(p){var s=h.arg(p);if(typeof s.push=="undefined"){s=[s,r];}q(s[0],s[1]);},onError:function(p){alert("Error getting file: "+j);}});},custom:function(h,j,k){j.arg.apply(this,arguments);},is:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_IS,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(j[k]!=m.arg){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},isNot:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_IS_NOT,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(j[k]==m.arg){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);}};var a=g.forms.feedback={};a.defaultFeedback=(function(){var h;function l(){if(!h){h=g.dom.create('<input type="hidden" value="0" name="1.7.3" id="1.7.3" />').appendTo(document.body);}h[0].value++;}function k(o){var n=o.fields,r,q,s,p,m;for(p=0,m=n.length;p<m;p++){r=g.dom.get(o.form.formNode[0].elements[n[p].name]);q=g.dom.get("."+n[p].name.replace(/(\W)/g,"\\$1")+"-msgContainer");if(!q[0]&&r.length==1){q=o.form.formNode.get("label").filter(function(){return this.htmlFor==r[0].id;});}s=q.get("span.glow-errorMsg");if(n[p].result){s.remove();r.removeClass("glow-invalid");}else{if(q.length){if(!s[0]){q.append((s=g.dom.create('<span class="glow-errorMsg"></span>')));}s.text(n[p].message);r.addClass("glow-invalid");}}}}function j(p){var s=p.fields,v,n,o,r,m,q,t;p.form.formNode.get("div.glow-errorSummary").remove();n=g.dom.create('<div class="glow-errorSummary" tabindex="-1"><ul></ul></div>');o=n.get("ul");for(q=0,t=s.length;q<t;q++){v=g.dom.get(p.form.formNode[0].elements[s[q].name]);r=g.dom.get("."+s[q].name.replace(/(\W)/g,"\\$1")+"-prompt");if(!r[0]&&v.length==1){r=p.form.formNode.get("label").filter(function(){return this.htmlFor==v[0].id;});}if(r[0]){r.get("span.glow-errorMsg").remove();m=g.lang.trim(r.text());if(m.slice(-1)==":"){m=m.slice(0,-1);}}else{m=s[q].name.replace(/^\w/,function(w){return w.toUpperCase();});}if(!s[q].result){o.append(g.dom.create("<li></li>").text(m+": "+s[q].message));}}p.form.formNode.prepend(n.css("opacity","0"));g.anim.css(n,"0.5",{opacity:{from:0,to:1}},{tween:g.tweens.easeOut()}).start();try{n[0].focus();}catch(u){}l();}return function(m){if(m.eventName=="submit"){if(!m.errorCount){m.form.formNode.get("div.glow-errorSummary").remove();return;}j(m);}setTimeout(function(){k(m);},0);return false;};}());}});(window.gloader||glow).module({name:"glow.embed",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.dom","glow.data","glow.i18n"]],builder:function(n){var o=n.i18n;o.addLocaleModule("GLOW_EMBED","en",{FLASH_MESSAGE:"This content requires Flash Player version {min} (installed version: {installed})",NO_PLAYER_MESSAGE:"No Flash Flayer installed, or version is pre 6.0.0"});function b(s){var r="";for(var t in s){if(t.toLowerCase()=="flashvars"&&typeof s[t]=="object"){r+=' FlashVars="'+n.data.encodeUrl(s[t])+'"';}else{r+=" "+t+'="'+s[t]+'"';}}return r;}function q(s){var u="",t,v;for(t in s){if(t.toLowerCase()=="flashvars"&&typeof s[t]=="object"){v=n.data.encodeUrl(s[t]);}else{v=s[t];}u+='<param name="'+t+'" value="'+v+'" />\n';}return u;}function h(s,r){s=s||{};for(var t in r){if(typeof s[t]=="undefined"){s[t]=r[t];}else{if(typeof r[t]=="object"){s[t]=h(s[t],r[t]);}}}return s;}function c(){var r=(navigator.platform||navigator.userAgent);return r.match(/win/i)?"win":r.match(/mac/i)?"mac":"other";}function l(r){var t=/^WIN (\d+),(\d+),(\d+),\d+$/;var s=r.GetVariable("$version");if($match=t.exec(s)){return{major:parseInt($match[1]),minor:parseInt($match[2]),release:parseInt($match[3]),actual:s};}else{}}function k(){var s,u,v={major:0,minor:0,release:0},r=v;if(n.env.ie){try{u=new ActiveXObject("ShockwaveFlash.ShockwaveFlash.7");r=l(u);}catch(w){try{u=new ActiveXObject("ShockwaveFlash.ShockwaveFlash.6");try{u.AllowScriptAccess="always";r=l(u);}catch(w){r={major:6,minor:0,release:29};}}catch(w){}}}else{var t=/^Shockwave Flash\s*(\d+)\.(\d+)\s*\w(\d+)$/;if((u=navigator.plugins["Shockwave Flash"])&&(s=t.exec(u.description))){r={major:parseInt(s[1]),minor:parseInt(s[2]),release:parseInt(s[3]),actual:u.description};}}r.toString=function(){return this.major?[this.major,this.minor,this.release].join("."):o.getLocaleModule("GLOW_EMBED").NO_PLAYER_MESSAGE;};return r;}var m=k();function f(t){if(typeof t!="object"){var s=String(t).match(/^(\d+)(?:\.(\d+)(?:\.(\d+))?)?$/);if(!s){throw new Error('glow.embed._meetsVersionRequirements: invalid format for version string, require "n.n.n" or "n.n" or simply "n" where n is a numeric value');}t={major:parseInt(s[1],10),minor:parseInt(s[2]||0,10),release:parseInt(s[3]||0,10)};}var r=m,u=t;return(r.major>u.major||(r.major==u.major&&r.minor>u.minor)||(r.major==u.major&&r.minor==u.minor&&r.release>=u.release));}var e=n.env.ie?j:g;function g(t,r,s){return'<embed type="application/x-shockwave-flash" src="'+t+'"'+b(r)+b(s)+"></embed>";}function j(t,r,s){return'<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" '+b(r)+'><param name="movie" value="'+t+'" />'+q(s)+"</object>";}var a={},p=0;function d(){return n.UID+"FlashEmbed"+(p++);}a.Flash=function(x,r,w,v){v=h(v,{width:"100%",height:"100%",params:{allowscriptaccess:"always",allowfullscreen:"true",quality:"high"},attributes:{},message:n.lang.interpolate(o.getLocaleModule("GLOW_EMBED").FLASH_MESSAGE,{min:w,installed:m}),id:(v&&v.attributes&&v.attributes.id)||d()});r=n.dom.get(r);if(!r.length){throw new Error("glow.embed.Flash unable to locate container");}this.container=r;this.movie=null;this._displayErrorMessage=typeof v.message=="function"?v.message:function(){return v.message;};this.isSupported;if(this.isSupported=f(w)){var s=v.attributes,u=["id","width","height"],t=u.length;while(t--){if(v[u[t]]){s[u[t]]=v[u[t]];}}if(v.className){s["class"]=v.className;}this._embed_tag=e(x,s,v.params);}};a.Flash.version=function(){return m;};a.Flash.prototype.embed=function(){var s=this.container[0];if(this.isSupported){s.innerHTML=this._embed_tag;this.movie=s.firstChild;}else{var r=this._displayErrorMessage();if(r){s.innerHTML=r;}}return this;};n.embed=a;}});(window.gloader||glow).module({name:"glow.dragdrop",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.tweens","glow.events","glow.dom","glow.anim"]],builder:function(j){var c=j.events,k=c.addListener,v=c.fire,g=c.removeListener,q=j.dom,h=q.get,m=q.create;var n={},w=1000,a=(document.compatMode=="CSS1Compat"&&j.env.ie>=5)?true:false,x=(document.compatMode!="CSS1Compat"&&j.env.ie>=5)?true:false,d=j.env.ie>=5,s=["top","right","bottom","left"];function b(r,z){var A=r.prototype[z];var y="cached_"+z;r.prototype[z]=function(){if(y in this){return this[y];}return this[y]=A.apply(this,arguments);};}function u(A,z){var r=s.length,y;while(r--){y="margin-"+s[r];A.css(y,z.css(y));}}function f(r,y){var A=r.prototype[y];var z="cached_"+y;r.prototype[y]=function(B){if(!this[z]){this[z]={};}if(B in this[z]){return this[z][B];}return this[z][B]=A.apply(this,arguments);};}function t(A,z){for(var y=0,r=z.length;y<r;y++){delete A["cached_"+z[y]];}}var o=function(r){this.el=r;};o.prototype={val:function(r){var y=parseInt(this.el.css(r));return y||0;},width:function(){return this.borderWidth()-this.val("border-left-width")-this.val("border-right-width");},height:function(){return this.borderHeight()-this.val("border-top-width")-this.val("border-bottom-width");},offsetParentPageTop:function(){var r=this.el[0],z,y;while(r=r.offsetParent){if(h(r).css("position")!="static"){break;}}return r?h(r).offset().top:0;},offsetTop:function(){return this.el.position().top;},offsetLeft:function(){return this.el.position().left;},borderWidth:function(){var r=this.el[0].offsetWidth;if(j.env.khtml){r-=this.val("margin-left")+this.val("margin-right")+this.val("border-left-width")+this.val("border-right-width");}return r;},borderHeight:function(){if(this._logicalBottom){return this._logicalBottom-this.offsetTop();}var r=this.el[0].offsetHeight;if(j.env.khtml){r-=this.val("margin-top")+this.val("margin-bottom")+this.val("border-top-width")+this.val("border-bottom-width");}return r;},outerWidth:function(){return this.borderWidth()+this.val("margin-left")+this.val("margin-right");},outerHeight:function(){return this.borderHeight()+this.val("margin-top")+this.val("margin-bottom");},innerLeftPos:function(){return this.offsetLeft()+this.val("margin-left")+this.val("border-left-width")+this.val("padding-left");},innerTopPos:function(){return this.offsetTop()+this.val("margin-top")+this.val("border-top-width")+this.val("padding-top");},surroundWidth:function(){return this.val("border-left-width")+this.val("padding-left")+this.val("padding-right")+this.val("border-right-width");},surroundHeight:function(){return this.val("border-top-width")+this.val("padding-top")+this.val("padding-bottom")+this.val("border-bottom-width");},verticalCenter:function(){return this.offsetTop()+(this.outerHeight()/2);},horizontalCenter:function(){return this.offsetTop()+(this.outerWidth()/2);}};for(var p in o.prototype){if(p=="val"){f(o,p);}else{b(o,p);}}j.lang.apply(o.prototype,{resetPosition:function(){t(this,["offsetTop","offsetLeft","borderTopPos","borderLeftPos","innerTopPos","innerLeftPos","verticalCenter","horizontalCenter"]);},setLogicalBottom:function(r){this._logicalBottom=r;},boundsFor:function(z){var y,r,A=this.el.css("position");if(A!="static"){y=r=0;}else{y=this.innerTopPos();r=this.innerLeftPos();}return[y,r+this.width()-z.outerWidth(),y+this.height()-z.outerHeight(),r];},outerBounds:function(){var z=this.el.offset(),y=z.left,r=z.top;return[r,y+this.borderWidth(),r+this.borderHeight(),y];},intersectSize:function(z,A){var y=this.outerBounds(),r=z.outerBounds();if(A){y[1]++;r[1]++;y[2]++;r[2]++;}return(y[2]<r[0]?0:r[2]<y[0]?0:y[0]<r[0]?(y[2]<r[2]?y[2]-r[0]:r[2]-r[0]):r[2]<y[2]?r[2]-y[0]:y[2]-y[0])*(y[1]<r[3]?0:r[1]<y[3]?0:y[3]<r[3]?(y[1]<r[1]?y[1]-r[3]:r[1]-r[3]):r[1]<y[1]?r[1]-y[3]:y[1]-y[3]);},sizePlaceholder:function(C,D,A,y){var B=new o(C),z=this.el,r=D||z.css("position");C.css("display","none");z.after(C);C.css("width",(z[0].offsetWidth-B.surroundWidth())+"px").css("height",(z[0].offsetHeight-B.surroundHeight())+"px");u(C,z);C.remove();C.css("display","block");if(r!="static"){C.css("left",A+"px");C.css("top",y+"px");}C.css("position",r);},contains:function(z){var y=this.boundsFor(z),r=z.el.position(),B=r.top,A=r.left;return B>=y[0]&&A<=y[1]&&B<=y[2]&&A>=y[3];},containsPoint:function(y){var r=this.el.offset();return y.x>=r.left&&y.y>=r.top&&y.x<=r.left+this.borderWidth()&&y.y<=r.top+this.borderHeight();},positionedAncestorBox:function(){var r=this.el.parent(),y;while(r[0]){y=r.css("position")||"static";if(y=="relative"||y=="absolute"||y=="fixed"){return new o(r);}r=r.parent();}return null;}});function e(y){var r=y[0].tagName.toLowerCase()=="li"?"li":"div";var z=m("<"+r+"></"+r+">");if(r=="li"){z.css("list-style-type","none");}return z;}n.Draggable=function(A,B){this.element=h(A);this._opts=B=j.lang.apply({dragPrevention:["input","textarea","button","select","option","a"],placeholder:"spacer",placeholderClass:"glow-dragdrop-placeholder",step:{x:1,y:1}},B||{});if(typeof B.step=="number"){B.step={x:B.step,y:B.step};}else{B.step.x=B.step.x||1;B.step.y=B.step.y||1;}this._preventDrag=[];for(var y=0,r=B.dragPrevention.length;y<r;y++){this._preventDrag[y]=B.dragPrevention[y].toLowerCase();}if(B.container){this.container=h(B.container);}this._handle=B.handle&&this.element.get(B.handle)||this.element;if(B.dropTargets){this.dropTargets=h(B.dropTargets);}var z=this._listeners=[],y=0;if(B.onDrag){z[y++]=k(this,"drag",this._opts.onDrag,this);}if(B.onEnter){z[y++]=k(this,"enter",this._opts.onEnter,this);}if(B.onLeave){z[y++]=k(this,"leave",this._opts.onLeave,this);}if(B.onDrop){z[y++]=k(this,"drop",this._opts.onDrop,this);}if(B.onAfterDrop){z[y++]=k(this,"afterDrop",this._opts.onAfterDrop,this);}this._dragListener=k(this._handle,"mousedown",this._startDragMouse,this);return;};n.Draggable.prototype={_createPlaceholder:function(){var r=this.element,z,y=this._box;if(this._opts.placeholder=="clone"){z=r.clone();}else{z=e(r);}if(this._opts.placeholderClass){z.addClass(this._opts.placeholderClass);}y.sizePlaceholder(z,null,this._startLeft,this._startTop);r.after(z);this._placeholder=z;},_removePlaceholder:function(){this._placeholder.remove();},_resetPosition:function(){var G=this._preDragPosition,r=this.element,z=this._box,B=this._startOffset,D=r.css("position"),F,C;z.resetPosition();var A=z.el.position(),y={x:A.left,y:A.top};if(this._placeholder||this._dropIndicator){r.remove();}if(G=="static"&&y.y==B.y&&y.x==B.x){r.css("position","static");r.css("left","");r.css("top","");}else{r.css("z-index",this._preDragZIndex);r.css("position",G=="static"?"relative":G);if(G=="static"){F=y.x-B.x;C=y.y-B.y;}else{if(G=="relative"&&D!="relative"){F=this._startLeft+(y.x-B.x);C=this._startTop+(y.y-B.y);}}if(D!=G){r.css("left",F?F+"px":"");r.css("top",C?C+"px":"");}}if(this._dropIndicator){var E=this._dropIndicator.parent()[0];if(E){E.replaceChild(r[0],this._dropIndicator[0]);}delete this._dropIndicator;if(this._placeholder){this._placeholder.remove();delete this._placeholder;}r.css("position",G);if(G=="relative"&&D!="relative"){r.css("left",this._startLeft);r.css("top",this._startTop);}}else{if(this._placeholder){var E=this._placeholder.parent()[0];if(E){E.replaceChild(r[0],this._placeholder[0]);}delete this._placeholder;}}},_startDragMouse:function(I){var D=this._preventDrag,y=I.source,M=y.tagName.toLowerCase();for(var F=0,E=D.length;F<E;F++){if(D[F]==M){return;}}if(v(this,"drag").defaultPrevented()){return;}if(this._dragging==1){return this.endDrag();}else{if(this._dragging){return;}}this._dragging=1;var B=this.element,A=this.container,r=this._opts,H=this._box=new o(B),C=r.step;this._preDragPosition=B.css("position");var G=H.el.position(),J=this._startOffset={x:G.left,y:G.top};if(A){this._containerBox=new o(A);this._bounds=this._containerBox.boundsFor(H);if(C.x!=1){this._bounds[3]-=(this._bounds[3]-J.x)%C.x;this._bounds[1]-=(this._bounds[1]-J.x)%C.x;}if(C.y!=1){this._bounds[0]-=(this._bounds[0]-J.y)%C.y;this._bounds[2]-=(this._bounds[2]-J.y)%C.y;}}else{delete this._bounds;}this._mouseStart={x:I.pageX,y:I.pageY};this._preDragStyle=B.attr("style");this._preDragZIndex=B.css("z-index");B.css("z-index",w++);this._startLeft=B[0].style.left?parseInt(B[0].style.left):0;this._startTop=B[0].style.top?parseInt(B[0].style.top):0;if(r.placeholder&&r.placeholder!="none"){this._createPlaceholder();}B.css("position","absolute");B.css("left",J.x+"px");B.css("top",J.y+"px");if(a){this._scrollY=document.documentElement.scrollTop;this._innerHeight=document.documentElement.clientHeight;}else{if(x){this._scrollY=document.body.scrollTop;this._innerHeight=document.body.clientHeight;}else{this._scrollY=window.scrollY;this._innerHeight=window.innerHeight;}}var K=function(){return false;},L=document.documentElement;if(this.dropTargets){var z=new c.Event();z.draggable=this;for(var F=0,E=this.dropTargets.length;F<E;F++){v(this.dropTargets[F],"active",z);}this._mousePos={x:I.pageX,y:I.pageY};this._testForDropTargets();}this._dragListeners=[k(L,"selectstart",K),k(L,"dragstart",K),k(L,"mousedown",K),k(L,"mousemove",this._dragMouse,this),k(L,"mouseup",this._releaseElement,this)];return false;},_dragMouse:function(C){var r=this.element,y=this._opts.axis,D=y=="y"?this._startOffset.x:(this._startOffset.x+C.pageX-this._mouseStart.x),B=y=="x"?this._startOffset.y:(this._startOffset.y+C.pageY-this._mouseStart.y),A=this._bounds,z=this._opts.step;if(z.x!=1){D=Math.round((D-this._startOffset.x)/z.x)*z.x+this._startOffset.x;}if(z.y!=1){B=Math.round((B-this._startOffset.y)/z.y)*z.y+this._startOffset.y;}if(A){if(y!="y"){D=D<A[3]?A[3]:D>A[1]?A[1]:D;}if(y!="x"){B=B<A[0]?A[0]:B>A[2]?A[2]:B;}}r[0].style.left=D+"px";r[0].style.top=B+"px";if(this.dropTargets){this._mousePos={x:C.pageX,y:C.pageY};}if(d&&C.nativeEvent.button==0){this._releaseElement(C);return false;}return false;},_testForDropTargets:function(H){if(!this._lock){this._lock=0;}if(H){this._lock--;}else{if(this.lock){return;}}if(this._dragging!=1){return;}var z=this.activeTarget,y,R=this.dropTargets,S,C,F=this._box,K=this._mousePos;F.resetPosition();var D=0;for(var L=0,J=R.length;L<J;L++){S=R[L];C=S._box;if(S._opts.tolerance=="contained"){if(C.contains(F)){y=S;break;}}else{if(S._opts.tolerance=="cursor"){if(C.containsPoint(K)){y=S;break;}}else{var M=C.intersectSize(F,true);if(M>D){D=M;y=S;}}}}this.activeTarget=y;if(y!==z){if(y){var I=new c.Event();I.draggable=this;v(y,"enter",I);var r=new c.Event();r.dropTarget=y;v(this,"enter",r);}if(z){var E=new c.Event();E.draggable=this;v(z,"leave",E);var G=new c.Event();G.dropTarget=z;v(this,"leave",G);}}if(y&&y._opts.dropIndicator!="none"){var P,N=y._childBoxes,A=y._children;F.resetPosition();var Q=y._box.innerTopPos();var T=K.y-F.offsetParentPageTop();var O=0;for(var L=0,J=N.length;L<J;L++){if(A[L]==this.element[0]){continue;}P=N[L];Q+=P.outerHeight();if(T<=Q){if(y._dropIndicatorAt!=L){h(P.el).before(y._dropIndicator);y._dropIndicatorAt=L;}O=1;break;}}if(!O){if(P){h(P.el).after(y._dropIndicator);y._dropIndicatorAt=L+1;}else{y.element.append(y._dropIndicator);y._dropIndicatorAt=0;}}}this._lock++;var B=this;setTimeout(function(){B._testForDropTargets(1);},100);},_releaseElement:function(){if(this._dragging!=1){return;}this._dragging=2;var z,r;var D=this.dropTargets,C=this.activeTarget;if(D){for(z=0,r=D.length;z<r;z++){var B=new c.Event();B.draggable=this;B.droppedOnThis=C&&C==D[z];v(D[z],"inactive",B);}}if(C){var B=new c.Event();B.draggable=this;v(C,"drop",B);}var y=this._dragListeners;for(z=0,r=y.length;z<r;z++){c.removeListener(y[z]);}var A=v(this,"drop");if(!A.defaultPrevented()&&this.dropTargets){this.returnHome();}else{this.endDrag();}},endDrag:function(){if(this._dragging!=2){return;}this._dragging=0;if(this._reset){this._reset();delete this._reset;}if(this.placeholder){this.placeholder.remove();}this._resetPosition();delete this.activeTarget;v(this,"afterDrop");},returnHome:function(G){var z=(G)?G:j.tweens.linear(),C,r,A=this.element,E=this._box.el.position(),y=Math.pow(Math.pow(this._startOffset.x-E.left,2)+Math.pow(this._startOffset.y-E.top,2),0.5),B=0.3+(y/1000);var D=[[j.anim.css(A,B,{left:this._startOffset.x,top:this._startOffset.y},{tween:z})]];if(this._dropIndicator){D.push([j.anim.css(this._dropIndicator,B-0.1,{opacity:{to:0}})]);}var F=new j.anim.Timeline(D);k(F,"complete",function(){this.endDrag();},this);F.start();return;}};var l=0;n.DropTarget=function(r,y){r=this.element=h(r);if(!r.length){throw"no element passed into DropTarget constuctor";}if(r.length>1){throw"more than one element passed into DropTarget constructor";}this._id=++l;this._opts=y=j.lang.apply({dropIndicator:"none",dropIndicatorClass:"glow-dragdrop-dropindicator",tolerance:"intersect"},y||{});if(y.onActive){k(this,"active",y.onActive);}if(y.onInactive){k(this,"inactive",y.onInactive);}if(y.onEnter){k(this,"enter",y.onEnter);}if(y.onLeave){k(this,"leave",y.onLeave);}if(y.onDrop){k(this,"drop",y.onDrop);}k(this,"active",this._onActive);k(this,"inactive",this._onInactive);return this;};n.DropTarget.prototype={setLogicalBottom:function(r){this._logicalBottom=r;},_onActive:function(A){var y=A.draggable;this._box=new o(this.element);if(this._logicalBottom){this._box.setLogicalBottom(this._logicalBottom);}if(this._opts.dropIndicator=="none"){return;}this._onEnterListener=k(this,"enter",this._onEnter);this._onLeaveListener=k(this,"leave",this._onLeave);this._dropIndicator=e(y.element);if(this._opts.dropIndicatorClass){this._dropIndicator.addClass(this._opts.dropIndicatorClass);}y._box.sizePlaceholder(this._dropIndicator,"relative",0,0);var z=this._children=h(this.element.children()).filter(function(){var B=h(this);return(!A.draggable._placeholder||!B.eq(A.draggable._placeholder))&&(!this._dropIndicator||!B.eq(this._dropIndicator));});var r=this._childBoxes=[];z.each(function(B){r[B]=new o(h(z[B]));});},_onInactive:function(r){g(this._onEnterListener);g(this._onLeaveListener);delete this._box;if(this._opts.dropIndicator=="none"){return;}if(!r.droppedOnThis&&this._dropIndicator){this._dropIndicator.remove();delete this._dropIndicator;}delete this._childBoxes;delete this._children;},_onEnter:function(){this._dropIndicatorAt=-1;},_onLeave:function(){this._dropIndicator.remove();},moveToPosition:function(y){var C=this._dropIndicator,z=new o(C);var B=parseInt(C.css("margin-left"))||0,A=parseInt(C.css("margin-top"))||0,r=z.el.position();y._startOffset={x:r.left,y:r.top};y._dropIndicator=C;delete this._dropIndicator;}};j.dragdrop=n;}});
+/*@end @*/
diff --git a/minsrc/baselibs/prototype.js b/minsrc/baselibs/prototype.js
new file mode 100644 (file)
index 0000000..474b223
--- /dev/null
@@ -0,0 +1,6082 @@
+/*  Prototype JavaScript framework, version 1.7
+ *  (c) 2005-2010 Sam Stephenson
+ *
+ *  Prototype is freely distributable under the terms of an MIT-style license.
+ *  For details, see the Prototype web site: http://www.prototypejs.org/
+ *
+ *--------------------------------------------------------------------------*/
+
+var Prototype = {
+
+  Version: '1.7',
+
+  Browser: (function(){
+    var ua = navigator.userAgent;
+    var isOpera = Object.prototype.toString.call(window.opera) == '[object Opera]';
+    return {
+      IE:             !!window.attachEvent && !isOpera,
+      Opera:          isOpera,
+      WebKit:         ua.indexOf('AppleWebKit/') > -1,
+      Gecko:          ua.indexOf('Gecko') > -1 && ua.indexOf('KHTML') === -1,
+      MobileSafari:   /Apple.*Mobile/.test(ua)
+    }
+  })(),
+
+  BrowserFeatures: {
+    XPath: !!document.evaluate,
+
+    SelectorsAPI: !!document.querySelector,
+
+    ElementExtensions: (function() {
+      var constructor = window.Element || window.HTMLElement;
+      return !!(constructor && constructor.prototype);
+    })(),
+    SpecificElementExtensions: (function() {
+      if (typeof window.HTMLDivElement !== 'undefined')
+        return true;
+
+      var div = document.createElement('div'),
+          form = document.createElement('form'),
+          isSupported = false;
+
+      if (div['__proto__'] && (div['__proto__'] !== form['__proto__'])) {
+        isSupported = true;
+      }
+
+      div = form = null;
+
+      return isSupported;
+    })()
+  },
+
+  ScriptFragment: '<script[^>]*>([\\S\\s]*?)<\/script>',
+  JSONFilter: /^\/\*-secure-([\s\S]*)\*\/\s*$/,
+
+  emptyFunction: function() { },
+
+  K: function(x) { return x }
+};
+
+if (Prototype.Browser.MobileSafari)
+  Prototype.BrowserFeatures.SpecificElementExtensions = false;
+
+
+var Abstract = { };
+
+
+var Try = {
+  these: function() {
+    var returnValue;
+
+    for (var i = 0, length = arguments.length; i < length; i++) {
+      var lambda = arguments[i];
+      try {
+        returnValue = lambda();
+        break;
+      } catch (e) { }
+    }
+
+    return returnValue;
+  }
+};
+
+/* Based on Alex Arnell's inheritance implementation. */
+
+var Class = (function() {
+
+  var IS_DONTENUM_BUGGY = (function(){
+    for (var p in { toString: 1 }) {
+      if (p === 'toString') return false;
+    }
+    return true;
+  })();
+
+  function subclass() {};
+  function create() {
+    var parent = null, properties = $A(arguments);
+    if (Object.isFunction(properties[0]))
+      parent = properties.shift();
+
+    function klass() {
+      this.initialize.apply(this, arguments);
+    }
+
+    Object.extend(klass, Class.Methods);
+    klass.superclass = parent;
+    klass.subclasses = [];
+
+    if (parent) {
+      subclass.prototype = parent.prototype;
+      klass.prototype = new subclass;
+      parent.subclasses.push(klass);
+    }
+
+    for (var i = 0, length = properties.length; i < length; i++)
+      klass.addMethods(properties[i]);
+
+    if (!klass.prototype.initialize)
+      klass.prototype.initialize = Prototype.emptyFunction;
+
+    klass.prototype.constructor = klass;
+    return klass;
+  }
+
+  function addMethods(source) {
+    var ancestor   = this.superclass && this.superclass.prototype,
+        properties = Object.keys(source);
+
+    if (IS_DONTENUM_BUGGY) {
+      if (source.toString != Object.prototype.toString)
+        properties.push("toString");
+      if (source.valueOf != Object.prototype.valueOf)
+        properties.push("valueOf");
+    }
+
+    for (var i = 0, length = properties.length; i < length; i++) {
+      var property = properties[i], value = source[property];
+      if (ancestor && Object.isFunction(value) &&
+          value.argumentNames()[0] == "$super") {
+        var method = value;
+        value = (function(m) {
+          return function() { return ancestor[m].apply(this, arguments); };
+        })(property).wrap(method);
+
+        value.valueOf = method.valueOf.bind(method);
+        value.toString = method.toString.bind(method);
+      }
+      this.prototype[property] = value;
+    }
+
+    return this;
+  }
+
+  return {
+    create: create,
+    Methods: {
+      addMethods: addMethods
+    }
+  };
+})();
+(function() {
+
+  var _toString = Object.prototype.toString,
+      NULL_TYPE = 'Null',
+      UNDEFINED_TYPE = 'Undefined',
+      BOOLEAN_TYPE = 'Boolean',
+      NUMBER_TYPE = 'Number',
+      STRING_TYPE = 'String',
+      OBJECT_TYPE = 'Object',
+      FUNCTION_CLASS = '[object Function]',
+      BOOLEAN_CLASS = '[object Boolean]',
+      NUMBER_CLASS = '[object Number]',
+      STRING_CLASS = '[object String]',
+      ARRAY_CLASS = '[object Array]',
+      DATE_CLASS = '[object Date]',
+      NATIVE_JSON_STRINGIFY_SUPPORT = window.JSON &&
+        typeof JSON.stringify === 'function' &&
+        JSON.stringify(0) === '0' &&
+        typeof JSON.stringify(Prototype.K) === 'undefined';
+
+  function Type(o) {
+    switch(o) {
+      case null: return NULL_TYPE;
+      case (void 0): return UNDEFINED_TYPE;
+    }
+    var type = typeof o;
+    switch(type) {
+      case 'boolean': return BOOLEAN_TYPE;
+      case 'number':  return NUMBER_TYPE;
+      case 'string':  return STRING_TYPE;
+    }
+    return OBJECT_TYPE;
+  }
+
+  function extend(destination, source) {
+    for (var property in source)
+      destination[property] = source[property];
+    return destination;
+  }
+
+  function inspect(object) {
+    try {
+      if (isUndefined(object)) return 'undefined';
+      if (object === null) return 'null';
+      return object.inspect ? object.inspect() : String(object);
+    } catch (e) {
+      if (e instanceof RangeError) return '...';
+      throw e;
+    }
+  }
+
+  function toJSON(value) {
+    return Str('', { '': value }, []);
+  }
+
+  function Str(key, holder, stack) {
+    var value = holder[key],
+        type = typeof value;
+
+    if (Type(value) === OBJECT_TYPE && typeof value.toJSON === 'function') {
+      value = value.toJSON(key);
+    }
+
+    var _class = _toString.call(value);
+
+    switch (_class) {
+      case NUMBER_CLASS:
+      case BOOLEAN_CLASS:
+      case STRING_CLASS:
+        value = value.valueOf();
+    }
+
+    switch (value) {
+      case null: return 'null';
+      case true: return 'true';
+      case false: return 'false';
+    }
+
+    type = typeof value;
+    switch (type) {
+      case 'string':
+        return value.inspect(true);
+      case 'number':
+        return isFinite(value) ? String(value) : 'null';
+      case 'object':
+
+        for (var i = 0, length = stack.length; i < length; i++) {
+          if (stack[i] === value) { throw new TypeError(); }
+        }
+        stack.push(value);
+
+        var partial = [];
+        if (_class === ARRAY_CLASS) {
+          for (var i = 0, length = value.length; i < length; i++) {
+            var str = Str(i, value, stack);
+            partial.push(typeof str === 'undefined' ? 'null' : str);
+          }
+          partial = '[' + partial.join(',') + ']';
+        } else {
+          var keys = Object.keys(value);
+          for (var i = 0, length = keys.length; i < length; i++) {
+            var key = keys[i], str = Str(key, value, stack);
+            if (typeof str !== "undefined") {
+               partial.push(key.inspect(true)+ ':' + str);
+             }
+          }
+          partial = '{' + partial.join(',') + '}';
+        }
+        stack.pop();
+        return partial;
+    }
+  }
+
+  function stringify(object) {
+    return JSON.stringify(object);
+  }
+
+  function toQueryString(object) {
+    return $H(object).toQueryString();
+  }
+
+  function toHTML(object) {
+    return object && object.toHTML ? object.toHTML() : String.interpret(object);
+  }
+
+  function keys(object) {
+    if (Type(object) !== OBJECT_TYPE) { throw new TypeError(); }
+    var results = [];
+    for (var property in object) {
+      if (object.hasOwnProperty(property)) {
+        results.push(property);
+      }
+    }
+    return results;
+  }
+
+  function values(object) {
+    var results = [];
+    for (var property in object)
+      results.push(object[property]);
+    return results;
+  }
+
+  function clone(object) {
+    return extend({ }, object);
+  }
+
+  function isElement(object) {
+    return !!(object && object.nodeType == 1);
+  }
+
+  function isArray(object) {
+    return _toString.call(object) === ARRAY_CLASS;
+  }
+
+  var hasNativeIsArray = (typeof Array.isArray == 'function')
+    && Array.isArray([]) && !Array.isArray({});
+
+  if (hasNativeIsArray) {
+    isArray = Array.isArray;
+  }
+
+  function isHash(object) {
+    return object instanceof Hash;
+  }
+
+  function isFunction(object) {
+    return _toString.call(object) === FUNCTION_CLASS;
+  }
+
+  function isString(object) {
+    return _toString.call(object) === STRING_CLASS;
+  }
+
+  function isNumber(object) {
+    return _toString.call(object) === NUMBER_CLASS;
+  }
+
+  function isDate(object) {
+    return _toString.call(object) === DATE_CLASS;
+  }
+
+  function isUndefined(object) {
+    return typeof object === "undefined";
+  }
+
+  extend(Object, {
+    extend:        extend,
+    inspect:       inspect,
+    toJSON:        NATIVE_JSON_STRINGIFY_SUPPORT ? stringify : toJSON,
+    toQueryString: toQueryString,
+    toHTML:        toHTML,
+    keys:          Object.keys || keys,
+    values:        values,
+    clone:         clone,
+    isElement:     isElement,
+    isArray:       isArray,
+    isHash:        isHash,
+    isFunction:    isFunction,
+    isString:      isString,
+    isNumber:      isNumber,
+    isDate:        isDate,
+    isUndefined:   isUndefined
+  });
+})();
+Object.extend(Function.prototype, (function() {
+  var slice = Array.prototype.slice;
+
+  function update(array, args) {
+    var arrayLength = array.length, length = args.length;
+    while (length--) array[arrayLength + length] = args[length];
+    return array;
+  }
+
+  function merge(array, args) {
+    array = slice.call(array, 0);
+    return update(array, args);
+  }
+
+  function argumentNames() {
+    var names = this.toString().match(/^[\s\(]*function[^(]*\(([^)]*)\)/)[1]
+      .replace(/\/\/.*?[\r\n]|\/\*(?:.|[\r\n])*?\*\//g, '')
+      .replace(/\s+/g, '').split(',');
+    return names.length == 1 && !names[0] ? [] : names;
+  }
+
+  function bind(context) {
+    if (arguments.length < 2 && Object.isUndefined(arguments[0])) return this;
+    var __method = this, args = slice.call(arguments, 1);
+    return function() {
+      var a = merge(args, arguments);
+      return __method.apply(context, a);
+    }
+  }
+
+  function bindAsEventListener(context) {
+    var __method = this, args = slice.call(arguments, 1);
+    return function(event) {
+      var a = update([event || window.event], args);
+      return __method.apply(context, a);
+    }
+  }
+
+  function curry() {
+    if (!arguments.length) return this;
+    var __method = this, args = slice.call(arguments, 0);
+    return function() {
+      var a = merge(args, arguments);
+      return __method.apply(this, a);
+    }
+  }
+
+  function delay(timeout) {
+    var __method = this, args = slice.call(arguments, 1);
+    timeout = timeout * 1000;
+    return window.setTimeout(function() {
+      return __method.apply(__method, args);
+    }, timeout);
+  }
+
+  function defer() {
+    var args = update([0.01], arguments);
+    return this.delay.apply(this, args);
+  }
+
+  function wrap(wrapper) {
+    var __method = this;
+    return function() {
+      var a = update([__method.bind(this)], arguments);
+      return wrapper.apply(this, a);
+    }
+  }
+
+  function methodize() {
+    if (this._methodized) return this._methodized;
+    var __method = this;
+    return this._methodized = function() {
+      var a = update([this], arguments);
+      return __method.apply(null, a);
+    };
+  }
+
+  return {
+    argumentNames:       argumentNames,
+    bind:                bind,
+    bindAsEventListener: bindAsEventListener,
+    curry:               curry,
+    delay:               delay,
+    defer:               defer,
+    wrap:                wrap,
+    methodize:           methodize
+  }
+})());
+
+
+
+(function(proto) {
+
+
+  function toISOString() {
+    return this.getUTCFullYear() + '-' +
+      (this.getUTCMonth() + 1).toPaddedString(2) + '-' +
+      this.getUTCDate().toPaddedString(2) + 'T' +
+      this.getUTCHours().toPaddedString(2) + ':' +
+      this.getUTCMinutes().toPaddedString(2) + ':' +
+      this.getUTCSeconds().toPaddedString(2) + 'Z';
+  }
+
+
+  function toJSON() {
+    return this.toISOString();
+  }
+
+  if (!proto.toISOString) proto.toISOString = toISOString;
+  if (!proto.toJSON) proto.toJSON = toJSON;
+
+})(Date.prototype);
+
+
+RegExp.prototype.match = RegExp.prototype.test;
+
+RegExp.escape = function(str) {
+  return String(str).replace(/([.*+?^=!:${}()|[\]\/\\])/g, '\\$1');
+};
+var PeriodicalExecuter = Class.create({
+  initialize: function(callback, frequency) {
+    this.callback = callback;
+    this.frequency = frequency;
+    this.currentlyExecuting = false;
+
+    this.registerCallback();
+  },
+
+  registerCallback: function() {
+    this.timer = setInterval(this.onTimerEvent.bind(this), this.frequency * 1000);
+  },
+
+  execute: function() {
+    this.callback(this);
+  },
+
+  stop: function() {
+    if (!this.timer) return;
+    clearInterval(this.timer);
+    this.timer = null;
+  },
+
+  onTimerEvent: function() {
+    if (!this.currentlyExecuting) {
+      try {
+        this.currentlyExecuting = true;
+        this.execute();
+        this.currentlyExecuting = false;
+      } catch(e) {
+        this.currentlyExecuting = false;
+        throw e;
+      }
+    }
+  }
+});
+Object.extend(String, {
+  interpret: function(value) {
+    return value == null ? '' : String(value);
+  },
+  specialChar: {
+    '\b': '\\b',
+    '\t': '\\t',
+    '\n': '\\n',
+    '\f': '\\f',
+    '\r': '\\r',
+    '\\': '\\\\'
+  }
+});
+
+Object.extend(String.prototype, (function() {
+  var NATIVE_JSON_PARSE_SUPPORT = window.JSON &&
+    typeof JSON.parse === 'function' &&
+    JSON.parse('{"test": true}').test;
+
+  function prepareReplacement(replacement) {
+    if (Object.isFunction(replacement)) return replacement;
+    var template = new Template(replacement);
+    return function(match) { return template.evaluate(match) };
+  }
+
+  function gsub(pattern, replacement) {
+    var result = '', source = this, match;
+    replacement = prepareReplacement(replacement);
+
+    if (Object.isString(pattern))
+      pattern = RegExp.escape(pattern);
+
+    if (!(pattern.length || pattern.source)) {
+      replacement = replacement('');
+      return replacement + source.split('').join(replacement) + replacement;
+    }
+
+    while (source.length > 0) {
+      if (match = source.match(pattern)) {
+        result += source.slice(0, match.index);
+        result += String.interpret(replacement(match));
+        source  = source.slice(match.index + match[0].length);
+      } else {
+        result += source, source = '';
+      }
+    }
+    return result;
+  }
+
+  function sub(pattern, replacement, count) {
+    replacement = prepareReplacement(replacement);
+    count = Object.isUndefined(count) ? 1 : count;
+
+    return this.gsub(pattern, function(match) {
+      if (--count < 0) return match[0];
+      return replacement(match);
+    });
+  }
+
+  function scan(pattern, iterator) {
+    this.gsub(pattern, iterator);
+    return String(this);
+  }
+
+  function truncate(length, truncation) {
+    length = length || 30;
+    truncation = Object.isUndefined(truncation) ? '...' : truncation;
+    return this.length > length ?
+      this.slice(0, length - truncation.length) + truncation : String(this);
+  }
+
+  function strip() {
+    return this.replace(/^\s+/, '').replace(/\s+$/, '');
+  }
+
+  function stripTags() {
+    return this.replace(/<\w+(\s+("[^"]*"|'[^']*'|[^>])+)?>|<\/\w+>/gi, '');
+  }
+
+  function stripScripts() {
+    return this.replace(new RegExp(Prototype.ScriptFragment, 'img'), '');
+  }
+
+  function extractScripts() {
+    var matchAll = new RegExp(Prototype.ScriptFragment, 'img'),
+        matchOne = new RegExp(Prototype.ScriptFragment, 'im');
+    return (this.match(matchAll) || []).map(function(scriptTag) {
+      return (scriptTag.match(matchOne) || ['', ''])[1];
+    });
+  }
+
+  function evalScripts() {
+    return this.extractScripts().map(function(script) { return eval(script) });
+  }
+
+  function escapeHTML() {
+    return this.replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
+  }
+
+  function unescapeHTML() {
+    return this.stripTags().replace(/&lt;/g,'<').replace(/&gt;/g,'>').replace(/&amp;/g,'&');
+  }
+
+
+  function toQueryParams(separator) {
+    var match = this.strip().match(/([^?#]*)(#.*)?$/);
+    if (!match) return { };
+
+    return match[1].split(separator || '&').inject({ }, function(hash, pair) {
+      if ((pair = pair.split('='))[0]) {
+        var key = decodeURIComponent(pair.shift()),
+            value = pair.length > 1 ? pair.join('=') : pair[0];
+
+        if (value != undefined) value = decodeURIComponent(value);
+
+        if (key in hash) {
+          if (!Object.isArray(hash[key])) hash[key] = [hash[key]];
+          hash[key].push(value);
+        }
+        else hash[key] = value;
+      }
+      return hash;
+    });
+  }
+
+  function toArray() {
+    return this.split('');
+  }
+
+  function succ() {
+    return this.slice(0, this.length - 1) +
+      String.fromCharCode(this.charCodeAt(this.length - 1) + 1);
+  }
+
+  function times(count) {
+    return count < 1 ? '' : new Array(count + 1).join(this);
+  }
+
+  function camelize() {
+    return this.replace(/-+(.)?/g, function(match, chr) {
+      return chr ? chr.toUpperCase() : '';
+    });
+  }
+
+  function capitalize() {
+    return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase();
+  }
+
+  function underscore() {
+    return this.replace(/::/g, '/')
+               .replace(/([A-Z]+)([A-Z][a-z])/g, '$1_$2')
+               .replace(/([a-z\d])([A-Z])/g, '$1_$2')
+               .replace(/-/g, '_')
+               .toLowerCase();
+  }
+
+  function dasherize() {
+    return this.replace(/_/g, '-');
+  }
+
+  function inspect(useDoubleQuotes) {
+    var escapedString = this.replace(/[\x00-\x1f\\]/g, function(character) {
+      if (character in String.specialChar) {
+        return String.specialChar[character];
+      }
+      return '\\u00' + character.charCodeAt().toPaddedString(2, 16);
+    });
+    if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"';
+    return "'" + escapedString.replace(/'/g, '\\\'') + "'";
+  }
+
+  function unfilterJSON(filter) {
+    return this.replace(filter || Prototype.JSONFilter, '$1');
+  }
+
+  function isJSON() {
+    var str = this;
+    if (str.blank()) return false;
+    str = str.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@');
+    str = str.replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']');
+    str = str.replace(/(?:^|:|,)(?:\s*\[)+/g, '');
+    return (/^[\],:{}\s]*$/).test(str);
+  }
+
+  function evalJSON(sanitize) {
+    var json = this.unfilterJSON(),
+        cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
+    if (cx.test(json)) {
+      json = json.replace(cx, function (a) {
+        return '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
+      });
+    }
+    try {
+      if (!sanitize || json.isJSON()) return eval('(' + json + ')');
+    } catch (e) { }
+    throw new SyntaxError('Badly formed JSON string: ' + this.inspect());
+  }
+
+  function parseJSON() {
+    var json = this.unfilterJSON();
+    return JSON.parse(json);
+  }
+
+  function include(pattern) {
+    return this.indexOf(pattern) > -1;
+  }
+
+  function startsWith(pattern) {
+    return this.lastIndexOf(pattern, 0) === 0;
+  }
+
+  function endsWith(pattern) {
+    var d = this.length - pattern.length;
+    return d >= 0 && this.indexOf(pattern, d) === d;
+  }
+
+  function empty() {
+    return this == '';
+  }
+
+  function blank() {
+    return /^\s*$/.test(this);
+  }
+
+  function interpolate(object, pattern) {
+    return new Template(this, pattern).evaluate(object);
+  }
+
+  return {
+    gsub:           gsub,
+    sub:            sub,
+    scan:           scan,
+    truncate:       truncate,
+    strip:          String.prototype.trim || strip,
+    stripTags:      stripTags,
+    stripScripts:   stripScripts,
+    extractScripts: extractScripts,
+    evalScripts:    evalScripts,
+    escapeHTML:     escapeHTML,
+    unescapeHTML:   unescapeHTML,
+    toQueryParams:  toQueryParams,
+    parseQuery:     toQueryParams,
+    toArray:        toArray,
+    succ:           succ,
+    times:          times,
+    camelize:       camelize,
+    capitalize:     capitalize,
+    underscore:     underscore,
+    dasherize:      dasherize,
+    inspect:        inspect,
+    unfilterJSON:   unfilterJSON,
+    isJSON:         isJSON,
+    evalJSON:       NATIVE_JSON_PARSE_SUPPORT ? parseJSON : evalJSON,
+    include:        include,
+    startsWith:     startsWith,
+    endsWith:       endsWith,
+    empty:          empty,
+    blank:          blank,
+    interpolate:    interpolate
+  };
+})());
+
+var Template = Class.create({
+  initialize: function(template, pattern) {
+    this.template = template.toString();
+    this.pattern = pattern || Template.Pattern;
+  },
+
+  evaluate: function(object) {
+    if (object && Object.isFunction(object.toTemplateReplacements))
+      object = object.toTemplateReplacements();
+
+    return this.template.gsub(this.pattern, function(match) {
+      if (object == null) return (match[1] + '');
+
+      var before = match[1] || '';
+      if (before == '\\') return match[2];
+
+      var ctx = object, expr = match[3],
+          pattern = /^([^.[]+|\[((?:.*?[^\\])?)\])(\.|\[|$)/;
+
+      match = pattern.exec(expr);
+      if (match == null) return before;
+
+      while (match != null) {
+        var comp = match[1].startsWith('[') ? match[2].replace(/\\\\]/g, ']') : match[1];
+        ctx = ctx[comp];
+        if (null == ctx || '' == match[3]) break;
+        expr = expr.substring('[' == match[3] ? match[1].length : match[0].length);
+        match = pattern.exec(expr);
+      }
+
+      return before + String.interpret(ctx);
+    });
+  }
+});
+Template.Pattern = /(^|.|\r|\n)(#\{(.*?)\})/;
+
+var $break = { };
+
+var Enumerable = (function() {
+  function each(iterator, context) {
+    var index = 0;
+    try {
+      this._each(function(value) {
+        iterator.call(context, value, index++);
+      });
+    } catch (e) {
+      if (e != $break) throw e;
+    }
+    return this;
+  }
+
+  function eachSlice(number, iterator, context) {
+    var index = -number, slices = [], array = this.toArray();
+    if (number < 1) return array;
+    while ((index += number) < array.length)
+      slices.push(array.slice(index, index+number));
+    return slices.collect(iterator, context);
+  }
+
+  function all(iterator, context) {
+    iterator = iterator || Prototype.K;
+    var result = true;
+    this.each(function(value, index) {
+      result = result && !!iterator.call(context, value, index);
+      if (!result) throw $break;
+    });
+    return result;
+  }
+
+  function any(iterator, context) {
+    iterator = iterator || Prototype.K;
+    var result = false;
+    this.each(function(value, index) {
+      if (result = !!iterator.call(context, value, index))
+        throw $break;
+    });
+    return result;
+  }
+
+  function collect(iterator, context) {
+    iterator = iterator || Prototype.K;
+    var results = [];
+    this.each(function(value, index) {
+      results.push(iterator.call(context, value, index));
+    });
+    return results;
+  }
+
+  function detect(iterator, context) {
+    var result;
+    this.each(function(value, index) {
+      if (iterator.call(context, value, index)) {
+        result = value;
+        throw $break;
+      }
+    });
+    return result;
+  }
+
+  function findAll(iterator, context) {
+    var results = [];
+    this.each(function(value, index) {
+      if (iterator.call(context, value, index))
+        results.push(value);
+    });
+    return results;
+  }
+
+  function grep(filter, iterator, context) {
+    iterator = iterator || Prototype.K;
+    var results = [];
+
+    if (Object.isString(filter))
+      filter = new RegExp(RegExp.escape(filter));
+
+    this.each(function(value, index) {
+      if (filter.match(value))
+        results.push(iterator.call(context, value, index));
+    });
+    return results;
+  }
+
+  function include(object) {
+    if (Object.isFunction(this.indexOf))
+      if (this.indexOf(object) != -1) return true;
+
+    var found = false;
+    this.each(function(value) {
+      if (value == object) {
+        found = true;
+        throw $break;
+      }
+    });
+    return found;
+  }
+
+  function inGroupsOf(number, fillWith) {
+    fillWith = Object.isUndefined(fillWith) ? null : fillWith;
+    return this.eachSlice(number, function(slice) {
+      while(slice.length < number) slice.push(fillWith);
+      return slice;
+    });
+  }
+
+  function inject(memo, iterator, context) {
+    this.each(function(value, index) {
+      memo = iterator.call(context, memo, value, index);
+    });
+    return memo;
+  }
+
+  function invoke(method) {
+    var args = $A(arguments).slice(1);
+    return this.map(function(value) {
+      return value[method].apply(value, args);
+    });
+  }
+
+  function max(iterator, context) {
+    iterator = iterator || Prototype.K;
+    var result;
+    this.each(function(value, index) {
+      value = iterator.call(context, value, index);
+      if (result == null || value >= result)
+        result = value;
+    });
+    return result;
+  }
+
+  function min(iterator, context) {
+    iterator = iterator || Prototype.K;
+    var result;
+    this.each(function(value, index) {
+      value = iterator.call(context, value, index);
+      if (result == null || value < result)
+        result = value;
+    });
+    return result;
+  }
+
+  function partition(iterator, context) {
+    iterator = iterator || Prototype.K;
+    var trues = [], falses = [];
+    this.each(function(value, index) {
+      (iterator.call(context, value, index) ?
+        trues : falses).push(value);
+    });
+    return [trues, falses];
+  }
+
+  function pluck(property) {
+    var results = [];
+    this.each(function(value) {
+      results.push(value[property]);
+    });
+    return results;
+  }
+
+  function reject(iterator, context) {
+    var results = [];
+    this.each(function(value, index) {
+      if (!iterator.call(context, value, index))
+        results.push(value);
+    });
+    return results;
+  }
+
+  function sortBy(iterator, context) {
+    return this.map(function(value, index) {
+      return {
+        value: value,
+        criteria: iterator.call(context, value, index)
+      };
+    }).sort(function(left, right) {
+      var a = left.criteria, b = right.criteria;
+      return a < b ? -1 : a > b ? 1 : 0;
+    }).pluck('value');
+  }
+
+  function toArray() {
+    return this.map();
+  }
+
+  function zip() {
+    var iterator = Prototype.K, args = $A(arguments);
+    if (Object.isFunction(args.last()))
+      iterator = args.pop();
+
+    var collections = [this].concat(args).map($A);
+    return this.map(function(value, index) {
+      return iterator(collections.pluck(index));
+    });
+  }
+
+  function size() {
+    return this.toArray().length;
+  }
+
+  function inspect() {
+    return '#<Enumerable:' + this.toArray().inspect() + '>';
+  }
+
+
+
+
+
+
+
+
+
+  return {
+    each:       each,
+    eachSlice:  eachSlice,
+    all:        all,
+    every:      all,
+    any:        any,
+    some:       any,
+    collect:    collect,
+    map:        collect,
+    detect:     detect,
+    findAll:    findAll,
+    select:     findAll,
+    filter:     findAll,
+    grep:       grep,
+    include:    include,
+    member:     include,
+    inGroupsOf: inGroupsOf,
+    inject:     inject,
+    invoke:     invoke,
+    max:        max,
+    min:        min,
+    partition:  partition,
+    pluck:      pluck,
+    reject:     reject,
+    sortBy:     sortBy,
+    toArray:    toArray,
+    entries:    toArray,
+    zip:        zip,
+    size:       size,
+    inspect:    inspect,
+    find:       detect
+  };
+})();
+
+function $A(iterable) {
+  if (!iterable) return [];
+  if ('toArray' in Object(iterable)) return iterable.toArray();
+  var length = iterable.length || 0, results = new Array(length);
+  while (length--) results[length] = iterable[length];
+  return results;
+}
+
+
+function $w(string) {
+  if (!Object.isString(string)) return [];
+  string = string.strip();
+  return string ? string.split(/\s+/) : [];
+}
+
+Array.from = $A;
+
+
+(function() {
+  var arrayProto = Array.prototype,
+      slice = arrayProto.slice,
+      _each = arrayProto.forEach; // use native browser JS 1.6 implementation if available
+
+  function each(iterator, context) {
+    for (var i = 0, length = this.length >>> 0; i < length; i++) {
+      if (i in this) iterator.call(context, this[i], i, this);
+    }
+  }
+  if (!_each) _each = each;
+
+  function clear() {
+    this.length = 0;
+    return this;
+  }
+
+  function first() {
+    return this[0];
+  }
+
+  function last() {
+    return this[this.length - 1];
+  }
+
+  function compact() {
+    return this.select(function(value) {
+      return value != null;
+    });
+  }
+
+  function flatten() {
+    return this.inject([], function(array, value) {
+      if (Object.isArray(value))
+        return array.concat(value.flatten());
+      array.push(value);
+      return array;
+    });
+  }
+
+  function without() {
+    var values = slice.call(arguments, 0);
+    return this.select(function(value) {
+      return !values.include(value);
+    });
+  }
+
+  function reverse(inline) {
+    return (inline === false ? this.toArray() : this)._reverse();
+  }
+
+  function uniq(sorted) {
+    return this.inject([], function(array, value, index) {
+      if (0 == index || (sorted ? array.last() != value : !array.include(value)))
+        array.push(value);
+      return array;
+    });
+  }
+
+  function intersect(array) {
+    return this.uniq().findAll(function(item) {
+      return array.detect(function(value) { return item === value });
+    });
+  }
+
+
+  function clone() {
+    return slice.call(this, 0);
+  }
+
+  function size() {
+    return this.length;
+  }
+
+  function inspect() {
+    return '[' + this.map(Object.inspect).join(', ') + ']';
+  }
+
+  function indexOf(item, i) {
+    i || (i = 0);
+    var length = this.length;
+    if (i < 0) i = length + i;
+    for (; i < length; i++)
+      if (this[i] === item) return i;
+    return -1;
+  }
+
+  function lastIndexOf(item, i) {
+    i = isNaN(i) ? this.length : (i < 0 ? this.length + i : i) + 1;
+    var n = this.slice(0, i).reverse().indexOf(item);
+    return (n < 0) ? n : i - n - 1;
+  }
+
+  function concat() {
+    var array = slice.call(this, 0), item;
+    for (var i = 0, length = arguments.length; i < length; i++) {
+      item = arguments[i];
+      if (Object.isArray(item) && !('callee' in item)) {
+        for (var j = 0, arrayLength = item.length; j < arrayLength; j++)
+          array.push(item[j]);
+      } else {
+        array.push(item);
+      }
+    }
+    return array;
+  }
+
+  Object.extend(arrayProto, Enumerable);
+
+  if (!arrayProto._reverse)
+    arrayProto._reverse = arrayProto.reverse;
+
+  Object.extend(arrayProto, {
+    _each:     _each,
+    clear:     clear,
+    first:     first,
+    last:      last,
+    compact:   compact,
+    flatten:   flatten,
+    without:   without,
+    reverse:   reverse,
+    uniq:      uniq,
+    intersect: intersect,
+    clone:     clone,
+    toArray:   clone,
+    size:      size,
+    inspect:   inspect
+  });
+
+  var CONCAT_ARGUMENTS_BUGGY = (function() {
+    return [].concat(arguments)[0][0] !== 1;
+  })(1,2)
+
+  if (CONCAT_ARGUMENTS_BUGGY) arrayProto.concat = concat;
+
+  if (!arrayProto.indexOf) arrayProto.indexOf = indexOf;
+  if (!arrayProto.lastIndexOf) arrayProto.lastIndexOf = lastIndexOf;
+})();
+function $H(object) {
+  return new Hash(object);
+};
+
+var Hash = Class.create(Enumerable, (function() {
+  function initialize(object) {
+    this._object = Object.isHash(object) ? object.toObject() : Object.clone(object);
+  }
+
+
+  function _each(iterator) {
+    for (var key in this._object) {
+      var value = this._object[key], pair = [key, value];
+      pair.key = key;
+      pair.value = value;
+      iterator(pair);
+    }
+  }
+
+  function set(key, value) {
+    return this._object[key] = value;
+  }
+
+  function get(key) {
+    if (this._object[key] !== Object.prototype[key])
+      return this._object[key];
+  }
+
+  function unset(key) {
+    var value = this._object[key];
+    delete this._object[key];
+    return value;
+  }
+
+  function toObject() {
+    return Object.clone(this._object);
+  }
+
+
+
+  function keys() {
+    return this.pluck('key');
+  }
+
+  function values() {
+    return this.pluck('value');
+  }
+
+  function index(value) {
+    var match = this.detect(function(pair) {
+      return pair.value === value;
+    });
+    return match && match.key;
+  }
+
+  function merge(object) {
+    return this.clone().update(object);
+  }
+
+  function update(object) {
+    return new Hash(object).inject(this, function(result, pair) {
+      result.set(pair.key, pair.value);
+      return result;
+    });
+  }
+
+  function toQueryPair(key, value) {
+    if (Object.isUndefined(value)) return key;
+    return key + '=' + encodeURIComponent(String.interpret(value));
+  }
+
+  function toQueryString() {
+    return this.inject([], function(results, pair) {
+      var key = encodeURIComponent(pair.key), values = pair.value;
+
+      if (values && typeof values == 'object') {
+        if (Object.isArray(values)) {
+          var queryValues = [];
+          for (var i = 0, len = values.length, value; i < len; i++) {
+            value = values[i];
+            queryValues.push(toQueryPair(key, value));
+          }
+          return results.concat(queryValues);
+        }
+      } else results.push(toQueryPair(key, values));
+      return results;
+    }).join('&');
+  }
+
+  function inspect() {
+    return '#<Hash:{' + this.map(function(pair) {
+      return pair.map(Object.inspect).join(': ');
+    }).join(', ') + '}>';
+  }
+
+  function clone() {
+    return new Hash(this);
+  }
+
+  return {
+    initialize:             initialize,
+    _each:                  _each,
+    set:                    set,
+    get:                    get,
+    unset:                  unset,
+    toObject:               toObject,
+    toTemplateReplacements: toObject,
+    keys:                   keys,
+    values:                 values,
+    index:                  index,
+    merge:                  merge,
+    update:                 update,
+    toQueryString:          toQueryString,
+    inspect:                inspect,
+    toJSON:                 toObject,
+    clone:                  clone
+  };
+})());
+
+Hash.from = $H;
+Object.extend(Number.prototype, (function() {
+  function toColorPart() {
+    return this.toPaddedString(2, 16);
+  }
+
+  function succ() {
+    return this + 1;
+  }
+
+  function times(iterator, context) {
+    $R(0, this, true).each(iterator, context);
+    return this;
+  }
+
+  function toPaddedString(length, radix) {
+    var string = this.toString(radix || 10);
+    return '0'.times(length - string.length) + string;
+  }
+
+  function abs() {
+    return Math.abs(this);
+  }
+
+  function round() {
+    return Math.round(this);
+  }
+
+  function ceil() {
+    return Math.ceil(this);
+  }
+
+  function floor() {
+    return Math.floor(this);
+  }
+
+  return {
+    toColorPart:    toColorPart,
+    succ:           succ,
+    times:          times,
+    toPaddedString: toPaddedString,
+    abs:            abs,
+    round:          round,
+    ceil:           ceil,
+    floor:          floor
+  };
+})());
+
+function $R(start, end, exclusive) {
+  return new ObjectRange(start, end, exclusive);
+}
+
+var ObjectRange = Class.create(Enumerable, (function() {
+  function initialize(start, end, exclusive) {
+    this.start = start;
+    this.end = end;
+    this.exclusive = exclusive;
+  }
+
+  function _each(iterator) {
+    var value = this.start;
+    while (this.include(value)) {
+      iterator(value);
+      value = value.succ();
+    }
+  }
+
+  function include(value) {
+    if (value < this.start)
+      return false;
+    if (this.exclusive)
+      return value < this.end;
+    return value <= this.end;
+  }
+
+  return {
+    initialize: initialize,
+    _each:      _each,
+    include:    include
+  };
+})());
+
+
+
+var Ajax = {
+  getTransport: function() {
+    return Try.these(
+      function() {return new XMLHttpRequest()},
+      function() {return new ActiveXObject('Msxml2.XMLHTTP')},
+      function() {return new ActiveXObject('Microsoft.XMLHTTP')}
+    ) || false;
+  },
+
+  activeRequestCount: 0
+};
+
+Ajax.Responders = {
+  responders: [],
+
+  _each: function(iterator) {
+    this.responders._each(iterator);
+  },
+
+  register: function(responder) {
+    if (!this.include(responder))
+      this.responders.push(responder);
+  },
+
+  unregister: function(responder) {
+    this.responders = this.responders.without(responder);
+  },
+
+  dispatch: function(callback, request, transport, json) {
+    this.each(function(responder) {
+      if (Object.isFunction(responder[callback])) {
+        try {
+          responder[callback].apply(responder, [request, transport, json]);
+        } catch (e) { }
+      }
+    });
+  }
+};
+
+Object.extend(Ajax.Responders, Enumerable);
+
+Ajax.Responders.register({
+  onCreate:   function() { Ajax.activeRequestCount++ },
+  onComplete: function() { Ajax.activeRequestCount-- }
+});
+Ajax.Base = Class.create({
+  initialize: function(options) {
+    this.options = {
+      method:       'post',
+      asynchronous: true,
+      contentType:  'application/x-www-form-urlencoded',
+      encoding:     'UTF-8',
+      parameters:   '',
+      evalJSON:     true,
+      evalJS:       true
+    };
+    Object.extend(this.options, options || { });
+
+    this.options.method = this.options.method.toLowerCase();
+
+    if (Object.isHash(this.options.parameters))
+      this.options.parameters = this.options.parameters.toObject();
+  }
+});
+Ajax.Request = Class.create(Ajax.Base, {
+  _complete: false,
+
+  initialize: function($super, url, options) {
+    $super(options);
+    this.transport = Ajax.getTransport();
+    this.request(url);
+  },
+
+  request: function(url) {
+    this.url = url;
+    this.method = this.options.method;
+    var params = Object.isString(this.options.parameters) ?
+          this.options.parameters :
+          Object.toQueryString(this.options.parameters);
+
+    if (!['get', 'post'].include(this.method)) {
+      params += (params ? '&' : '') + "_method=" + this.method;
+      this.method = 'post';
+    }
+
+    if (params && this.method === 'get') {
+      this.url += (this.url.include('?') ? '&' : '?') + params;
+    }
+
+    this.parameters = params.toQueryParams();
+
+    try {
+      var response = new Ajax.Response(this);
+      if (this.options.onCreate) this.options.onCreate(response);
+      Ajax.Responders.dispatch('onCreate', this, response);
+
+      this.transport.open(this.method.toUpperCase(), this.url,
+        this.options.asynchronous);
+
+      if (this.options.asynchronous) this.respondToReadyState.bind(this).defer(1);
+
+      this.transport.onreadystatechange = this.onStateChange.bind(this);
+      this.setRequestHeaders();
+
+      this.body = this.method == 'post' ? (this.options.postBody || params) : null;
+      this.transport.send(this.body);
+
+      /* Force Firefox to handle ready state 4 for synchronous requests */
+      if (!this.options.asynchronous && this.transport.overrideMimeType)
+        this.onStateChange();
+
+    }
+    catch (e) {
+      this.dispatchException(e);
+    }
+  },
+
+  onStateChange: function() {
+    var readyState = this.transport.readyState;
+    if (readyState > 1 && !((readyState == 4) && this._complete))
+      this.respondToReadyState(this.transport.readyState);
+  },
+
+  setRequestHeaders: function() {
+    var headers = {
+      'X-Requested-With': 'XMLHttpRequest',
+      'X-Prototype-Version': Prototype.Version,
+      'Accept': 'text/javascript, text/html, application/xml, text/xml, */*'
+    };
+
+    if (this.method == 'post') {
+      headers['Content-type'] = this.options.contentType +
+        (this.options.encoding ? '; charset=' + this.options.encoding : '');
+
+      /* Force "Connection: close" for older Mozilla browsers to work
+       * around a bug where XMLHttpRequest sends an incorrect
+       * Content-length header. See Mozilla Bugzilla #246651.
+       */
+      if (this.transport.overrideMimeType &&
+          (navigator.userAgent.match(/Gecko\/(\d{4})/) || [0,2005])[1] < 2005)
+            headers['Connection'] = 'close';
+    }
+
+    if (typeof this.options.requestHeaders == 'object') {
+      var extras = this.options.requestHeaders;
+
+      if (Object.isFunction(extras.push))
+        for (var i = 0, length = extras.length; i < length; i += 2)
+          headers[extras[i]] = extras[i+1];
+      else
+        $H(extras).each(function(pair) { headers[pair.key] = pair.value });
+    }
+
+    for (var name in headers)
+      this.transport.setRequestHeader(name, headers[name]);
+  },
+
+  success: function() {
+    var status = this.getStatus();
+    return !status || (status >= 200 && status < 300) || status == 304;
+  },
+
+  getStatus: function() {
+    try {
+      if (this.transport.status === 1223) return 204;
+      return this.transport.status || 0;
+    } catch (e) { return 0 }
+  },
+
+  respondToReadyState: function(readyState) {
+    var state = Ajax.Request.Events[readyState], response = new Ajax.Response(this);
+
+    if (state == 'Complete') {
+      try {
+        this._complete = true;
+        (this.options['on' + response.status]
+         || this.options['on' + (this.success() ? 'Success' : 'Failure')]
+         || Prototype.emptyFunction)(response, response.headerJSON);
+      } catch (e) {
+        this.dispatchException(e);
+      }
+
+      var contentType = response.getHeader('Content-type');
+      if (this.options.evalJS == 'force'
+          || (this.options.evalJS && this.isSameOrigin() && contentType
+          && contentType.match(/^\s*(text|application)\/(x-)?(java|ecma)script(;.*)?\s*$/i)))
+        this.evalResponse();
+    }
+
+    try {
+      (this.options['on' + state] || Prototype.emptyFunction)(response, response.headerJSON);
+      Ajax.Responders.dispatch('on' + state, this, response, response.headerJSON);
+    } catch (e) {
+      this.dispatchException(e);
+    }
+
+    if (state == 'Complete') {
+      this.transport.onreadystatechange = Prototype.emptyFunction;
+    }
+  },
+
+  isSameOrigin: function() {
+    var m = this.url.match(/^\s*https?:\/\/[^\/]*/);
+    return !m || (m[0] == '#{protocol}//#{domain}#{port}'.interpolate({
+      protocol: location.protocol,
+      domain: document.domain,
+      port: location.port ? ':' + location.port : ''
+    }));
+  },
+
+  getHeader: function(name) {
+    try {
+      return this.transport.getResponseHeader(name) || null;
+    } catch (e) { return null; }
+  },
+
+  evalResponse: function() {
+    try {
+      return eval((this.transport.responseText || '').unfilterJSON());
+    } catch (e) {
+      this.dispatchException(e);
+    }
+  },
+
+  dispatchException: function(exception) {
+    (this.options.onException || Prototype.emptyFunction)(this, exception);
+    Ajax.Responders.dispatch('onException', this, exception);
+  }
+});
+
+Ajax.Request.Events =
+  ['Uninitialized', 'Loading', 'Loaded', 'Interactive', 'Complete'];
+
+
+
+
+
+
+
+
+Ajax.Response = Class.create({
+  initialize: function(request){
+    this.request = request;
+    var transport  = this.transport  = request.transport,
+        readyState = this.readyState = transport.readyState;
+
+    if ((readyState > 2 && !Prototype.Browser.IE) || readyState == 4) {
+      this.status       = this.getStatus();
+      this.statusText   = this.getStatusText();
+      this.responseText = String.interpret(transport.responseText);
+      this.headerJSON   = this._getHeaderJSON();
+    }
+
+    if (readyState == 4) {
+      var xml = transport.responseXML;
+      this.responseXML  = Object.isUndefined(xml) ? null : xml;
+      this.responseJSON = this._getResponseJSON();
+    }
+  },
+
+  status:      0,
+
+  statusText: '',
+
+  getStatus: Ajax.Request.prototype.getStatus,
+
+  getStatusText: function() {
+    try {
+      return this.transport.statusText || '';
+    } catch (e) { return '' }
+  },
+
+  getHeader: Ajax.Request.prototype.getHeader,
+
+  getAllHeaders: function() {
+    try {
+      return this.getAllResponseHeaders();
+    } catch (e) { return null }
+  },
+
+  getResponseHeader: function(name) {
+    return this.transport.getResponseHeader(name);
+  },
+
+  getAllResponseHeaders: function() {
+    return this.transport.getAllResponseHeaders();
+  },
+
+  _getHeaderJSON: function() {
+    var json = this.getHeader('X-JSON');
+    if (!json) return null;
+    json = decodeURIComponent(escape(json));
+    try {
+      return json.evalJSON(this.request.options.sanitizeJSON ||
+        !this.request.isSameOrigin());
+    } catch (e) {
+      this.request.dispatchException(e);
+    }
+  },
+
+  _getResponseJSON: function() {
+    var options = this.request.options;
+    if (!options.evalJSON || (options.evalJSON != 'force' &&
+      !(this.getHeader('Content-type') || '').include('application/json')) ||
+        this.responseText.blank())
+          return null;
+    try {
+      return this.responseText.evalJSON(options.sanitizeJSON ||
+        !this.request.isSameOrigin());
+    } catch (e) {
+      this.request.dispatchException(e);
+    }
+  }
+});
+
+Ajax.Updater = Class.create(Ajax.Request, {
+  initialize: function($super, container, url, options) {
+    this.container = {
+      success: (container.success || container),
+      failure: (container.failure || (container.success ? null : container))
+    };
+
+    options = Object.clone(options);
+    var onComplete = options.onComplete;
+    options.onComplete = (function(response, json) {
+      this.updateContent(response.responseText);
+      if (Object.isFunction(onComplete)) onComplete(response, json);
+    }).bind(this);
+
+    $super(url, options);
+  },
+
+  updateContent: function(responseText) {
+    var receiver = this.container[this.success() ? 'success' : 'failure'],
+        options = this.options;
+
+    if (!options.evalScripts) responseText = responseText.stripScripts();
+
+    if (receiver = $(receiver)) {
+      if (options.insertion) {
+        if (Object.isString(options.insertion)) {
+          var insertion = { }; insertion[options.insertion] = responseText;
+          receiver.insert(insertion);
+        }
+        else options.insertion(receiver, responseText);
+      }
+      else receiver.update(responseText);
+    }
+  }
+});
+
+Ajax.PeriodicalUpdater = Class.create(Ajax.Base, {
+  initialize: function($super, container, url, options) {
+    $super(options);
+    this.onComplete = this.options.onComplete;
+
+    this.frequency = (this.options.frequency || 2);
+    this.decay = (this.options.decay || 1);
+
+    this.updater = { };
+    this.container = container;
+    this.url = url;
+
+    this.start();
+  },
+
+  start: function() {
+    this.options.onComplete = this.updateComplete.bind(this);
+    this.onTimerEvent();
+  },
+
+  stop: function() {
+    this.updater.options.onComplete = undefined;
+    clearTimeout(this.timer);
+    (this.onComplete || Prototype.emptyFunction).apply(this, arguments);
+  },
+
+  updateComplete: function(response) {
+    if (this.options.decay) {
+      this.decay = (response.responseText == this.lastText ?
+        this.decay * this.options.decay : 1);
+
+      this.lastText = response.responseText;
+    }
+    this.timer = this.onTimerEvent.bind(this).delay(this.decay * this.frequency);
+  },
+
+  onTimerEvent: function() {
+    this.updater = new Ajax.Updater(this.container, this.url, this.options);
+  }
+});
+
+
+function $(element) {
+  if (arguments.length > 1) {
+    for (var i = 0, elements = [], length = arguments.length; i < length; i++)
+      elements.push($(arguments[i]));
+    return elements;
+  }
+  if (Object.isString(element))
+    element = document.getElementById(element);
+  return Element.extend(element);
+}
+
+if (Prototype.BrowserFeatures.XPath) {
+  document._getElementsByXPath = function(expression, parentElement) {
+    var results = [];
+    var query = document.evaluate(expression, $(parentElement) || document,
+      null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
+    for (var i = 0, length = query.snapshotLength; i < length; i++)
+      results.push(Element.extend(query.snapshotItem(i)));
+    return results;
+  };
+}
+
+/*--------------------------------------------------------------------------*/
+
+if (!Node) var Node = { };
+
+if (!Node.ELEMENT_NODE) {
+  Object.extend(Node, {
+    ELEMENT_NODE: 1,
+    ATTRIBUTE_NODE: 2,
+    TEXT_NODE: 3,
+    CDATA_SECTION_NODE: 4,
+    ENTITY_REFERENCE_NODE: 5,
+    ENTITY_NODE: 6,
+    PROCESSING_INSTRUCTION_NODE: 7,
+    COMMENT_NODE: 8,
+    DOCUMENT_NODE: 9,
+    DOCUMENT_TYPE_NODE: 10,
+    DOCUMENT_FRAGMENT_NODE: 11,
+    NOTATION_NODE: 12
+  });
+}
+
+
+
+(function(global) {
+  function shouldUseCache(tagName, attributes) {
+    if (tagName === 'select') return false;
+    if ('type' in attributes) return false;
+    return true;
+  }
+
+  var HAS_EXTENDED_CREATE_ELEMENT_SYNTAX = (function(){
+    try {
+      var el = document.createElement('<input name="x">');
+      return el.tagName.toLowerCase() === 'input' && el.name === 'x';
+    }
+    catch(err) {
+      return false;
+    }
+  })();
+
+  var element = global.Element;
+
+  global.Element = function(tagName, attributes) {
+    attributes = attributes || { };
+    tagName = tagName.toLowerCase();
+    var cache = Element.cache;
+
+    if (HAS_EXTENDED_CREATE_ELEMENT_SYNTAX && attributes.name) {
+      tagName = '<' + tagName + ' name="' + attributes.name + '">';
+      delete attributes.name;
+      return Element.writeAttribute(document.createElement(tagName), attributes);
+    }
+
+    if (!cache[tagName]) cache[tagName] = Element.extend(document.createElement(tagName));
+
+    var node = shouldUseCache(tagName, attributes) ?
+     cache[tagName].cloneNode(false) : document.createElement(tagName);
+
+    return Element.writeAttribute(node, attributes);
+  };
+
+  Object.extend(global.Element, element || { });
+  if (element) global.Element.prototype = element.prototype;
+
+})(this);
+
+Element.idCounter = 1;
+Element.cache = { };
+
+Element._purgeElement = function(element) {
+  var uid = element._prototypeUID;
+  if (uid) {
+    Element.stopObserving(element);
+    element._prototypeUID = void 0;
+    delete Element.Storage[uid];
+  }
+}
+
+Element.Methods = {
+  visible: function(element) {
+    return $(element).style.display != 'none';
+  },
+
+  toggle: function(element) {
+    element = $(element);
+    Element[Element.visible(element) ? 'hide' : 'show'](element);
+    return element;
+  },
+
+  hide: function(element) {
+    element = $(element);
+    element.style.display = 'none';
+    return element;
+  },
+
+  show: function(element) {
+    element = $(element);
+    element.style.display = '';
+    return element;
+  },
+
+  remove: function(element) {
+    element = $(element);
+    element.parentNode.removeChild(element);
+    return element;
+  },
+
+  update: (function(){
+
+    var SELECT_ELEMENT_INNERHTML_BUGGY = (function(){
+      var el = document.createElement("select"),
+          isBuggy = true;
+      el.innerHTML = "<option value=\"test\">test</option>";
+      if (el.options && el.options[0]) {
+        isBuggy = el.options[0].nodeName.toUpperCase() !== "OPTION";
+      }
+      el = null;
+      return isBuggy;
+    })();
+
+    var TABLE_ELEMENT_INNERHTML_BUGGY = (function(){
+      try {
+        var el = document.createElement("table");
+        if (el && el.tBodies) {
+          el.innerHTML = "<tbody><tr><td>test</td></tr></tbody>";
+          var isBuggy = typeof el.tBodies[0] == "undefined";
+          el = null;
+          return isBuggy;
+        }
+      } catch (e) {
+        return true;
+      }
+    })();
+
+    var LINK_ELEMENT_INNERHTML_BUGGY = (function() {
+      try {
+        var el = document.createElement('div');
+        el.innerHTML = "<link>";
+        var isBuggy = (el.childNodes.length === 0);
+        el = null;
+        return isBuggy;
+      } catch(e) {
+        return true;
+      }
+    })();
+
+    var ANY_INNERHTML_BUGGY = SELECT_ELEMENT_INNERHTML_BUGGY ||
+     TABLE_ELEMENT_INNERHTML_BUGGY || LINK_ELEMENT_INNERHTML_BUGGY;
+
+    var SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING = (function () {
+      var s = document.createElement("script"),
+          isBuggy = false;
+      try {
+        s.appendChild(document.createTextNode(""));
+        isBuggy = !s.firstChild ||
+          s.firstChild && s.firstChild.nodeType !== 3;
+      } catch (e) {
+        isBuggy = true;
+      }
+      s = null;
+      return isBuggy;
+    })();
+
+
+    function update(element, content) {
+      element = $(element);
+      var purgeElement = Element._purgeElement;
+
+      var descendants = element.getElementsByTagName('*'),
+       i = descendants.length;
+      while (i--) purgeElement(descendants[i]);
+
+      if (content && content.toElement)
+        content = content.toElement();
+
+      if (Object.isElement(content))
+        return element.update().insert(content);
+
+      content = Object.toHTML(content);
+
+      var tagName = element.tagName.toUpperCase();
+
+      if (tagName === 'SCRIPT' && SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING) {
+        element.text = content;
+        return element;
+      }
+
+      if (ANY_INNERHTML_BUGGY) {
+        if (tagName in Element._insertionTranslations.tags) {
+          while (element.firstChild) {
+            element.removeChild(element.firstChild);
+          }
+          Element._getContentFromAnonymousElement(tagName, content.stripScripts())
+            .each(function(node) {
+              element.appendChild(node)
+            });
+        } else if (LINK_ELEMENT_INNERHTML_BUGGY && Object.isString(content) && content.indexOf('<link') > -1) {
+          while (element.firstChild) {
+            element.removeChild(element.firstChild);
+          }
+          var nodes = Element._getContentFromAnonymousElement(tagName, content.stripScripts(), true);
+          nodes.each(function(node) { element.appendChild(node) });
+        }
+        else {
+          element.innerHTML = content.stripScripts();
+        }
+      }
+      else {
+        element.innerHTML = content.stripScripts();
+      }
+
+      content.evalScripts.bind(content).defer();
+      return element;
+    }
+
+    return update;
+  })(),
+
+  replace: function(element, content) {
+    element = $(element);
+    if (content && content.toElement) content = content.toElement();
+    else if (!Object.isElement(content)) {
+      content = Object.toHTML(content);
+      var range = element.ownerDocument.createRange();
+      range.selectNode(element);
+      content.evalScripts.bind(content).defer();
+      content = range.createContextualFragment(content.stripScripts());
+    }
+    element.parentNode.replaceChild(content, element);
+    return element;
+  },
+
+  insert: function(element, insertions) {
+    element = $(element);
+
+    if (Object.isString(insertions) || Object.isNumber(insertions) ||
+        Object.isElement(insertions) || (insertions && (insertions.toElement || insertions.toHTML)))
+          insertions = {bottom:insertions};
+
+    var content, insert, tagName, childNodes;
+
+    for (var position in insertions) {
+      content  = insertions[position];
+      position = position.toLowerCase();
+      insert = Element._insertionTranslations[position];
+
+      if (content && content.toElement) content = content.toElement();
+      if (Object.isElement(content)) {
+        insert(element, content);
+        continue;
+      }
+
+      content = Object.toHTML(content);
+
+      tagName = ((position == 'before' || position == 'after')
+        ? element.parentNode : element).tagName.toUpperCase();
+
+      childNodes = Element._getContentFromAnonymousElement(tagName, content.stripScripts());
+
+      if (position == 'top' || position == 'after') childNodes.reverse();
+      childNodes.each(insert.curry(element));
+
+      content.evalScripts.bind(content).defer();
+    }
+
+    return element;
+  },
+
+  wrap: function(element, wrapper, attributes) {
+    element = $(element);
+    if (Object.isElement(wrapper))
+      $(wrapper).writeAttribute(attributes || { });
+    else if (Object.isString(wrapper)) wrapper = new Element(wrapper, attributes);
+    else wrapper = new Element('div', wrapper);
+    if (element.parentNode)
+      element.parentNode.replaceChild(wrapper, element);
+    wrapper.appendChild(element);
+    return wrapper;
+  },
+
+  inspect: function(element) {
+    element = $(element);
+    var result = '<' + element.tagName.toLowerCase();
+    $H({'id': 'id', 'className': 'class'}).each(function(pair) {
+      var property = pair.first(),
+          attribute = pair.last(),
+          value = (element[property] || '').toString();
+      if (value) result += ' ' + attribute + '=' + value.inspect(true);
+    });
+    return result + '>';
+  },
+
+  recursivelyCollect: function(element, property, maximumLength) {
+    element = $(element);
+    maximumLength = maximumLength || -1;
+    var elements = [];
+
+    while (element = element[property]) {
+      if (element.nodeType == 1)
+        elements.push(Element.extend(element));
+      if (elements.length == maximumLength)
+        break;
+    }
+
+    return elements;
+  },
+
+  ancestors: function(element) {
+    return Element.recursivelyCollect(element, 'parentNode');
+  },
+
+  descendants: function(element) {
+    return Element.select(element, "*");
+  },
+
+  firstDescendant: function(element) {
+    element = $(element).firstChild;
+    while (element && element.nodeType != 1) element = element.nextSibling;
+    return $(element);
+  },
+
+  immediateDescendants: function(element) {
+    var results = [], child = $(element).firstChild;
+    while (child) {
+      if (child.nodeType === 1) {
+        results.push(Element.extend(child));
+      }
+      child = child.nextSibling;
+    }
+    return results;
+  },
+
+  previousSiblings: function(element, maximumLength) {
+    return Element.recursivelyCollect(element, 'previousSibling');
+  },
+
+  nextSiblings: function(element) {
+    return Element.recursivelyCollect(element, 'nextSibling');
+  },
+
+  siblings: function(element) {
+    element = $(element);
+    return Element.previousSiblings(element).reverse()
+      .concat(Element.nextSiblings(element));
+  },
+
+  match: function(element, selector) {
+    element = $(element);
+    if (Object.isString(selector))
+      return Prototype.Selector.match(element, selector);
+    return selector.match(element);
+  },
+
+  up: function(element, expression, index) {
+    element = $(element);
+    if (arguments.length == 1) return $(element.parentNode);
+    var ancestors = Element.ancestors(element);
+    return Object.isNumber(expression) ? ancestors[expression] :
+      Prototype.Selector.find(ancestors, expression, index);
+  },
+
+  down: function(element, expression, index) {
+    element = $(element);
+    if (arguments.length == 1) return Element.firstDescendant(element);
+    return Object.isNumber(expression) ? Element.descendants(element)[expression] :
+      Element.select(element, expression)[index || 0];
+  },
+
+  previous: function(element, expression, index) {
+    element = $(element);
+    if (Object.isNumber(expression)) index = expression, expression = false;
+    if (!Object.isNumber(index)) index = 0;
+
+    if (expression) {
+      return Prototype.Selector.find(element.previousSiblings(), expression, index);
+    } else {
+      return element.recursivelyCollect("previousSibling", index + 1)[index];
+    }
+  },
+
+  next: function(element, expression, index) {
+    element = $(element);
+    if (Object.isNumber(expression)) index = expression, expression = false;
+    if (!Object.isNumber(index)) index = 0;
+
+    if (expression) {
+      return Prototype.Selector.find(element.nextSiblings(), expression, index);
+    } else {
+      var maximumLength = Object.isNumber(index) ? index + 1 : 1;
+      return element.recursivelyCollect("nextSibling", index + 1)[index];
+    }
+  },
+
+
+  select: function(element) {
+    element = $(element);
+    var expressions = Array.prototype.slice.call(arguments, 1).join(', ');
+    return Prototype.Selector.select(expressions, element);
+  },
+
+  adjacent: function(element) {
+    element = $(element);
+    var expressions = Array.prototype.slice.call(arguments, 1).join(', ');
+    return Prototype.Selector.select(expressions, element.parentNode).without(element);
+  },
+
+  identify: function(element) {
+    element = $(element);
+    var id = Element.readAttribute(element, 'id');
+    if (id) return id;
+    do { id = 'anonymous_element_' + Element.idCounter++ } while ($(id));
+    Element.writeAttribute(element, 'id', id);
+    return id;
+  },
+
+  readAttribute: function(element, name) {
+    element = $(element);
+    if (Prototype.Browser.IE) {
+      var t = Element._attributeTranslations.read;
+      if (t.values[name]) return t.values[name](element, name);
+      if (t.names[name]) name = t.names[name];
+      if (name.include(':')) {
+        return (!element.attributes || !element.attributes[name]) ? null :
+         element.attributes[name].value;
+      }
+    }
+    return element.getAttribute(name);
+  },
+
+  writeAttribute: function(element, name, value) {
+    element = $(element);
+    var attributes = { }, t = Element._attributeTranslations.write;
+
+    if (typeof name == 'object') attributes = name;
+    else attributes[name] = Object.isUndefined(value) ? true : value;
+
+    for (var attr in attributes) {
+      name = t.names[attr] || attr;
+      value = attributes[attr];
+      if (t.values[attr]) name = t.values[attr](element, value);
+      if (value === false || value === null)
+        element.removeAttribute(name);
+      else if (value === true)
+        element.setAttribute(name, name);
+      else element.setAttribute(name, value);
+    }
+    return element;
+  },
+
+  getHeight: function(element) {
+    return Element.getDimensions(element).height;
+  },
+
+  getWidth: function(element) {
+    return Element.getDimensions(element).width;
+  },
+
+  classNames: function(element) {
+    return new Element.ClassNames(element);
+  },
+
+  hasClassName: function(element, className) {
+    if (!(element = $(element))) return;
+    var elementClassName = element.className;
+    return (elementClassName.length > 0 && (elementClassName == className ||
+      new RegExp("(^|\\s)" + className + "(\\s|$)").test(elementClassName)));
+  },
+
+  addClassName: function(element, className) {
+    if (!(element = $(element))) return;
+    if (!Element.hasClassName(element, className))
+      element.className += (element.className ? ' ' : '') + className;
+    return element;
+  },
+
+  removeClassName: function(element, className) {
+    if (!(element = $(element))) return;
+    element.className = element.className.replace(
+      new RegExp("(^|\\s+)" + className + "(\\s+|$)"), ' ').strip();
+    return element;
+  },
+
+  toggleClassName: function(element, className) {
+    if (!(element = $(element))) return;
+    return Element[Element.hasClassName(element, className) ?
+      'removeClassName' : 'addClassName'](element, className);
+  },
+
+  cleanWhitespace: function(element) {
+    element = $(element);
+    var node = element.firstChild;
+    while (node) {
+      var nextNode = node.nextSibling;
+      if (node.nodeType == 3 && !/\S/.test(node.nodeValue))
+        element.removeChild(node);
+      node = nextNode;
+    }
+    return element;
+  },
+
+  empty: function(element) {
+    return $(element).innerHTML.blank();
+  },
+
+  descendantOf: function(element, ancestor) {
+    element = $(element), ancestor = $(ancestor);
+
+    if (element.compareDocumentPosition)
+      return (element.compareDocumentPosition(ancestor) & 8) === 8;
+
+    if (ancestor.contains)
+      return ancestor.contains(element) && ancestor !== element;
+
+    while (element = element.parentNode)
+      if (element == ancestor) return true;
+
+    return false;
+  },
+
+  scrollTo: function(element) {
+    element = $(element);
+    var pos = Element.cumulativeOffset(element);
+    window.scrollTo(pos[0], pos[1]);
+    return element;
+  },
+
+  getStyle: function(element, style) {
+    element = $(element);
+    style = style == 'float' ? 'cssFloat' : style.camelize();
+    var value = element.style[style];
+    if (!value || value == 'auto') {
+      var css = document.defaultView.getComputedStyle(element, null);
+      value = css ? css[style] : null;
+    }
+    if (style == 'opacity') return value ? parseFloat(value) : 1.0;
+    return value == 'auto' ? null : value;
+  },
+
+  getOpacity: function(element) {
+    return $(element).getStyle('opacity');
+  },
+
+  setStyle: function(element, styles) {
+    element = $(element);
+    var elementStyle = element.style, match;
+    if (Object.isString(styles)) {
+      element.style.cssText += ';' + styles;
+      return styles.include('opacity') ?
+        element.setOpacity(styles.match(/opacity:\s*(\d?\.?\d*)/)[1]) : element;
+    }
+    for (var property in styles)
+      if (property == 'opacity') element.setOpacity(styles[property]);
+      else
+        elementStyle[(property == 'float' || property == 'cssFloat') ?
+          (Object.isUndefined(elementStyle.styleFloat) ? 'cssFloat' : 'styleFloat') :
+            property] = styles[property];
+
+    return element;
+  },
+
+  setOpacity: function(element, value) {
+    element = $(element);
+    element.style.opacity = (value == 1 || value === '') ? '' :
+      (value < 0.00001) ? 0 : value;
+    return element;
+  },
+
+  makePositioned: function(element) {
+    element = $(element);
+    var pos = Element.getStyle(element, 'position');
+    if (pos == 'static' || !pos) {
+      element._madePositioned = true;
+      element.style.position = 'relative';
+      if (Prototype.Browser.Opera) {
+        element.style.top = 0;
+        element.style.left = 0;
+      }
+    }
+    return element;
+  },
+
+  undoPositioned: function(element) {
+    element = $(element);
+    if (element._madePositioned) {
+      element._madePositioned = undefined;
+      element.style.position =
+        element.style.top =
+        element.style.left =
+        element.style.bottom =
+        element.style.right = '';
+    }
+    return element;
+  },
+
+  makeClipping: function(element) {
+    element = $(element);
+    if (element._overflow) return element;
+    element._overflow = Element.getStyle(element, 'overflow') || 'auto';
+    if (element._overflow !== 'hidden')
+      element.style.overflow = 'hidden';
+    return element;
+  },
+
+  undoClipping: function(element) {
+    element = $(element);
+    if (!element._overflow) return element;
+    element.style.overflow = element._overflow == 'auto' ? '' : element._overflow;
+    element._overflow = null;
+    return element;
+  },
+
+  clonePosition: function(element, source) {
+    var options = Object.extend({
+      setLeft:    true,
+      setTop:     true,
+      setWidth:   true,
+      setHeight:  true,
+      offsetTop:  0,
+      offsetLeft: 0
+    }, arguments[2] || { });
+
+    source = $(source);
+    var p = Element.viewportOffset(source), delta = [0, 0], parent = null;
+
+    element = $(element);
+
+    if (Element.getStyle(element, 'position') == 'absolute') {
+      parent = Element.getOffsetParent(element);
+      delta = Element.viewportOffset(parent);
+    }
+
+    if (parent == document.body) {
+      delta[0] -= document.body.offsetLeft;
+      delta[1] -= document.body.offsetTop;
+    }
+
+    if (options.setLeft)   element.style.left  = (p[0] - delta[0] + options.offsetLeft) + 'px';
+    if (options.setTop)    element.style.top   = (p[1] - delta[1] + options.offsetTop) + 'px';
+    if (options.setWidth)  element.style.width = source.offsetWidth + 'px';
+    if (options.setHeight) element.style.height = source.offsetHeight + 'px';
+    return element;
+  }
+};
+
+Object.extend(Element.Methods, {
+  getElementsBySelector: Element.Methods.select,
+
+  childElements: Element.Methods.immediateDescendants
+});
+
+Element._attributeTranslations = {
+  write: {
+    names: {
+      className: 'class',
+      htmlFor:   'for'
+    },
+    values: { }
+  }
+};
+
+if (Prototype.Browser.Opera) {
+  Element.Methods.getStyle = Element.Methods.getStyle.wrap(
+    function(proceed, element, style) {
+      switch (style) {
+        case 'height': case 'width':
+          if (!Element.visible(element)) return null;
+
+          var dim = parseInt(proceed(element, style), 10);
+
+          if (dim !== element['offset' + style.capitalize()])
+            return dim + 'px';
+
+          var properties;
+          if (style === 'height') {
+            properties = ['border-top-width', 'padding-top',
+             'padding-bottom', 'border-bottom-width'];
+          }
+          else {
+            properties = ['border-left-width', 'padding-left',
+             'padding-right', 'border-right-width'];
+          }
+          return properties.inject(dim, function(memo, property) {
+            var val = proceed(element, property);
+            return val === null ? memo : memo - parseInt(val, 10);
+          }) + 'px';
+        default: return proceed(element, style);
+      }
+    }
+  );
+
+  Element.Methods.readAttribute = Element.Methods.readAttribute.wrap(
+    function(proceed, element, attribute) {
+      if (attribute === 'title') return element.title;
+      return proceed(element, attribute);
+    }
+  );
+}
+
+else if (Prototype.Browser.IE) {
+  Element.Methods.getStyle = function(element, style) {
+    element = $(element);
+    style = (style == 'float' || style == 'cssFloat') ? 'styleFloat' : style.camelize();
+    var value = element.style[style];
+    if (!value && element.currentStyle) value = element.currentStyle[style];
+
+    if (style == 'opacity') {
+      if (value = (element.getStyle('filter') || '').match(/alpha\(opacity=(.*)\)/))
+        if (value[1]) return parseFloat(value[1]) / 100;
+      return 1.0;
+    }
+
+    if (value == 'auto') {
+      if ((style == 'width' || style == 'height') && (element.getStyle('display') != 'none'))
+        return element['offset' + style.capitalize()] + 'px';
+      return null;
+    }
+    return value;
+  };
+
+  Element.Methods.setOpacity = function(element, value) {
+    function stripAlpha(filter){
+      return filter.replace(/alpha\([^\)]*\)/gi,'');
+    }
+    element = $(element);
+    var currentStyle = element.currentStyle;
+    if ((currentStyle && !currentStyle.hasLayout) ||
+      (!currentStyle && element.style.zoom == 'normal'))
+        element.style.zoom = 1;
+
+    var filter = element.getStyle('filter'), style = element.style;
+    if (value == 1 || value === '') {
+      (filter = stripAlpha(filter)) ?
+        style.filter = filter : style.removeAttribute('filter');
+      return element;
+    } else if (value < 0.00001) value = 0;
+    style.filter = stripAlpha(filter) +
+      'alpha(opacity=' + (value * 100) + ')';
+    return element;
+  };
+
+  Element._attributeTranslations = (function(){
+
+    var classProp = 'className',
+        forProp = 'for',
+        el = document.createElement('div');
+
+    el.setAttribute(classProp, 'x');
+
+    if (el.className !== 'x') {
+      el.setAttribute('class', 'x');
+      if (el.className === 'x') {
+        classProp = 'class';
+      }
+    }
+    el = null;
+
+    el = document.createElement('label');
+    el.setAttribute(forProp, 'x');
+    if (el.htmlFor !== 'x') {
+      el.setAttribute('htmlFor', 'x');
+      if (el.htmlFor === 'x') {
+        forProp = 'htmlFor';
+      }
+    }
+    el = null;
+
+    return {
+      read: {
+        names: {
+          'class':      classProp,
+          'className':  classProp,
+          'for':        forProp,
+          'htmlFor':    forProp
+        },
+        values: {
+          _getAttr: function(element, attribute) {
+            return element.getAttribute(attribute);
+          },
+          _getAttr2: function(element, attribute) {
+            return element.getAttribute(attribute, 2);
+          },
+          _getAttrNode: function(element, attribute) {
+            var node = element.getAttributeNode(attribute);
+            return node ? node.value : "";
+          },
+          _getEv: (function(){
+
+            var el = document.createElement('div'), f;
+            el.onclick = Prototype.emptyFunction;
+            var value = el.getAttribute('onclick');
+
+            if (String(value).indexOf('{') > -1) {
+              f = function(element, attribute) {
+                attribute = element.getAttribute(attribute);
+                if (!attribute) return null;
+                attribute = attribute.toString();
+                attribute = attribute.split('{')[1];
+                attribute = attribute.split('}')[0];
+                return attribute.strip();
+              };
+            }
+            else if (value === '') {
+              f = function(element, attribute) {
+                attribute = element.getAttribute(attribute);
+                if (!attribute) return null;
+                return attribute.strip();
+              };
+            }
+            el = null;
+            return f;
+          })(),
+          _flag: function(element, attribute) {
+            return $(element).hasAttribute(attribute) ? attribute : null;
+          },
+          style: function(element) {
+            return element.style.cssText.toLowerCase();
+          },
+          title: function(element) {
+            return element.title;
+          }
+        }
+      }
+    }
+  })();
+
+  Element._attributeTranslations.write = {
+    names: Object.extend({
+      cellpadding: 'cellPadding',
+      cellspacing: 'cellSpacing'
+    }, Element._attributeTranslations.read.names),
+    values: {
+      checked: function(element, value) {
+        element.checked = !!value;
+      },
+
+      style: function(element, value) {
+        element.style.cssText = value ? value : '';
+      }
+    }
+  };
+
+  Element._attributeTranslations.has = {};
+
+  $w('colSpan rowSpan vAlign dateTime accessKey tabIndex ' +
+      'encType maxLength readOnly longDesc frameBorder').each(function(attr) {
+    Element._attributeTranslations.write.names[attr.toLowerCase()] = attr;
+    Element._attributeTranslations.has[attr.toLowerCase()] = attr;
+  });
+
+  (function(v) {
+    Object.extend(v, {
+      href:        v._getAttr2,
+      src:         v._getAttr2,
+      type:        v._getAttr,
+      action:      v._getAttrNode,
+      disabled:    v._flag,
+      checked:     v._flag,
+      readonly:    v._flag,
+      multiple:    v._flag,
+      onload:      v._getEv,
+      onunload:    v._getEv,
+      onclick:     v._getEv,
+      ondblclick:  v._getEv,
+      onmousedown: v._getEv,
+      onmouseup:   v._getEv,
+      onmouseover: v._getEv,
+      onmousemove: v._getEv,
+      onmouseout:  v._getEv,
+      onfocus:     v._getEv,
+      onblur:      v._getEv,
+      onkeypress:  v._getEv,
+      onkeydown:   v._getEv,
+      onkeyup:     v._getEv,
+      onsubmit:    v._getEv,
+      onreset:     v._getEv,
+      onselect:    v._getEv,
+      onchange:    v._getEv
+    });
+  })(Element._attributeTranslations.read.values);
+
+  if (Prototype.BrowserFeatures.ElementExtensions) {
+    (function() {
+      function _descendants(element) {
+        var nodes = element.getElementsByTagName('*'), results = [];
+        for (var i = 0, node; node = nodes[i]; i++)
+          if (node.tagName !== "!") // Filter out comment nodes.
+            results.push(node);
+        return results;
+      }
+
+      Element.Methods.down = function(element, expression, index) {
+        element = $(element);
+        if (arguments.length == 1) return element.firstDescendant();
+        return Object.isNumber(expression) ? _descendants(element)[expression] :
+          Element.select(element, expression)[index || 0];
+      }
+    })();
+  }
+
+}
+
+else if (Prototype.Browser.Gecko && /rv:1\.8\.0/.test(navigator.userAgent)) {
+  Element.Methods.setOpacity = function(element, value) {
+    element = $(element);
+    element.style.opacity = (value == 1) ? 0.999999 :
+      (value === '') ? '' : (value < 0.00001) ? 0 : value;
+    return element;
+  };
+}
+
+else if (Prototype.Browser.WebKit) {
+  Element.Methods.setOpacity = function(element, value) {
+    element = $(element);
+    element.style.opacity = (value == 1 || value === '') ? '' :
+      (value < 0.00001) ? 0 : value;
+
+    if (value == 1)
+      if (element.tagName.toUpperCase() == 'IMG' && element.width) {
+        element.width++; element.width--;
+      } else try {
+        var n = document.createTextNode(' ');
+        element.appendChild(n);
+        element.removeChild(n);
+      } catch (e) { }
+
+    return element;
+  };
+}
+
+if ('outerHTML' in document.documentElement) {
+  Element.Methods.replace = function(element, content) {
+    element = $(element);
+
+    if (content && content.toElement) content = content.toElement();
+    if (Object.isElement(content)) {
+      element.parentNode.replaceChild(content, element);
+      return element;
+    }
+
+    content = Object.toHTML(content);
+    var parent = element.parentNode, tagName = parent.tagName.toUpperCase();
+
+    if (Element._insertionTranslations.tags[tagName]) {
+      var nextSibling = element.next(),
+          fragments = Element._getContentFromAnonymousElement(tagName, content.stripScripts());
+      parent.removeChild(element);
+      if (nextSibling)
+        fragments.each(function(node) { parent.insertBefore(node, nextSibling) });
+      else
+        fragments.each(function(node) { parent.appendChild(node) });
+    }
+    else element.outerHTML = content.stripScripts();
+
+    content.evalScripts.bind(content).defer();
+    return element;
+  };
+}
+
+Element._returnOffset = function(l, t) {
+  var result = [l, t];
+  result.left = l;
+  result.top = t;
+  return result;
+};
+
+Element._getContentFromAnonymousElement = function(tagName, html, force) {
+  var div = new Element('div'),
+      t = Element._insertionTranslations.tags[tagName];
+
+  var workaround = false;
+  if (t) workaround = true;
+  else if (force) {
+    workaround = true;
+    t = ['', '', 0];
+  }
+
+  if (workaround) {
+    div.innerHTML = '&nbsp;' + t[0] + html + t[1];
+    div.removeChild(div.firstChild);
+    for (var i = t[2]; i--; ) {
+      div = div.firstChild;
+    }
+  }
+  else {
+    div.innerHTML = html;
+  }
+  return $A(div.childNodes);
+};
+
+Element._insertionTranslations = {
+  before: function(element, node) {
+    element.parentNode.insertBefore(node, element);
+  },
+  top: function(element, node) {
+    element.insertBefore(node, element.firstChild);
+  },
+  bottom: function(element, node) {
+    element.appendChild(node);
+  },
+  after: function(element, node) {
+    element.parentNode.insertBefore(node, element.nextSibling);
+  },
+  tags: {
+    TABLE:  ['<table>',                '</table>',                   1],
+    TBODY:  ['<table><tbody>',         '</tbody></table>',           2],
+    TR:     ['<table><tbody><tr>',     '</tr></tbody></table>',      3],
+    TD:     ['<table><tbody><tr><td>', '</td></tr></tbody></table>', 4],
+    SELECT: ['<select>',               '</select>',                  1]
+  }
+};
+
+(function() {
+  var tags = Element._insertionTranslations.tags;
+  Object.extend(tags, {
+    THEAD: tags.TBODY,
+    TFOOT: tags.TBODY,
+    TH:    tags.TD
+  });
+})();
+
+Element.Methods.Simulated = {
+  hasAttribute: function(element, attribute) {
+    attribute = Element._attributeTranslations.has[attribute] || attribute;
+    var node = $(element).getAttributeNode(attribute);
+    return !!(node && node.specified);
+  }
+};
+
+Element.Methods.ByTag = { };
+
+Object.extend(Element, Element.Methods);
+
+(function(div) {
+
+  if (!Prototype.BrowserFeatures.ElementExtensions && div['__proto__']) {
+    window.HTMLElement = { };
+    window.HTMLElement.prototype = div['__proto__'];
+    Prototype.BrowserFeatures.ElementExtensions = true;
+  }
+
+  div = null;
+
+})(document.createElement('div'));
+
+Element.extend = (function() {
+
+  function checkDeficiency(tagName) {
+    if (typeof window.Element != 'undefined') {
+      var proto = window.Element.prototype;
+      if (proto) {
+        var id = '_' + (Math.random()+'').slice(2),
+            el = document.createElement(tagName);
+        proto[id] = 'x';
+        var isBuggy = (el[id] !== 'x');
+        delete proto[id];
+        el = null;
+        return isBuggy;
+      }
+    }
+    return false;
+  }
+
+  function extendElementWith(element, methods) {
+    for (var property in methods) {
+      var value = methods[property];
+      if (Object.isFunction(value) && !(property in element))
+        element[property] = value.methodize();
+    }
+  }
+
+  var HTMLOBJECTELEMENT_PROTOTYPE_BUGGY = checkDeficiency('object');
+
+  if (Prototype.BrowserFeatures.SpecificElementExtensions) {
+    if (HTMLOBJECTELEMENT_PROTOTYPE_BUGGY) {
+      return function(element) {
+        if (element && typeof element._extendedByPrototype == 'undefined') {
+          var t = element.tagName;
+          if (t && (/^(?:object|applet|embed)$/i.test(t))) {
+            extendElementWith(element, Element.Methods);
+            extendElementWith(element, Element.Methods.Simulated);
+            extendElementWith(element, Element.Methods.ByTag[t.toUpperCase()]);
+          }
+        }
+        return element;
+      }
+    }
+    return Prototype.K;
+  }
+
+  var Methods = { }, ByTag = Element.Methods.ByTag;
+
+  var extend = Object.extend(function(element) {
+    if (!element || typeof element._extendedByPrototype != 'undefined' ||
+        element.nodeType != 1 || element == window) return element;
+
+    var methods = Object.clone(Methods),
+        tagName = element.tagName.toUpperCase();
+
+    if (ByTag[tagName]) Object.extend(methods, ByTag[tagName]);
+
+    extendElementWith(element, methods);
+
+    element._extendedByPrototype = Prototype.emptyFunction;
+    return element;
+
+  }, {
+    refresh: function() {
+      if (!Prototype.BrowserFeatures.ElementExtensions) {
+        Object.extend(Methods, Element.Methods);
+        Object.extend(Methods, Element.Methods.Simulated);
+      }
+    }
+  });
+
+  extend.refresh();
+  return extend;
+})();
+
+if (document.documentElement.hasAttribute) {
+  Element.hasAttribute = function(element, attribute) {
+    return element.hasAttribute(attribute);
+  };
+}
+else {
+  Element.hasAttribute = Element.Methods.Simulated.hasAttribute;
+}
+
+Element.addMethods = function(methods) {
+  var F = Prototype.BrowserFeatures, T = Element.Methods.ByTag;
+
+  if (!methods) {
+    Object.extend(Form, Form.Methods);
+    Object.extend(Form.Element, Form.Element.Methods);
+    Object.extend(Element.Methods.ByTag, {
+      "FORM":     Object.clone(Form.Methods),
+      "INPUT":    Object.clone(Form.Element.Methods),
+      "SELECT":   Object.clone(Form.Element.Methods),
+      "TEXTAREA": Object.clone(Form.Element.Methods),
+      "BUTTON":   Object.clone(Form.Element.Methods)
+    });
+  }
+
+  if (arguments.length == 2) {
+    var tagName = methods;
+    methods = arguments[1];
+  }
+
+  if (!tagName) Object.extend(Element.Methods, methods || { });
+  else {
+    if (Object.isArray(tagName)) tagName.each(extend);
+    else extend(tagName);
+  }
+
+  function extend(tagName) {
+    tagName = tagName.toUpperCase();
+    if (!Element.Methods.ByTag[tagName])
+      Element.Methods.ByTag[tagName] = { };
+    Object.extend(Element.Methods.ByTag[tagName], methods);
+  }
+
+  function copy(methods, destination, onlyIfAbsent) {
+    onlyIfAbsent = onlyIfAbsent || false;
+    for (var property in methods) {
+      var value = methods[property];
+      if (!Object.isFunction(value)) continue;
+      if (!onlyIfAbsent || !(property in destination))
+        destination[property] = value.methodize();
+    }
+  }
+
+  function findDOMClass(tagName) {
+    var klass;
+    var trans = {
+      "OPTGROUP": "OptGroup", "TEXTAREA": "TextArea", "P": "Paragraph",
+      "FIELDSET": "FieldSet", "UL": "UList", "OL": "OList", "DL": "DList",
+      "DIR": "Directory", "H1": "Heading", "H2": "Heading", "H3": "Heading",
+      "H4": "Heading", "H5": "Heading", "H6": "Heading", "Q": "Quote",
+      "INS": "Mod", "DEL": "Mod", "A": "Anchor", "IMG": "Image", "CAPTION":
+      "TableCaption", "COL": "TableCol", "COLGROUP": "TableCol", "THEAD":
+      "TableSection", "TFOOT": "TableSection", "TBODY": "TableSection", "TR":
+      "TableRow", "TH": "TableCell", "TD": "TableCell", "FRAMESET":
+      "FrameSet", "IFRAME": "IFrame"
+    };
+    if (trans[tagName]) klass = 'HTML' + trans[tagName] + 'Element';
+    if (window[klass]) return window[klass];
+    klass = 'HTML' + tagName + 'Element';
+    if (window[klass]) return window[klass];
+    klass = 'HTML' + tagName.capitalize() + 'Element';
+    if (window[klass]) return window[klass];
+
+    var element = document.createElement(tagName),
+        proto = element['__proto__'] || element.constructor.prototype;
+
+    element = null;
+    return proto;
+  }
+
+  var elementPrototype = window.HTMLElement ? HTMLElement.prototype :
+   Element.prototype;
+
+  if (F.ElementExtensions) {
+    copy(Element.Methods, elementPrototype);
+    copy(Element.Methods.Simulated, elementPrototype, true);
+  }
+
+  if (F.SpecificElementExtensions) {
+    for (var tag in Element.Methods.ByTag) {
+      var klass = findDOMClass(tag);
+      if (Object.isUndefined(klass)) continue;
+      copy(T[tag], klass.prototype);
+    }
+  }
+
+  Object.extend(Element, Element.Methods);
+  delete Element.ByTag;
+
+  if (Element.extend.refresh) Element.extend.refresh();
+  Element.cache = { };
+};
+
+
+document.viewport = {
+
+  getDimensions: function() {
+    return { width: this.getWidth(), height: this.getHeight() };
+  },
+
+  getScrollOffsets: function() {
+    return Element._returnOffset(
+      window.pageXOffset || document.documentElement.scrollLeft || document.body.scrollLeft,
+      window.pageYOffset || document.documentElement.scrollTop  || document.body.scrollTop);
+  }
+};
+
+(function(viewport) {
+  var B = Prototype.Browser, doc = document, element, property = {};
+
+  function getRootElement() {
+    if (B.WebKit && !doc.evaluate)
+      return document;
+
+    if (B.Opera && window.parseFloat(window.opera.version()) < 9.5)
+      return document.body;
+
+    return document.documentElement;
+  }
+
+  function define(D) {
+    if (!element) element = getRootElement();
+
+    property[D] = 'client' + D;
+
+    viewport['get' + D] = function() { return element[property[D]] };
+    return viewport['get' + D]();
+  }
+
+  viewport.getWidth  = define.curry('Width');
+
+  viewport.getHeight = define.curry('Height');
+})(document.viewport);
+
+
+Element.Storage = {
+  UID: 1
+};
+
+Element.addMethods({
+  getStorage: function(element) {
+    if (!(element = $(element))) return;
+
+    var uid;
+    if (element === window) {
+      uid = 0;
+    } else {
+      if (typeof element._prototypeUID === "undefined")
+        element._prototypeUID = Element.Storage.UID++;
+      uid = element._prototypeUID;
+    }
+
+    if (!Element.Storage[uid])
+      Element.Storage[uid] = $H();
+
+    return Element.Storage[uid];
+  },
+
+  store: function(element, key, value) {
+    if (!(element = $(element))) return;
+
+    if (arguments.length === 2) {
+      Element.getStorage(element).update(key);
+    } else {
+      Element.getStorage(element).set(key, value);
+    }
+
+    return element;
+  },
+
+  retrieve: function(element, key, defaultValue) {
+    if (!(element = $(element))) return;
+    var hash = Element.getStorage(element), value = hash.get(key);
+
+    if (Object.isUndefined(value)) {
+      hash.set(key, defaultValue);
+      value = defaultValue;
+    }
+
+    return value;
+  },
+
+  clone: function(element, deep) {
+    if (!(element = $(element))) return;
+    var clone = element.cloneNode(deep);
+    clone._prototypeUID = void 0;
+    if (deep) {
+      var descendants = Element.select(clone, '*'),
+          i = descendants.length;
+      while (i--) {
+        descendants[i]._prototypeUID = void 0;
+      }
+    }
+    return Element.extend(clone);
+  },
+
+  purge: function(element) {
+    if (!(element = $(element))) return;
+    var purgeElement = Element._purgeElement;
+
+    purgeElement(element);
+
+    var descendants = element.getElementsByTagName('*'),
+     i = descendants.length;
+
+    while (i--) purgeElement(descendants[i]);
+
+    return null;
+  }
+});
+
+(function() {
+
+  function toDecimal(pctString) {
+    var match = pctString.match(/^(\d+)%?$/i);
+    if (!match) return null;
+    return (Number(match[1]) / 100);
+  }
+
+  function getPixelValue(value, property, context) {
+    var element = null;
+    if (Object.isElement(value)) {
+      element = value;
+      value = element.getStyle(property);
+    }
+
+    if (value === null) {
+      return null;
+    }
+
+    if ((/^(?:-)?\d+(\.\d+)?(px)?$/i).test(value)) {
+      return window.parseFloat(value);
+    }
+
+    var isPercentage = value.include('%'), isViewport = (context === document.viewport);
+
+    if (/\d/.test(value) && element && element.runtimeStyle && !(isPercentage && isViewport)) {
+      var style = element.style.left, rStyle = element.runtimeStyle.left;
+      element.runtimeStyle.left = element.currentStyle.left;
+      element.style.left = value || 0;
+      value = element.style.pixelLeft;
+      element.style.left = style;
+      element.runtimeStyle.left = rStyle;
+
+      return value;
+    }
+
+    if (element && isPercentage) {
+      context = context || element.parentNode;
+      var decimal = toDecimal(value);
+      var whole = null;
+      var position = element.getStyle('position');
+
+      var isHorizontal = property.include('left') || property.include('right') ||
+       property.include('width');
+
+      var isVertical =  property.include('top') || property.include('bottom') ||
+        property.include('height');
+
+      if (context === document.viewport) {
+        if (isHorizontal) {
+          whole = document.viewport.getWidth();
+        } else if (isVertical) {
+          whole = document.viewport.getHeight();
+        }
+      } else {
+        if (isHorizontal) {
+          whole = $(context).measure('width');
+        } else if (isVertical) {
+          whole = $(context).measure('height');
+        }
+      }
+
+      return (whole === null) ? 0 : whole * decimal;
+    }
+
+    return 0;
+  }
+
+  function toCSSPixels(number) {
+    if (Object.isString(number) && number.endsWith('px')) {
+      return number;
+    }
+    return number + 'px';
+  }
+
+  function isDisplayed(element) {
+    var originalElement = element;
+    while (element && element.parentNode) {
+      var display = element.getStyle('display');
+      if (display === 'none') {
+        return false;
+      }
+      element = $(element.parentNode);
+    }
+    return true;
+  }
+
+  var hasLayout = Prototype.K;
+  if ('currentStyle' in document.documentElement) {
+    hasLayout = function(element) {
+      if (!element.currentStyle.hasLayout) {
+        element.style.zoom = 1;
+      }
+      return element;
+    };
+  }
+
+  function cssNameFor(key) {
+    if (key.include('border')) key = key + '-width';
+    return key.camelize();
+  }
+
+  Element.Layout = Class.create(Hash, {
+    initialize: function($super, element, preCompute) {
+      $super();
+      this.element = $(element);
+
+      Element.Layout.PROPERTIES.each( function(property) {
+        this._set(property, null);
+      }, this);
+
+      if (preCompute) {
+        this._preComputing = true;
+        this._begin();
+        Element.Layout.PROPERTIES.each( this._compute, this );
+        this._end();
+        this._preComputing = false;
+      }
+    },
+
+    _set: function(property, value) {
+      return Hash.prototype.set.call(this, property, value);
+    },
+
+    set: function(property, value) {
+      throw "Properties of Element.Layout are read-only.";
+    },
+
+    get: function($super, property) {
+      var value = $super(property);
+      return value === null ? this._compute(property) : value;
+    },
+
+    _begin: function() {
+      if (this._prepared) return;
+
+      var element = this.element;
+      if (isDisplayed(element)) {
+        this._prepared = true;
+        return;
+      }
+
+      var originalStyles = {
+        position:   element.style.position   || '',
+        width:      element.style.width      || '',
+        visibility: element.style.visibility || '',
+        display:    element.style.display    || ''
+      };
+
+      element.store('prototype_original_styles', originalStyles);
+
+      var position = element.getStyle('position'),
+       width = element.getStyle('width');
+
+      if (width === "0px" || width === null) {
+        element.style.display = 'block';
+        width = element.getStyle('width');
+      }
+
+      var context = (position === 'fixed') ? document.viewport :
+       element.parentNode;
+
+      element.setStyle({
+        position:   'absolute',
+        visibility: 'hidden',
+        display:    'block'
+      });
+
+      var positionedWidth = element.getStyle('width');
+
+      var newWidth;
+      if (width && (positionedWidth === width)) {
+        newWidth = getPixelValue(element, 'width', context);
+      } else if (position === 'absolute' || position === 'fixed') {
+        newWidth = getPixelValue(element, 'width', context);
+      } else {
+        var parent = element.parentNode, pLayout = $(parent).getLayout();
+
+        newWidth = pLayout.get('width') -
+         this.get('margin-left') -
+         this.get('border-left') -
+         this.get('padding-left') -
+         this.get('padding-right') -
+         this.get('border-right') -
+         this.get('margin-right');
+      }
+
+      element.setStyle({ width: newWidth + 'px' });
+
+      this._prepared = true;
+    },
+
+    _end: function() {
+      var element = this.element;
+      var originalStyles = element.retrieve('prototype_original_styles');
+      element.store('prototype_original_styles', null);
+      element.setStyle(originalStyles);
+      this._prepared = false;
+    },
+
+    _compute: function(property) {
+      var COMPUTATIONS = Element.Layout.COMPUTATIONS;
+      if (!(property in COMPUTATIONS)) {
+        throw "Property not found.";
+      }
+
+      return this._set(property, COMPUTATIONS[property].call(this, this.element));
+    },
+
+    toObject: function() {
+      var args = $A(arguments);
+      var keys = (args.length === 0) ? Element.Layout.PROPERTIES :
+       args.join(' ').split(' ');
+      var obj = {};
+      keys.each( function(key) {
+        if (!Element.Layout.PROPERTIES.include(key)) return;
+        var value = this.get(key);
+        if (value != null) obj[key] = value;
+      }, this);
+      return obj;
+    },
+
+    toHash: function() {
+      var obj = this.toObject.apply(this, arguments);
+      return new Hash(obj);
+    },
+
+    toCSS: function() {
+      var args = $A(arguments);
+      var keys = (args.length === 0) ? Element.Layout.PROPERTIES :
+       args.join(' ').split(' ');
+      var css = {};
+
+      keys.each( function(key) {
+        if (!Element.Layout.PROPERTIES.include(key)) return;
+        if (Element.Layout.COMPOSITE_PROPERTIES.include(key)) return;
+
+        var value = this.get(key);
+        if (value != null) css[cssNameFor(key)] = value + 'px';
+      }, this);
+      return css;
+    },
+
+    inspect: function() {
+      return "#<Element.Layout>";
+    }
+  });
+
+  Object.extend(Element.Layout, {
+    PROPERTIES: $w('height width top left right bottom border-left border-right border-top border-bottom padding-left padding-right padding-top padding-bottom margin-top margin-bottom margin-left margin-right padding-box-width padding-box-height border-box-width border-box-height margin-box-width margin-box-height'),
+
+    COMPOSITE_PROPERTIES: $w('padding-box-width padding-box-height margin-box-width margin-box-height border-box-width border-box-height'),
+
+    COMPUTATIONS: {
+      'height': function(element) {
+        if (!this._preComputing) this._begin();
+
+        var bHeight = this.get('border-box-height');
+        if (bHeight <= 0) {
+          if (!this._preComputing) this._end();
+          return 0;
+        }
+
+        var bTop = this.get('border-top'),
+         bBottom = this.get('border-bottom');
+
+        var pTop = this.get('padding-top'),
+         pBottom = this.get('padding-bottom');
+
+        if (!this._preComputing) this._end();
+
+        return bHeight - bTop - bBottom - pTop - pBottom;
+      },
+
+      'width': function(element) {
+        if (!this._preComputing) this._begin();
+
+        var bWidth = this.get('border-box-width');
+        if (bWidth <= 0) {
+          if (!this._preComputing) this._end();
+          return 0;
+        }
+
+        var bLeft = this.get('border-left'),
+         bRight = this.get('border-right');
+
+        var pLeft = this.get('padding-left'),
+         pRight = this.get('padding-right');
+
+        if (!this._preComputing) this._end();
+
+        return bWidth - bLeft - bRight - pLeft - pRight;
+      },
+
+      'padding-box-height': function(element) {
+        var height = this.get('height'),
+         pTop = this.get('padding-top'),
+         pBottom = this.get('padding-bottom');
+
+        return height + pTop + pBottom;
+      },
+
+      'padding-box-width': function(element) {
+        var width = this.get('width'),
+         pLeft = this.get('padding-left'),
+         pRight = this.get('padding-right');
+
+        return width + pLeft + pRight;
+      },
+
+      'border-box-height': function(element) {
+        if (!this._preComputing) this._begin();
+        var height = element.offsetHeight;
+        if (!this._preComputing) this._end();
+        return height;
+      },
+
+      'border-box-width': function(element) {
+        if (!this._preComputing) this._begin();
+        var width = element.offsetWidth;
+        if (!this._preComputing) this._end();
+        return width;
+      },
+
+      'margin-box-height': function(element) {
+        var bHeight = this.get('border-box-height'),
+         mTop = this.get('margin-top'),
+         mBottom = this.get('margin-bottom');
+
+        if (bHeight <= 0) return 0;
+
+        return bHeight + mTop + mBottom;
+      },
+
+      'margin-box-width': function(element) {
+        var bWidth = this.get('border-box-width'),
+         mLeft = this.get('margin-left'),
+         mRight = this.get('margin-right');
+
+        if (bWidth <= 0) return 0;
+
+        return bWidth + mLeft + mRight;
+      },
+
+      'top': function(element) {
+        var offset = element.positionedOffset();
+        return offset.top;
+      },
+
+      'bottom': function(element) {
+        var offset = element.positionedOffset(),
+         parent = element.getOffsetParent(),
+         pHeight = parent.measure('height');
+
+        var mHeight = this.get('border-box-height');
+
+        return pHeight - mHeight - offset.top;
+      },
+
+      'left': function(element) {
+        var offset = element.positionedOffset();
+        return offset.left;
+      },
+
+      'right': function(element) {
+        var offset = element.positionedOffset(),
+         parent = element.getOffsetParent(),
+         pWidth = parent.measure('width');
+
+        var mWidth = this.get('border-box-width');
+
+        return pWidth - mWidth - offset.left;
+      },
+
+      'padding-top': function(element) {
+        return getPixelValue(element, 'paddingTop');
+      },
+
+      'padding-bottom': function(element) {
+        return getPixelValue(element, 'paddingBottom');
+      },
+
+      'padding-left': function(element) {
+        return getPixelValue(element, 'paddingLeft');
+      },
+
+      'padding-right': function(element) {
+        return getPixelValue(element, 'paddingRight');
+      },
+
+      'border-top': function(element) {
+        return getPixelValue(element, 'borderTopWidth');
+      },
+
+      'border-bottom': function(element) {
+        return getPixelValue(element, 'borderBottomWidth');
+      },
+
+      'border-left': function(element) {
+        return getPixelValue(element, 'borderLeftWidth');
+      },
+
+      'border-right': function(element) {
+        return getPixelValue(element, 'borderRightWidth');
+      },
+
+      'margin-top': function(element) {
+        return getPixelValue(element, 'marginTop');
+      },
+
+      'margin-bottom': function(element) {
+        return getPixelValue(element, 'marginBottom');
+      },
+
+      'margin-left': function(element) {
+        return getPixelValue(element, 'marginLeft');
+      },
+
+      'margin-right': function(element) {
+        return getPixelValue(element, 'marginRight');
+      }
+    }
+  });
+
+  if ('getBoundingClientRect' in document.documentElement) {
+    Object.extend(Element.Layout.COMPUTATIONS, {
+      'right': function(element) {
+        var parent = hasLayout(element.getOffsetParent());
+        var rect = element.getBoundingClientRect(),
+         pRect = parent.getBoundingClientRect();
+
+        return (pRect.right - rect.right).round();
+      },
+
+      'bottom': function(element) {
+        var parent = hasLayout(element.getOffsetParent());
+        var rect = element.getBoundingClientRect(),
+         pRect = parent.getBoundingClientRect();
+
+        return (pRect.bottom - rect.bottom).round();
+      }
+    });
+  }
+
+  Element.Offset = Class.create({
+    initialize: function(left, top) {
+      this.left = left.round();
+      this.top  = top.round();
+
+      this[0] = this.left;
+      this[1] = this.top;
+    },
+
+    relativeTo: function(offset) {
+      return new Element.Offset(
+        this.left - offset.left,
+        this.top  - offset.top
+      );
+    },
+
+    inspect: function() {
+      return "#<Element.Offset left: #{left} top: #{top}>".interpolate(this);
+    },
+
+    toString: function() {
+      return "[#{left}, #{top}]".interpolate(this);
+    },
+
+    toArray: function() {
+      return [this.left, this.top];
+    }
+  });
+
+  function getLayout(element, preCompute) {
+    return new Element.Layout(element, preCompute);
+  }
+
+  function measure(element, property) {
+    return $(element).getLayout().get(property);
+  }
+
+  function getDimensions(element) {
+    element = $(element);
+    var display = Element.getStyle(element, 'display');
+
+    if (display && display !== 'none') {
+      return { width: element.offsetWidth, height: element.offsetHeight };
+    }
+
+    var style = element.style;
+    var originalStyles = {
+      visibility: style.visibility,
+      position:   style.position,
+      display:    style.display
+    };
+
+    var newStyles = {
+      visibility: 'hidden',
+      display:    'block'
+    };
+
+    if (originalStyles.position !== 'fixed')
+      newStyles.position = 'absolute';
+
+    Element.setStyle(element, newStyles);
+
+    var dimensions = {
+      width:  element.offsetWidth,
+      height: element.offsetHeight
+    };
+
+    Element.setStyle(element, originalStyles);
+
+    return dimensions;
+  }
+
+  function getOffsetParent(element) {
+    element = $(element);
+
+    if (isDocument(element) || isDetached(element) || isBody(element) || isHtml(element))
+      return $(document.body);
+
+    var isInline = (Element.getStyle(element, 'display') === 'inline');
+    if (!isInline && element.offsetParent) return $(element.offsetParent);
+
+    while ((element = element.parentNode) && element !== document.body) {
+      if (Element.getStyle(element, 'position') !== 'static') {
+        return isHtml(element) ? $(document.body) : $(element);
+      }
+    }
+
+    return $(document.body);
+  }
+
+
+  function cumulativeOffset(element) {
+    element = $(element);
+    var valueT = 0, valueL = 0;
+    if (element.parentNode) {
+      do {
+        valueT += element.offsetTop  || 0;
+        valueL += element.offsetLeft || 0;
+        element = element.offsetParent;
+      } while (element);
+    }
+    return new Element.Offset(valueL, valueT);
+  }
+
+  function positionedOffset(element) {
+    element = $(element);
+
+    var layout = element.getLayout();
+
+    var valueT = 0, valueL = 0;
+    do {
+      valueT += element.offsetTop  || 0;
+      valueL += element.offsetLeft || 0;
+      element = element.offsetParent;
+      if (element) {
+        if (isBody(element)) break;
+        var p = Element.getStyle(element, 'position');
+        if (p !== 'static') break;
+      }
+    } while (element);
+
+    valueL -= layout.get('margin-top');
+    valueT -= layout.get('margin-left');
+
+    return new Element.Offset(valueL, valueT);
+  }
+
+  function cumulativeScrollOffset(element) {
+    var valueT = 0, valueL = 0;
+    do {
+      valueT += element.scrollTop  || 0;
+      valueL += element.scrollLeft || 0;
+      element = element.parentNode;
+    } while (element);
+    return new Element.Offset(valueL, valueT);
+  }
+
+  function viewportOffset(forElement) {
+    element = $(element);
+    var valueT = 0, valueL = 0, docBody = document.body;
+
+    var element = forElement;
+    do {
+      valueT += element.offsetTop  || 0;
+      valueL += element.offsetLeft || 0;
+      if (element.offsetParent == docBody &&
+        Element.getStyle(element, 'position') == 'absolute') break;
+    } while (element = element.offsetParent);
+
+    element = forElement;
+    do {
+      if (element != docBody) {
+        valueT -= element.scrollTop  || 0;
+        valueL -= element.scrollLeft || 0;
+      }
+    } while (element = element.parentNode);
+    return new Element.Offset(valueL, valueT);
+  }
+
+  function absolutize(element) {
+    element = $(element);
+
+    if (Element.getStyle(element, 'position') === 'absolute') {
+      return element;
+    }
+
+    var offsetParent = getOffsetParent(element);
+    var eOffset = element.viewportOffset(),
+     pOffset = offsetParent.viewportOffset();
+
+    var offset = eOffset.relativeTo(pOffset);
+    var layout = element.getLayout();
+
+    element.store('prototype_absolutize_original_styles', {
+      left:   element.getStyle('left'),
+      top:    element.getStyle('top'),
+      width:  element.getStyle('width'),
+      height: element.getStyle('height')
+    });
+
+    element.setStyle({
+      position: 'absolute',
+      top:    offset.top + 'px',
+      left:   offset.left + 'px',
+      width:  layout.get('width') + 'px',
+      height: layout.get('height') + 'px'
+    });
+
+    return element;
+  }
+
+  function relativize(element) {
+    element = $(element);
+    if (Element.getStyle(element, 'position') === 'relative') {
+      return element;
+    }
+
+    var originalStyles =
+     element.retrieve('prototype_absolutize_original_styles');
+
+    if (originalStyles) element.setStyle(originalStyles);
+    return element;
+  }
+
+  if (Prototype.Browser.IE) {
+    getOffsetParent = getOffsetParent.wrap(
+      function(proceed, element) {
+        element = $(element);
+
+        if (isDocument(element) || isDetached(element) || isBody(element) || isHtml(element))
+          return $(document.body);
+
+        var position = element.getStyle('position');
+        if (position !== 'static') return proceed(element);
+
+        element.setStyle({ position: 'relative' });
+        var value = proceed(element);
+        element.setStyle({ position: position });
+        return value;
+      }
+    );
+
+    positionedOffset = positionedOffset.wrap(function(proceed, element) {
+      element = $(element);
+      if (!element.parentNode) return new Element.Offset(0, 0);
+      var position = element.getStyle('position');
+      if (position !== 'static') return proceed(element);
+
+      var offsetParent = element.getOffsetParent();
+      if (offsetParent && offsetParent.getStyle('position') === 'fixed')
+        hasLayout(offsetParent);
+
+      element.setStyle({ position: 'relative' });
+      var value = proceed(element);
+      element.setStyle({ position: position });
+      return value;
+    });
+  } else if (Prototype.Browser.Webkit) {
+    cumulativeOffset = function(element) {
+      element = $(element);
+      var valueT = 0, valueL = 0;
+      do {
+        valueT += element.offsetTop  || 0;
+        valueL += element.offsetLeft || 0;
+        if (element.offsetParent == document.body)
+          if (Element.getStyle(element, 'position') == 'absolute') break;
+
+        element = element.offsetParent;
+      } while (element);
+
+      return new Element.Offset(valueL, valueT);
+    };
+  }
+
+
+  Element.addMethods({
+    getLayout:              getLayout,
+    measure:                measure,
+    getDimensions:          getDimensions,
+    getOffsetParent:        getOffsetParent,
+    cumulativeOffset:       cumulativeOffset,
+    positionedOffset:       positionedOffset,
+    cumulativeScrollOffset: cumulativeScrollOffset,
+    viewportOffset:         viewportOffset,
+    absolutize:             absolutize,
+    relativize:             relativize
+  });
+
+  function isBody(element) {
+    return element.nodeName.toUpperCase() === 'BODY';
+  }
+
+  function isHtml(element) {
+    return element.nodeName.toUpperCase() === 'HTML';
+  }
+
+  function isDocument(element) {
+    return element.nodeType === Node.DOCUMENT_NODE;
+  }
+
+  function isDetached(element) {
+    return element !== document.body &&
+     !Element.descendantOf(element, document.body);
+  }
+
+  if ('getBoundingClientRect' in document.documentElement) {
+    Element.addMethods({
+      viewportOffset: function(element) {
+        element = $(element);
+        if (isDetached(element)) return new Element.Offset(0, 0);
+
+        var rect = element.getBoundingClientRect(),
+         docEl = document.documentElement;
+        return new Element.Offset(rect.left - docEl.clientLeft,
+         rect.top - docEl.clientTop);
+      }
+    });
+  }
+})();
+window.$$ = function() {
+  var expression = $A(arguments).join(', ');
+  return Prototype.Selector.select(expression, document);
+};
+
+Prototype.Selector = (function() {
+
+  function select() {
+    throw new Error('Method "Prototype.Selector.select" must be defined.');
+  }
+
+  function match() {
+    throw new Error('Method "Prototype.Selector.match" must be defined.');
+  }
+
+  function find(elements, expression, index) {
+    index = index || 0;
+    var match = Prototype.Selector.match, length = elements.length, matchIndex = 0, i;
+
+    for (i = 0; i < length; i++) {
+      if (match(elements[i], expression) && index == matchIndex++) {
+        return Element.extend(elements[i]);
+      }
+    }
+  }
+
+  function extendElements(elements) {
+    for (var i = 0, length = elements.length; i < length; i++) {
+      Element.extend(elements[i]);
+    }
+    return elements;
+  }
+
+
+  var K = Prototype.K;
+
+  return {
+    select: select,
+    match: match,
+    find: find,
+    extendElements: (Element.extend === K) ? K : extendElements,
+    extendElement: Element.extend
+  };
+})();
+Prototype._original_property = window.Sizzle;
+/*!
+ * Sizzle CSS Selector Engine - v1.0
+ *  Copyright 2009, The Dojo Foundation
+ *  Released under the MIT, BSD, and GPL Licenses.
+ *  More information: http://sizzlejs.com/
+ */
+(function(){
+
+var chunker = /((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^[\]]*\]|['"][^'"]*['"]|[^[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,
+       done = 0,
+       toString = Object.prototype.toString,
+       hasDuplicate = false,
+       baseHasDuplicate = true;
+
+[0, 0].sort(function(){
+       baseHasDuplicate = false;
+       return 0;
+});
+
+var Sizzle = function(selector, context, results, seed) {
+       results = results || [];
+       var origContext = context = context || document;
+
+       if ( context.nodeType !== 1 && context.nodeType !== 9 ) {
+               return [];
+       }
+
+       if ( !selector || typeof selector !== "string" ) {
+               return results;
+       }
+
+       var parts = [], m, set, checkSet, check, mode, extra, prune = true, contextXML = isXML(context),
+               soFar = selector;
+
+       while ( (chunker.exec(""), m = chunker.exec(soFar)) !== null ) {
+               soFar = m[3];
+
+               parts.push( m[1] );
+
+               if ( m[2] ) {
+                       extra = m[3];
+                       break;
+               }
+       }
+
+       if ( parts.length > 1 && origPOS.exec( selector ) ) {
+               if ( parts.length === 2 && Expr.relative[ parts[0] ] ) {
+                       set = posProcess( parts[0] + parts[1], context );
+               } else {
+                       set = Expr.relative[ parts[0] ] ?
+                               [ context ] :
+                               Sizzle( parts.shift(), context );
+
+                       while ( parts.length ) {
+                               selector = parts.shift();
+
+                               if ( Expr.relative[ selector ] )
+                                       selector += parts.shift();
+
+                               set = posProcess( selector, set );
+                       }
+               }
+       } else {
+               if ( !seed && parts.length > 1 && context.nodeType === 9 && !contextXML &&
+                               Expr.match.ID.test(parts[0]) && !Expr.match.ID.test(parts[parts.length - 1]) ) {
+                       var ret = Sizzle.find( parts.shift(), context, contextXML );
+                       context = ret.expr ? Sizzle.filter( ret.expr, ret.set )[0] : ret.set[0];
+               }
+
+               if ( context ) {
+                       var ret = seed ?
+                               { expr: parts.pop(), set: makeArray(seed) } :
+                               Sizzle.find( parts.pop(), parts.length === 1 && (parts[0] === "~" || parts[0] === "+") && context.parentNode ? context.parentNode : context, contextXML );
+                       set = ret.expr ? Sizzle.filter( ret.expr, ret.set ) : ret.set;
+
+                       if ( parts.length > 0 ) {
+                               checkSet = makeArray(set);
+                       } else {
+                               prune = false;
+                       }
+
+                       while ( parts.length ) {
+                               var cur = parts.pop(), pop = cur;
+
+                               if ( !Expr.relative[ cur ] ) {
+                                       cur = "";
+                               } else {
+                                       pop = parts.pop();
+                               }
+
+                               if ( pop == null ) {
+                                       pop = context;
+                               }
+
+                               Expr.relative[ cur ]( checkSet, pop, contextXML );
+                       }
+               } else {
+                       checkSet = parts = [];
+               }
+       }
+
+       if ( !checkSet ) {
+               checkSet = set;
+       }
+
+       if ( !checkSet ) {
+               throw "Syntax error, unrecognized expression: " + (cur || selector);
+       }
+
+       if ( toString.call(checkSet) === "[object Array]" ) {
+               if ( !prune ) {
+                       results.push.apply( results, checkSet );
+               } else if ( context && context.nodeType === 1 ) {
+                       for ( var i = 0; checkSet[i] != null; i++ ) {
+                               if ( checkSet[i] && (checkSet[i] === true || checkSet[i].nodeType === 1 && contains(context, checkSet[i])) ) {
+                                       results.push( set[i] );
+                               }
+                       }
+               } else {
+                       for ( var i = 0; checkSet[i] != null; i++ ) {
+                               if ( checkSet[i] && checkSet[i].nodeType === 1 ) {
+                                       results.push( set[i] );
+                               }
+                       }
+               }
+       } else {
+               makeArray( checkSet, results );
+       }
+
+       if ( extra ) {
+               Sizzle( extra, origContext, results, seed );
+               Sizzle.uniqueSort( results );
+       }
+
+       return results;
+};
+
+Sizzle.uniqueSort = function(results){
+       if ( sortOrder ) {
+               hasDuplicate = baseHasDuplicate;
+               results.sort(sortOrder);
+
+               if ( hasDuplicate ) {
+                       for ( var i = 1; i < results.length; i++ ) {
+                               if ( results[i] === results[i-1] ) {
+                                       results.splice(i--, 1);
+                               }
+                       }
+               }
+       }
+
+       return results;
+};
+
+Sizzle.matches = function(expr, set){
+       return Sizzle(expr, null, null, set);
+};
+
+Sizzle.find = function(expr, context, isXML){
+       var set, match;
+
+       if ( !expr ) {
+               return [];
+       }
+
+       for ( var i = 0, l = Expr.order.length; i < l; i++ ) {
+               var type = Expr.order[i], match;
+
+               if ( (match = Expr.leftMatch[ type ].exec( expr )) ) {
+                       var left = match[1];
+                       match.splice(1,1);
+
+                       if ( left.substr( left.length - 1 ) !== "\\" ) {
+                               match[1] = (match[1] || "").replace(/\\/g, "");
+                               set = Expr.find[ type ]( match, context, isXML );
+                               if ( set != null ) {
+                                       expr = expr.replace( Expr.match[ type ], "" );
+                                       break;
+                               }
+                       }
+               }
+       }
+
+       if ( !set ) {
+               set = context.getElementsByTagName("*");
+       }
+
+       return {set: set, expr: expr};
+};
+
+Sizzle.filter = function(expr, set, inplace, not){
+       var old = expr, result = [], curLoop = set, match, anyFound,
+               isXMLFilter = set && set[0] && isXML(set[0]);
+
+       while ( expr && set.length ) {
+               for ( var type in Expr.filter ) {
+                       if ( (match = Expr.match[ type ].exec( expr )) != null ) {
+                               var filter = Expr.filter[ type ], found, item;
+                               anyFound = false;
+
+                               if ( curLoop == result ) {
+                                       result = [];
+                               }
+
+                               if ( Expr.preFilter[ type ] ) {
+                                       match = Expr.preFilter[ type ]( match, curLoop, inplace, result, not, isXMLFilter );
+
+                                       if ( !match ) {
+                                               anyFound = found = true;
+                                       } else if ( match === true ) {
+                                               continue;
+                                       }
+                               }
+
+                               if ( match ) {
+                                       for ( var i = 0; (item = curLoop[i]) != null; i++ ) {
+                                               if ( item ) {
+                                                       found = filter( item, match, i, curLoop );
+                                                       var pass = not ^ !!found;
+
+                                                       if ( inplace && found != null ) {
+                                                               if ( pass ) {
+                                                                       anyFound = true;
+                                                               } else {
+                                                                       curLoop[i] = false;
+                                                               }
+                                                       } else if ( pass ) {
+                                                               result.push( item );
+                                                               anyFound = true;
+                                                       }
+                                               }
+                                       }
+                               }
+
+                               if ( found !== undefined ) {
+                                       if ( !inplace ) {
+                                               curLoop = result;
+                                       }
+
+                                       expr = expr.replace( Expr.match[ type ], "" );
+
+                                       if ( !anyFound ) {
+                                               return [];
+                                       }
+
+                                       break;
+                               }
+                       }
+               }
+
+               if ( expr == old ) {
+                       if ( anyFound == null ) {
+                               throw "Syntax error, unrecognized expression: " + expr;
+                       } else {
+                               break;
+                       }
+               }
+
+               old = expr;
+       }
+
+       return curLoop;
+};
+
+var Expr = Sizzle.selectors = {
+       order: [ "ID", "NAME", "TAG" ],
+       match: {
+               ID: /#((?:[\w\u00c0-\uFFFF-]|\\.)+)/,
+               CLASS: /\.((?:[\w\u00c0-\uFFFF-]|\\.)+)/,
+               NAME: /\[name=['"]*((?:[\w\u00c0-\uFFFF-]|\\.)+)['"]*\]/,
+               ATTR: /\[\s*((?:[\w\u00c0-\uFFFF-]|\\.)+)\s*(?:(\S?=)\s*(['"]*)(.*?)\3|)\s*\]/,
+               TAG: /^((?:[\w\u00c0-\uFFFF\*-]|\\.)+)/,
+               CHILD: /:(only|nth|last|first)-child(?:\((even|odd|[\dn+-]*)\))?/,
+               POS: /:(nth|eq|gt|lt|first|last|even|odd)(?:\((\d*)\))?(?=[^-]|$)/,
+               PSEUDO: /:((?:[\w\u00c0-\uFFFF-]|\\.)+)(?:\((['"]*)((?:\([^\)]+\)|[^\2\(\)]*)+)\2\))?/
+       },
+       leftMatch: {},
+       attrMap: {
+               "class": "className",
+               "for": "htmlFor"
+       },
+       attrHandle: {
+               href: function(elem){
+                       return elem.getAttribute("href");
+               }
+       },
+       relative: {
+               "+": function(checkSet, part, isXML){
+                       var isPartStr = typeof part === "string",
+                               isTag = isPartStr && !/\W/.test(part),
+                               isPartStrNotTag = isPartStr && !isTag;
+
+                       if ( isTag && !isXML ) {
+                               part = part.toUpperCase();
+                       }
+
+                       for ( var i = 0, l = checkSet.length, elem; i < l; i++ ) {
+                               if ( (elem = checkSet[i]) ) {
+                                       while ( (elem = elem.previousSibling) && elem.nodeType !== 1 ) {}
+
+                                       checkSet[i] = isPartStrNotTag || elem && elem.nodeName === part ?
+                                               elem || false :
+                                               elem === part;
+                               }
+                       }
+
+                       if ( isPartStrNotTag ) {
+                               Sizzle.filter( part, checkSet, true );
+                       }
+               },
+               ">": function(checkSet, part, isXML){
+                       var isPartStr = typeof part === "string";
+
+                       if ( isPartStr && !/\W/.test(part) ) {
+                               part = isXML ? part : part.toUpperCase();
+
+                               for ( var i = 0, l = checkSet.length; i < l; i++ ) {
+                                       var elem = checkSet[i];
+                                       if ( elem ) {
+                                               var parent = elem.parentNode;
+                                               checkSet[i] = parent.nodeName === part ? parent : false;
+                                       }
+                               }
+                       } else {
+                               for ( var i = 0, l = checkSet.length; i < l; i++ ) {
+                                       var elem = checkSet[i];
+                                       if ( elem ) {
+                                               checkSet[i] = isPartStr ?
+                                                       elem.parentNode :
+                                                       elem.parentNode === part;
+                                       }
+                               }
+
+                               if ( isPartStr ) {
+                                       Sizzle.filter( part, checkSet, true );
+                               }
+                       }
+               },
+               "": function(checkSet, part, isXML){
+                       var doneName = done++, checkFn = dirCheck;
+
+                       if ( !/\W/.test(part) ) {
+                               var nodeCheck = part = isXML ? part : part.toUpperCase();
+                               checkFn = dirNodeCheck;
+                       }
+
+                       checkFn("parentNode", part, doneName, checkSet, nodeCheck, isXML);
+               },
+               "~": function(checkSet, part, isXML){
+                       var doneName = done++, checkFn = dirCheck;
+
+                       if ( typeof part === "string" && !/\W/.test(part) ) {
+                               var nodeCheck = part = isXML ? part : part.toUpperCase();
+                               checkFn = dirNodeCheck;
+                       }
+
+                       checkFn("previousSibling", part, doneName, checkSet, nodeCheck, isXML);
+               }
+       },
+       find: {
+               ID: function(match, context, isXML){
+                       if ( typeof context.getElementById !== "undefined" && !isXML ) {
+                               var m = context.getElementById(match[1]);
+                               return m ? [m] : [];
+                       }
+               },
+               NAME: function(match, context, isXML){
+                       if ( typeof context.getElementsByName !== "undefined" ) {
+                               var ret = [], results = context.getElementsByName(match[1]);
+
+                               for ( var i = 0, l = results.length; i < l; i++ ) {
+                                       if ( results[i].getAttribute("name") === match[1] ) {
+                                               ret.push( results[i] );
+                                       }
+                               }
+
+                               return ret.length === 0 ? null : ret;
+                       }
+               },
+               TAG: function(match, context){
+                       return context.getElementsByTagName(match[1]);
+               }
+       },
+       preFilter: {
+               CLASS: function(match, curLoop, inplace, result, not, isXML){
+                       match = " " + match[1].replace(/\\/g, "") + " ";
+
+                       if ( isXML ) {
+                               return match;
+                       }
+
+                       for ( var i = 0, elem; (elem = curLoop[i]) != null; i++ ) {
+                               if ( elem ) {
+                                       if ( not ^ (elem.className && (" " + elem.className + " ").indexOf(match) >= 0) ) {
+                                               if ( !inplace )
+                                                       result.push( elem );
+                                       } else if ( inplace ) {
+                                               curLoop[i] = false;
+                                       }
+                               }
+                       }
+
+                       return false;
+               },
+               ID: function(match){
+                       return match[1].replace(/\\/g, "");
+               },
+               TAG: function(match, curLoop){
+                       for ( var i = 0; curLoop[i] === false; i++ ){}
+                       return curLoop[i] && isXML(curLoop[i]) ? match[1] : match[1].toUpperCase();
+               },
+               CHILD: function(match){
+                       if ( match[1] == "nth" ) {
+                               var test = /(-?)(\d*)n((?:\+|-)?\d*)/.exec(
+                                       match[2] == "even" && "2n" || match[2] == "odd" && "2n+1" ||
+                                       !/\D/.test( match[2] ) && "0n+" + match[2] || match[2]);
+
+                               match[2] = (test[1] + (test[2] || 1)) - 0;
+                               match[3] = test[3] - 0;
+                       }
+
+                       match[0] = done++;
+
+                       return match;
+               },
+               ATTR: function(match, curLoop, inplace, result, not, isXML){
+                       var name = match[1].replace(/\\/g, "");
+
+                       if ( !isXML && Expr.attrMap[name] ) {
+                               match[1] = Expr.attrMap[name];
+                       }
+
+                       if ( match[2] === "~=" ) {
+                               match[4] = " " + match[4] + " ";
+                       }
+
+                       return match;
+               },
+               PSEUDO: function(match, curLoop, inplace, result, not){
+                       if ( match[1] === "not" ) {
+                               if ( ( chunker.exec(match[3]) || "" ).length > 1 || /^\w/.test(match[3]) ) {
+                                       match[3] = Sizzle(match[3], null, null, curLoop);
+                               } else {
+                                       var ret = Sizzle.filter(match[3], curLoop, inplace, true ^ not);
+                                       if ( !inplace ) {
+                                               result.push.apply( result, ret );
+                                       }
+                                       return false;
+                               }
+                       } else if ( Expr.match.POS.test( match[0] ) || Expr.match.CHILD.test( match[0] ) ) {
+                               return true;
+                       }
+
+                       return match;
+               },
+               POS: function(match){
+                       match.unshift( true );
+                       return match;
+               }
+       },
+       filters: {
+               enabled: function(elem){
+                       return elem.disabled === false && elem.type !== "hidden";
+               },
+               disabled: function(elem){
+                       return elem.disabled === true;
+               },
+               checked: function(elem){
+                       return elem.checked === true;
+               },
+               selected: function(elem){
+                       elem.parentNode.selectedIndex;
+                       return elem.selected === true;
+               },
+               parent: function(elem){
+                       return !!elem.firstChild;
+               },
+               empty: function(elem){
+                       return !elem.firstChild;
+               },
+               has: function(elem, i, match){
+                       return !!Sizzle( match[3], elem ).length;
+               },
+               header: function(elem){
+                       return /h\d/i.test( elem.nodeName );
+               },
+               text: function(elem){
+                       return "text" === elem.type;
+               },
+               radio: function(elem){
+                       return "radio" === elem.type;
+               },
+               checkbox: function(elem){
+                       return "checkbox" === elem.type;
+               },
+               file: function(elem){
+                       return "file" === elem.type;
+               },
+               password: function(elem){
+                       return "password" === elem.type;
+               },
+               submit: function(elem){
+                       return "submit" === elem.type;
+               },
+               image: function(elem){
+                       return "image" === elem.type;
+               },
+               reset: function(elem){
+                       return "reset" === elem.type;
+               },
+               button: function(elem){
+                       return "button" === elem.type || elem.nodeName.toUpperCase() === "BUTTON";
+               },
+               input: function(elem){
+                       return /input|select|textarea|button/i.test(elem.nodeName);
+               }
+       },
+       setFilters: {
+               first: function(elem, i){
+                       return i === 0;
+               },
+               last: function(elem, i, match, array){
+                       return i === array.length - 1;
+               },
+               even: function(elem, i){
+                       return i % 2 === 0;
+               },
+               odd: function(elem, i){
+                       return i % 2 === 1;
+               },
+               lt: function(elem, i, match){
+                       return i < match[3] - 0;
+               },
+               gt: function(elem, i, match){
+                       return i > match[3] - 0;
+               },
+               nth: function(elem, i, match){
+                       return match[3] - 0 == i;
+               },
+               eq: function(elem, i, match){
+                       return match[3] - 0 == i;
+               }
+       },
+       filter: {
+               PSEUDO: function(elem, match, i, array){
+                       var name = match[1], filter = Expr.filters[ name ];
+
+                       if ( filter ) {
+                               return filter( elem, i, match, array );
+                       } else if ( name === "contains" ) {
+                               return (elem.textContent || elem.innerText || "").indexOf(match[3]) >= 0;
+                       } else if ( name === "not" ) {
+                               var not = match[3];
+
+                               for ( var i = 0, l = not.length; i < l; i++ ) {
+                                       if ( not[i] === elem ) {
+                                               return false;
+                                       }
+                               }
+
+                               return true;
+                       }
+               },
+               CHILD: function(elem, match){
+                       var type = match[1], node = elem;
+                       switch (type) {
+                               case 'only':
+                               case 'first':
+                                       while ( (node = node.previousSibling) )  {
+                                               if ( node.nodeType === 1 ) return false;
+                                       }
+                                       if ( type == 'first') return true;
+                                       node = elem;
+                               case 'last':
+                                       while ( (node = node.nextSibling) )  {
+                                               if ( node.nodeType === 1 ) return false;
+                                       }
+                                       return true;
+                               case 'nth':
+                                       var first = match[2], last = match[3];
+
+                                       if ( first == 1 && last == 0 ) {
+                                               return true;
+                                       }
+
+                                       var doneName = match[0],
+                                               parent = elem.parentNode;
+
+                                       if ( parent && (parent.sizcache !== doneName || !elem.nodeIndex) ) {
+                                               var count = 0;
+                                               for ( node = parent.firstChild; node; node = node.nextSibling ) {
+                                                       if ( node.nodeType === 1 ) {
+                                                               node.nodeIndex = ++count;
+                                                       }
+                                               }
+                                               parent.sizcache = doneName;
+                                       }
+
+                                       var diff = elem.nodeIndex - last;
+                                       if ( first == 0 ) {
+                                               return diff == 0;
+                                       } else {
+                                               return ( diff % first == 0 && diff / first >= 0 );
+                                       }
+                       }
+               },
+               ID: function(elem, match){
+                       return elem.nodeType === 1 && elem.getAttribute("id") === match;
+               },
+               TAG: function(elem, match){
+                       return (match === "*" && elem.nodeType === 1) || elem.nodeName === match;
+               },
+               CLASS: function(elem, match){
+                       return (" " + (elem.className || elem.getAttribute("class")) + " ")
+                               .indexOf( match ) > -1;
+               },
+               ATTR: function(elem, match){
+                       var name = match[1],
+                               result = Expr.attrHandle[ name ] ?
+                                       Expr.attrHandle[ name ]( elem ) :
+                                       elem[ name ] != null ?
+                                               elem[ name ] :
+                                               elem.getAttribute( name ),
+                               value = result + "",
+                               type = match[2],
+                               check = match[4];
+
+                       return result == null ?
+                               type === "!=" :
+                               type === "=" ?
+                               value === check :
+                               type === "*=" ?
+                               value.indexOf(check) >= 0 :
+                               type === "~=" ?
+                               (" " + value + " ").indexOf(check) >= 0 :
+                               !check ?
+                               value && result !== false :
+                               type === "!=" ?
+                               value != check :
+                               type === "^=" ?
+                               value.indexOf(check) === 0 :
+                               type === "$=" ?
+                               value.substr(value.length - check.length) === check :
+                               type === "|=" ?
+                               value === check || value.substr(0, check.length + 1) === check + "-" :
+                               false;
+               },
+               POS: function(elem, match, i, array){
+                       var name = match[2], filter = Expr.setFilters[ name ];
+
+                       if ( filter ) {
+                               return filter( elem, i, match, array );
+                       }
+               }
+       }
+};
+
+var origPOS = Expr.match.POS;
+
+for ( var type in Expr.match ) {
+       Expr.match[ type ] = new RegExp( Expr.match[ type ].source + /(?![^\[]*\])(?![^\(]*\))/.source );
+       Expr.leftMatch[ type ] = new RegExp( /(^(?:.|\r|\n)*?)/.source + Expr.match[ type ].source );
+}
+
+var makeArray = function(array, results) {
+       array = Array.prototype.slice.call( array, 0 );
+
+       if ( results ) {
+               results.push.apply( results, array );
+               return results;
+       }
+
+       return array;
+};
+
+try {
+       Array.prototype.slice.call( document.documentElement.childNodes, 0 );
+
+} catch(e){
+       makeArray = function(array, results) {
+               var ret = results || [];
+
+               if ( toString.call(array) === "[object Array]" ) {
+                       Array.prototype.push.apply( ret, array );
+               } else {
+                       if ( typeof array.length === "number" ) {
+                               for ( var i = 0, l = array.length; i < l; i++ ) {
+                                       ret.push( array[i] );
+                               }
+                       } else {
+                               for ( var i = 0; array[i]; i++ ) {
+                                       ret.push( array[i] );
+                               }
+                       }
+               }
+
+               return ret;
+       };
+}
+
+var sortOrder;
+
+if ( document.documentElement.compareDocumentPosition ) {
+       sortOrder = function( a, b ) {
+               if ( !a.compareDocumentPosition || !b.compareDocumentPosition ) {
+                       if ( a == b ) {
+                               hasDuplicate = true;
+                       }
+                       return 0;
+               }
+
+               var ret = a.compareDocumentPosition(b) & 4 ? -1 : a === b ? 0 : 1;
+               if ( ret === 0 ) {
+                       hasDuplicate = true;
+               }
+               return ret;
+       };
+} else if ( "sourceIndex" in document.documentElement ) {
+       sortOrder = function( a, b ) {
+               if ( !a.sourceIndex || !b.sourceIndex ) {
+                       if ( a == b ) {
+                               hasDuplicate = true;
+                       }
+                       return 0;
+               }
+
+               var ret = a.sourceIndex - b.sourceIndex;
+               if ( ret === 0 ) {
+                       hasDuplicate = true;
+               }
+               return ret;
+       };
+} else if ( document.createRange ) {
+       sortOrder = function( a, b ) {
+               if ( !a.ownerDocument || !b.ownerDocument ) {
+                       if ( a == b ) {
+                               hasDuplicate = true;
+                       }
+                       return 0;
+               }
+
+               var aRange = a.ownerDocument.createRange(), bRange = b.ownerDocument.createRange();
+               aRange.setStart(a, 0);
+               aRange.setEnd(a, 0);
+               bRange.setStart(b, 0);
+               bRange.setEnd(b, 0);
+               var ret = aRange.compareBoundaryPoints(Range.START_TO_END, bRange);
+               if ( ret === 0 ) {
+                       hasDuplicate = true;
+               }
+               return ret;
+       };
+}
+
+(function(){
+       var form = document.createElement("div"),
+               id = "script" + (new Date).getTime();
+       form.innerHTML = "<a name='" + id + "'/>";
+
+       var root = document.documentElement;
+       root.insertBefore( form, root.firstChild );
+
+       if ( !!document.getElementById( id ) ) {
+               Expr.find.ID = function(match, context, isXML){
+                       if ( typeof context.getElementById !== "undefined" && !isXML ) {
+                               var m = context.getElementById(match[1]);
+                               return m ? m.id === match[1] || typeof m.getAttributeNode !== "undefined" && m.getAttributeNode("id").nodeValue === match[1] ? [m] : undefined : [];
+                       }
+               };
+
+               Expr.filter.ID = function(elem, match){
+                       var node = typeof elem.getAttributeNode !== "undefined" && elem.getAttributeNode("id");
+                       return elem.nodeType === 1 && node && node.nodeValue === match;
+               };
+       }
+
+       root.removeChild( form );
+       root = form = null; // release memory in IE
+})();
+
+(function(){
+
+       var div = document.createElement("div");
+       div.appendChild( document.createComment("") );
+
+       if ( div.getElementsByTagName("*").length > 0 ) {
+               Expr.find.TAG = function(match, context){
+                       var results = context.getElementsByTagName(match[1]);
+
+                       if ( match[1] === "*" ) {
+                               var tmp = [];
+
+                               for ( var i = 0; results[i]; i++ ) {
+                                       if ( results[i].nodeType === 1 ) {
+                                               tmp.push( results[i] );
+                                       }
+                               }
+
+                               results = tmp;
+                       }
+
+                       return results;
+               };
+       }
+
+       div.innerHTML = "<a href='#'></a>";
+       if ( div.firstChild && typeof div.firstChild.getAttribute !== "undefined" &&
+                       div.firstChild.getAttribute("href") !== "#" ) {
+               Expr.attrHandle.href = function(elem){
+                       return elem.getAttribute("href", 2);
+               };
+       }
+
+       div = null; // release memory in IE
+})();
+
+if ( document.querySelectorAll ) (function(){
+       var oldSizzle = Sizzle, div = document.createElement("div");
+       div.innerHTML = "<p class='TEST'></p>";
+
+       if ( div.querySelectorAll && div.querySelectorAll(".TEST").length === 0 ) {
+               return;
+       }
+
+       Sizzle = function(query, context, extra, seed){
+               context = context || document;
+
+               if ( !seed && context.nodeType === 9 && !isXML(context) ) {
+                       try {
+                               return makeArray( context.querySelectorAll(query), extra );
+                       } catch(e){}
+               }
+
+               return oldSizzle(query, context, extra, seed);
+       };
+
+       for ( var prop in oldSizzle ) {
+               Sizzle[ prop ] = oldSizzle[ prop ];
+       }
+
+       div = null; // release memory in IE
+})();
+
+if ( document.getElementsByClassName && document.documentElement.getElementsByClassName ) (function(){
+       var div = document.createElement("div");
+       div.innerHTML = "<div class='test e'></div><div class='test'></div>";
+
+       if ( div.getElementsByClassName("e").length === 0 )
+               return;
+
+       div.lastChild.className = "e";
+
+       if ( div.getElementsByClassName("e").length === 1 )
+               return;
+
+       Expr.order.splice(1, 0, "CLASS");
+       Expr.find.CLASS = function(match, context, isXML) {
+               if ( typeof context.getElementsByClassName !== "undefined" && !isXML ) {
+                       return context.getElementsByClassName(match[1]);
+               }
+       };
+
+       div = null; // release memory in IE
+})();
+
+function dirNodeCheck( dir, cur, doneName, checkSet, nodeCheck, isXML ) {
+       var sibDir = dir == "previousSibling" && !isXML;
+       for ( var i = 0, l = checkSet.length; i < l; i++ ) {
+               var elem = checkSet[i];
+               if ( elem ) {
+                       if ( sibDir && elem.nodeType === 1 ){
+                               elem.sizcache = doneName;
+                               elem.sizset = i;
+                       }
+                       elem = elem[dir];
+                       var match = false;
+
+                       while ( elem ) {
+                               if ( elem.sizcache === doneName ) {
+                                       match = checkSet[elem.sizset];
+                                       break;
+                               }
+
+                               if ( elem.nodeType === 1 && !isXML ){
+                                       elem.sizcache = doneName;
+                                       elem.sizset = i;
+                               }
+
+                               if ( elem.nodeName === cur ) {
+                                       match = elem;
+                                       break;
+                               }
+
+                               elem = elem[dir];
+                       }
+
+                       checkSet[i] = match;
+               }
+       }
+}
+
+function dirCheck( dir, cur, doneName, checkSet, nodeCheck, isXML ) {
+       var sibDir = dir == "previousSibling" && !isXML;
+       for ( var i = 0, l = checkSet.length; i < l; i++ ) {
+               var elem = checkSet[i];
+               if ( elem ) {
+                       if ( sibDir && elem.nodeType === 1 ) {
+                               elem.sizcache = doneName;
+                               elem.sizset = i;
+                       }
+                       elem = elem[dir];
+                       var match = false;
+
+                       while ( elem ) {
+                               if ( elem.sizcache === doneName ) {
+                                       match = checkSet[elem.sizset];
+                                       break;
+                               }
+
+                               if ( elem.nodeType === 1 ) {
+                                       if ( !isXML ) {
+                                               elem.sizcache = doneName;
+                                               elem.sizset = i;
+                                       }
+                                       if ( typeof cur !== "string" ) {
+                                               if ( elem === cur ) {
+                                                       match = true;
+                                                       break;
+                                               }
+
+                                       } else if ( Sizzle.filter( cur, [elem] ).length > 0 ) {
+                                               match = elem;
+                                               break;
+                                       }
+                               }
+
+                               elem = elem[dir];
+                       }
+
+                       checkSet[i] = match;
+               }
+       }
+}
+
+var contains = document.compareDocumentPosition ?  function(a, b){
+       return a.compareDocumentPosition(b) & 16;
+} : function(a, b){
+       return a !== b && (a.contains ? a.contains(b) : true);
+};
+
+var isXML = function(elem){
+       return elem.nodeType === 9 && elem.documentElement.nodeName !== "HTML" ||
+               !!elem.ownerDocument && elem.ownerDocument.documentElement.nodeName !== "HTML";
+};
+
+var posProcess = function(selector, context){
+       var tmpSet = [], later = "", match,
+               root = context.nodeType ? [context] : context;
+
+       while ( (match = Expr.match.PSEUDO.exec( selector )) ) {
+               later += match[0];
+               selector = selector.replace( Expr.match.PSEUDO, "" );
+       }
+
+       selector = Expr.relative[selector] ? selector + "*" : selector;
+
+       for ( var i = 0, l = root.length; i < l; i++ ) {
+               Sizzle( selector, root[i], tmpSet );
+       }
+
+       return Sizzle.filter( later, tmpSet );
+};
+
+
+window.Sizzle = Sizzle;
+
+})();
+
+;(function(engine) {
+  var extendElements = Prototype.Selector.extendElements;
+
+  function select(selector, scope) {
+    return extendElements(engine(selector, scope || document));
+  }
+
+  function match(element, selector) {
+    return engine.matches(selector, [element]).length == 1;
+  }
+
+  Prototype.Selector.engine = engine;
+  Prototype.Selector.select = select;
+  Prototype.Selector.match = match;
+})(Sizzle);
+
+window.Sizzle = Prototype._original_property;
+delete Prototype._original_property;
+
+var Form = {
+  reset: function(form) {
+    form = $(form);
+    form.reset();
+    return form;
+  },
+
+  serializeElements: function(elements, options) {
+    if (typeof options != 'object') options = { hash: !!options };
+    else if (Object.isUndefined(options.hash)) options.hash = true;
+    var key, value, submitted = false, submit = options.submit, accumulator, initial;
+
+    if (options.hash) {
+      initial = {};
+      accumulator = function(result, key, value) {
+        if (key in result) {
+          if (!Object.isArray(result[key])) result[key] = [result[key]];
+          result[key].push(value);
+        } else result[key] = value;
+        return result;
+      };
+    } else {
+      initial = '';
+      accumulator = function(result, key, value) {
+        return result + (result ? '&' : '') + encodeURIComponent(key) + '=' + encodeURIComponent(value);
+      }
+    }
+
+    return elements.inject(initial, function(result, element) {
+      if (!element.disabled && element.name) {
+        key = element.name; value = $(element).getValue();
+        if (value != null && element.type != 'file' && (element.type != 'submit' || (!submitted &&
+            submit !== false && (!submit || key == submit) && (submitted = true)))) {
+          result = accumulator(result, key, value);
+        }
+      }
+      return result;
+    });
+  }
+};
+
+Form.Methods = {
+  serialize: function(form, options) {
+    return Form.serializeElements(Form.getElements(form), options);
+  },
+
+  getElements: function(form) {
+    var elements = $(form).getElementsByTagName('*'),
+        element,
+        arr = [ ],
+        serializers = Form.Element.Serializers;
+    for (var i = 0; element = elements[i]; i++) {
+      arr.push(element);
+    }
+    return arr.inject([], function(elements, child) {
+      if (serializers[child.tagName.toLowerCase()])
+        elements.push(Element.extend(child));
+      return elements;
+    })
+  },
+
+  getInputs: function(form, typeName, name) {
+    form = $(form);
+    var inputs = form.getElementsByTagName('input');
+
+    if (!typeName && !name) return $A(inputs).map(Element.extend);
+
+    for (var i = 0, matchingInputs = [], length = inputs.length; i < length; i++) {
+      var input = inputs[i];
+      if ((typeName && input.type != typeName) || (name && input.name != name))
+        continue;
+      matchingInputs.push(Element.extend(input));
+    }
+
+    return matchingInputs;
+  },
+
+  disable: function(form) {
+    form = $(form);
+    Form.getElements(form).invoke('disable');
+    return form;
+  },
+
+  enable: function(form) {
+    form = $(form);
+    Form.getElements(form).invoke('enable');
+    return form;
+  },
+
+  findFirstElement: function(form) {
+    var elements = $(form).getElements().findAll(function(element) {
+      return 'hidden' != element.type && !element.disabled;
+    });
+    var firstByIndex = elements.findAll(function(element) {
+      return element.hasAttribute('tabIndex') && element.tabIndex >= 0;
+    }).sortBy(function(element) { return element.tabIndex }).first();
+
+    return firstByIndex ? firstByIndex : elements.find(function(element) {
+      return /^(?:input|select|textarea)$/i.test(element.tagName);
+    });
+  },
+
+  focusFirstElement: function(form) {
+    form = $(form);
+    var element = form.findFirstElement();
+    if (element) element.activate();
+    return form;
+  },
+
+  request: function(form, options) {
+    form = $(form), options = Object.clone(options || { });
+
+    var params = options.parameters, action = form.readAttribute('action') || '';
+    if (action.blank()) action = window.location.href;
+    options.parameters = form.serialize(true);
+
+    if (params) {
+      if (Object.isString(params)) params = params.toQueryParams();
+      Object.extend(options.parameters, params);
+    }
+
+    if (form.hasAttribute('method') && !options.method)
+      options.method = form.method;
+
+    return new Ajax.Request(action, options);
+  }
+};
+
+/*--------------------------------------------------------------------------*/
+
+
+Form.Element = {
+  focus: function(element) {
+    $(element).focus();
+    return element;
+  },
+
+  select: function(element) {
+    $(element).select();
+    return element;
+  }
+};
+
+Form.Element.Methods = {
+
+  serialize: function(element) {
+    element = $(element);
+    if (!element.disabled && element.name) {
+      var value = element.getValue();
+      if (value != undefined) {
+        var pair = { };
+        pair[element.name] = value;
+        return Object.toQueryString(pair);
+      }
+    }
+    return '';
+  },
+
+  getValue: function(element) {
+    element = $(element);
+    var method = element.tagName.toLowerCase();
+    return Form.Element.Serializers[method](element);
+  },
+
+  setValue: function(element, value) {
+    element = $(element);
+    var method = element.tagName.toLowerCase();
+    Form.Element.Serializers[method](element, value);
+    return element;
+  },
+
+  clear: function(element) {
+    $(element).value = '';
+    return element;
+  },
+
+  present: function(element) {
+    return $(element).value != '';
+  },
+
+  activate: function(element) {
+    element = $(element);
+    try {
+      element.focus();
+      if (element.select && (element.tagName.toLowerCase() != 'input' ||
+          !(/^(?:button|reset|submit)$/i.test(element.type))))
+        element.select();
+    } catch (e) { }
+    return element;
+  },
+
+  disable: function(element) {
+    element = $(element);
+    element.disabled = true;
+    return element;
+  },
+
+  enable: function(element) {
+    element = $(element);
+    element.disabled = false;
+    return element;
+  }
+};
+
+/*--------------------------------------------------------------------------*/
+
+var Field = Form.Element;
+
+var $F = Form.Element.Methods.getValue;
+
+/*--------------------------------------------------------------------------*/
+
+Form.Element.Serializers = (function() {
+  function input(element, value) {
+    switch (element.type.toLowerCase()) {
+      case 'checkbox':
+      case 'radio':
+        return inputSelector(element, value);
+      default:
+        return valueSelector(element, value);
+    }
+  }
+
+  function inputSelector(element, value) {
+    if (Object.isUndefined(value))
+      return element.checked ? element.value : null;
+    else element.checked = !!value;
+  }
+
+  function valueSelector(element, value) {
+    if (Object.isUndefined(value)) return element.value;
+    else element.value = value;
+  }
+
+  function select(element, value) {
+    if (Object.isUndefined(value))
+      return (element.type === 'select-one' ? selectOne : selectMany)(element);
+
+    var opt, currentValue, single = !Object.isArray(value);
+    for (var i = 0, length = element.length; i < length; i++) {
+      opt = element.options[i];
+      currentValue = this.optionValue(opt);
+      if (single) {
+        if (currentValue == value) {
+          opt.selected = true;
+          return;
+        }
+      }
+      else opt.selected = value.include(currentValue);
+    }
+  }
+
+  function selectOne(element) {
+    var index = element.selectedIndex;
+    return index >= 0 ? optionValue(element.options[index]) : null;
+  }
+
+  function selectMany(element) {
+    var values, length = element.length;
+    if (!length) return null;
+
+    for (var i = 0, values = []; i < length; i++) {
+      var opt = element.options[i];
+      if (opt.selected) values.push(optionValue(opt));
+    }
+    return values;
+  }
+
+  function optionValue(opt) {
+    return Element.hasAttribute(opt, 'value') ? opt.value : opt.text;
+  }
+
+  return {
+    input:         input,
+    inputSelector: inputSelector,
+    textarea:      valueSelector,
+    select:        select,
+    selectOne:     selectOne,
+    selectMany:    selectMany,
+    optionValue:   optionValue,
+    button:        valueSelector
+  };
+})();
+
+/*--------------------------------------------------------------------------*/
+
+
+Abstract.TimedObserver = Class.create(PeriodicalExecuter, {
+  initialize: function($super, element, frequency, callback) {
+    $super(callback, frequency);
+    this.element   = $(element);
+    this.lastValue = this.getValue();
+  },
+
+  execute: function() {
+    var value = this.getValue();
+    if (Object.isString(this.lastValue) && Object.isString(value) ?
+        this.lastValue != value : String(this.lastValue) != String(value)) {
+      this.callback(this.element, value);
+      this.lastValue = value;
+    }
+  }
+});
+
+Form.Element.Observer = Class.create(Abstract.TimedObserver, {
+  getValue: function() {
+    return Form.Element.getValue(this.element);
+  }
+});
+
+Form.Observer = Class.create(Abstract.TimedObserver, {
+  getValue: function() {
+    return Form.serialize(this.element);
+  }
+});
+
+/*--------------------------------------------------------------------------*/
+
+Abstract.EventObserver = Class.create({
+  initialize: function(element, callback) {
+    this.element  = $(element);
+    this.callback = callback;
+
+    this.lastValue = this.getValue();
+    if (this.element.tagName.toLowerCase() == 'form')
+      this.registerFormCallbacks();
+    else
+      this.registerCallback(this.element);
+  },
+
+  onElementEvent: function() {
+    var value = this.getValue();
+    if (this.lastValue != value) {
+      this.callback(this.element, value);
+      this.lastValue = value;
+    }
+  },
+
+  registerFormCallbacks: function() {
+    Form.getElements(this.element).each(this.registerCallback, this);
+  },
+
+  registerCallback: function(element) {
+    if (element.type) {
+      switch (element.type.toLowerCase()) {
+        case 'checkbox':
+        case 'radio':
+          Event.observe(element, 'click', this.onElementEvent.bind(this));
+          break;
+        default:
+          Event.observe(element, 'change', this.onElementEvent.bind(this));
+          break;
+      }
+    }
+  }
+});
+
+Form.Element.EventObserver = Class.create(Abstract.EventObserver, {
+  getValue: function() {
+    return Form.Element.getValue(this.element);
+  }
+});
+
+Form.EventObserver = Class.create(Abstract.EventObserver, {
+  getValue: function() {
+    return Form.serialize(this.element);
+  }
+});
+(function() {
+
+  var Event = {
+    KEY_BACKSPACE: 8,
+    KEY_TAB:       9,
+    KEY_RETURN:   13,
+    KEY_ESC:      27,
+    KEY_LEFT:     37,
+    KEY_UP:       38,
+    KEY_RIGHT:    39,
+    KEY_DOWN:     40,
+    KEY_DELETE:   46,
+    KEY_HOME:     36,
+    KEY_END:      35,
+    KEY_PAGEUP:   33,
+    KEY_PAGEDOWN: 34,
+    KEY_INSERT:   45,
+
+    cache: {}
+  };
+
+  var docEl = document.documentElement;
+  var MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED = 'onmouseenter' in docEl
+    && 'onmouseleave' in docEl;
+
+
+
+  var isIELegacyEvent = function(event) { return false; };
+
+  if (window.attachEvent) {
+    if (window.addEventListener) {
+      isIELegacyEvent = function(event) {
+        return !(event instanceof window.Event);
+      };
+    } else {
+      isIELegacyEvent = function(event) { return true; };
+    }
+  }
+
+  var _isButton;
+
+  function _isButtonForDOMEvents(event, code) {
+    return event.which ? (event.which === code + 1) : (event.button === code);
+  }
+
+  var legacyButtonMap = { 0: 1, 1: 4, 2: 2 };
+  function _isButtonForLegacyEvents(event, code) {
+    return event.button === legacyButtonMap[code];
+  }
+
+  function _isButtonForWebKit(event, code) {
+    switch (code) {
+      case 0: return event.which == 1 && !event.metaKey;
+      case 1: return event.which == 2 || (event.which == 1 && event.metaKey);
+      case 2: return event.which == 3;
+      default: return false;
+    }
+  }
+
+  if (window.attachEvent) {
+    if (!window.addEventListener) {
+      _isButton = _isButtonForLegacyEvents;
+    } else {
+      _isButton = function(event, code) {
+        return isIELegacyEvent(event) ? _isButtonForLegacyEvents(event, code) :
+         _isButtonForDOMEvents(event, code);
+      }
+    }
+  } else if (Prototype.Browser.WebKit) {
+    _isButton = _isButtonForWebKit;
+  } else {
+    _isButton = _isButtonForDOMEvents;
+  }
+
+  function isLeftClick(event)   { return _isButton(event, 0) }
+
+  function isMiddleClick(event) { return _isButton(event, 1) }
+
+  function isRightClick(event)  { return _isButton(event, 2) }
+
+  function element(event) {
+    event = Event.extend(event);
+
+    var node = event.target, type = event.type,
+     currentTarget = event.currentTarget;
+
+    if (currentTarget && currentTarget.tagName) {
+      if (type === 'load' || type === 'error' ||
+        (type === 'click' && currentTarget.tagName.toLowerCase() === 'input'
+          && currentTarget.type === 'radio'))
+            node = currentTarget;
+    }
+
+    if (node.nodeType == Node.TEXT_NODE)
+      node = node.parentNode;
+
+    return Element.extend(node);
+  }
+
+  function findElement(event, expression) {
+    var element = Event.element(event);
+
+    if (!expression) return element;
+    while (element) {
+      if (Object.isElement(element) && Prototype.Selector.match(element, expression)) {
+        return Element.extend(element);
+      }
+      element = element.parentNode;
+    }
+  }
+
+  function pointer(event) {
+    return { x: pointerX(event), y: pointerY(event) };
+  }
+
+  function pointerX(event) {
+    var docElement = document.documentElement,
+     body = document.body || { scrollLeft: 0 };
+
+    return event.pageX || (event.clientX +
+      (docElement.scrollLeft || body.scrollLeft) -
+      (docElement.clientLeft || 0));
+  }
+
+  function pointerY(event) {
+    var docElement = document.documentElement,
+     body = document.body || { scrollTop: 0 };
+
+    return  event.pageY || (event.clientY +
+       (docElement.scrollTop || body.scrollTop) -
+       (docElement.clientTop || 0));
+  }
+
+
+  function stop(event) {
+    Event.extend(event);
+    event.preventDefault();
+    event.stopPropagation();
+
+    event.stopped = true;
+  }
+
+
+  Event.Methods = {
+    isLeftClick:   isLeftClick,
+    isMiddleClick: isMiddleClick,
+    isRightClick:  isRightClick,
+
+    element:     element,
+    findElement: findElement,
+
+    pointer:  pointer,
+    pointerX: pointerX,
+    pointerY: pointerY,
+
+    stop: stop
+  };
+
+  var methods = Object.keys(Event.Methods).inject({ }, function(m, name) {
+    m[name] = Event.Methods[name].methodize();
+    return m;
+  });
+
+  if (window.attachEvent) {
+    function _relatedTarget(event) {
+      var element;
+      switch (event.type) {
+        case 'mouseover':
+        case 'mouseenter':
+          element = event.fromElement;
+          break;
+        case 'mouseout':
+        case 'mouseleave':
+          element = event.toElement;
+          break;
+        default:
+          return null;
+      }
+      return Element.extend(element);
+    }
+
+    var additionalMethods = {
+      stopPropagation: function() { this.cancelBubble = true },
+      preventDefault:  function() { this.returnValue = false },
+      inspect: function() { return '[object Event]' }
+    };
+
+    Event.extend = function(event, element) {
+      if (!event) return false;
+
+      if (!isIELegacyEvent(event)) return event;
+
+      if (event._extendedByPrototype) return event;
+      event._extendedByPrototype = Prototype.emptyFunction;
+
+      var pointer = Event.pointer(event);
+
+      Object.extend(event, {
+        target: event.srcElement || element,
+        relatedTarget: _relatedTarget(event),
+        pageX:  pointer.x,
+        pageY:  pointer.y
+      });
+
+      Object.extend(event, methods);
+      Object.extend(event, additionalMethods);
+
+      return event;
+    };
+  } else {
+    Event.extend = Prototype.K;
+  }
+
+  if (window.addEventListener) {
+    Event.prototype = window.Event.prototype || document.createEvent('HTMLEvents').__proto__;
+    Object.extend(Event.prototype, methods);
+  }
+
+  function _createResponder(element, eventName, handler) {
+    var registry = Element.retrieve(element, 'prototype_event_registry');
+
+    if (Object.isUndefined(registry)) {
+      CACHE.push(element);
+      registry = Element.retrieve(element, 'prototype_event_registry', $H());
+    }
+
+    var respondersForEvent = registry.get(eventName);
+    if (Object.isUndefined(respondersForEvent)) {
+      respondersForEvent = [];
+      registry.set(eventName, respondersForEvent);
+    }
+
+    if (respondersForEvent.pluck('handler').include(handler)) return false;
+
+    var responder;
+    if (eventName.include(":")) {
+      responder = function(event) {
+        if (Object.isUndefined(event.eventName))
+          return false;
+
+        if (event.eventName !== eventName)
+          return false;
+
+        Event.extend(event, element);
+        handler.call(element, event);
+      };
+    } else {
+      if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED &&
+       (eventName === "mouseenter" || eventName === "mouseleave")) {
+        if (eventName === "mouseenter" || eventName === "mouseleave") {
+          responder = function(event) {
+            Event.extend(event, element);
+
+            var parent = event.relatedTarget;
+            while (parent && parent !== element) {
+              try { parent = parent.parentNode; }
+              catch(e) { parent = element; }
+            }
+
+            if (parent === element) return;
+
+            handler.call(element, event);
+          };
+        }
+      } else {
+        responder = function(event) {
+          Event.extend(event, element);
+          handler.call(element, event);
+        };
+      }
+    }
+
+    responder.handler = handler;
+    respondersForEvent.push(responder);
+    return responder;
+  }
+
+  function _destroyCache() {
+    for (var i = 0, length = CACHE.length; i < length; i++) {
+      Event.stopObserving(CACHE[i]);
+      CACHE[i] = null;
+    }
+  }
+
+  var CACHE = [];
+
+  if (Prototype.Browser.IE)
+    window.attachEvent('onunload', _destroyCache);
+
+  if (Prototype.Browser.WebKit)
+    window.addEventListener('unload', Prototype.emptyFunction, false);
+
+
+  var _getDOMEventName = Prototype.K,
+      translations = { mouseenter: "mouseover", mouseleave: "mouseout" };
+
+  if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED) {
+    _getDOMEventName = function(eventName) {
+      return (translations[eventName] || eventName);
+    };
+  }
+
+  function observe(element, eventName, handler) {
+    element = $(element);
+
+    var responder = _createResponder(element, eventName, handler);
+
+    if (!responder) return element;
+
+    if (eventName.include(':')) {
+      if (element.addEventListener)
+        element.addEventListener("dataavailable", responder, false);
+      else {
+        element.attachEvent("ondataavailable", responder);
+        element.attachEvent("onlosecapture", responder);
+      }
+    } else {
+      var actualEventName = _getDOMEventName(eventName);
+
+      if (element.addEventListener)
+        element.addEventListener(actualEventName, responder, false);
+      else
+        element.attachEvent("on" + actualEventName, responder);
+    }
+
+    return element;
+  }
+
+  function stopObserving(element, eventName, handler) {
+    element = $(element);
+
+    var registry = Element.retrieve(element, 'prototype_event_registry');
+    if (!registry) return element;
+
+    if (!eventName) {
+      registry.each( function(pair) {
+        var eventName = pair.key;
+        stopObserving(element, eventName);
+      });
+      return element;
+    }
+
+    var responders = registry.get(eventName);
+    if (!responders) return element;
+
+    if (!handler) {
+      responders.each(function(r) {
+        stopObserving(element, eventName, r.handler);
+      });
+      return element;
+    }
+
+    var i = responders.length, responder;
+    while (i--) {
+      if (responders[i].handler === handler) {
+        responder = responders[i];
+        break;
+      }
+    }
+    if (!responder) return element;
+
+    if (eventName.include(':')) {
+      if (element.removeEventListener)
+        element.removeEventListener("dataavailable", responder, false);
+      else {
+        element.detachEvent("ondataavailable", responder);
+        element.detachEvent("onlosecapture", responder);
+      }
+    } else {
+      var actualEventName = _getDOMEventName(eventName);
+      if (element.removeEventListener)
+        element.removeEventListener(actualEventName, responder, false);
+      else
+        element.detachEvent('on' + actualEventName, responder);
+    }
+
+    registry.set(eventName, responders.without(responder));
+
+    return element;
+  }
+
+  function fire(element, eventName, memo, bubble) {
+    element = $(element);
+
+    if (Object.isUndefined(bubble))
+      bubble = true;
+
+    if (element == document && document.createEvent && !element.dispatchEvent)
+      element = document.documentElement;
+
+    var event;
+    if (document.createEvent) {
+      event = document.createEvent('HTMLEvents');
+      event.initEvent('dataavailable', bubble, true);
+    } else {
+      event = document.createEventObject();
+      event.eventType = bubble ? 'ondataavailable' : 'onlosecapture';
+    }
+
+    event.eventName = eventName;
+    event.memo = memo || { };
+
+    if (document.createEvent)
+      element.dispatchEvent(event);
+    else
+      element.fireEvent(event.eventType, event);
+
+    return Event.extend(event);
+  }
+
+  Event.Handler = Class.create({
+    initialize: function(element, eventName, selector, callback) {
+      this.element   = $(element);
+      this.eventName = eventName;
+      this.selector  = selector;
+      this.callback  = callback;
+      this.handler   = this.handleEvent.bind(this);
+    },
+
+    start: function() {
+      Event.observe(this.element, this.eventName, this.handler);
+      return this;
+    },
+
+    stop: function() {
+      Event.stopObserving(this.element, this.eventName, this.handler);
+      return this;
+    },
+
+    handleEvent: function(event) {
+      var element = Event.findElement(event, this.selector);
+      if (element) this.callback.call(this.element, event, element);
+    }
+  });
+
+  function on(element, eventName, selector, callback) {
+    element = $(element);
+    if (Object.isFunction(selector) && Object.isUndefined(callback)) {
+      callback = selector, selector = null;
+    }
+
+    return new Event.Handler(element, eventName, selector, callback).start();
+  }
+
+  Object.extend(Event, Event.Methods);
+
+  Object.extend(Event, {
+    fire:          fire,
+    observe:       observe,
+    stopObserving: stopObserving,
+    on:            on
+  });
+
+  Element.addMethods({
+    fire:          fire,
+
+    observe:       observe,
+
+    stopObserving: stopObserving,
+
+    on:            on
+  });
+
+  Object.extend(document, {
+    fire:          fire.methodize(),
+
+    observe:       observe.methodize(),
+
+    stopObserving: stopObserving.methodize(),
+
+    on:            on.methodize(),
+
+    loaded:        false
+  });
+
+  if (window.Event) Object.extend(window.Event, Event);
+  else window.Event = Event;
+})();
+
+(function() {
+  /* Support for the DOMContentLoaded event is based on work by Dan Webb,
+     Matthias Miller, Dean Edwards, John Resig, and Diego Perini. */
+
+  var timer;
+
+  function fireContentLoadedEvent() {
+    if (document.loaded) return;
+    if (timer) window.clearTimeout(timer);
+    document.loaded = true;
+    document.fire('dom:loaded');
+  }
+
+  function checkReadyState() {
+    if (document.readyState === 'complete') {
+      document.stopObserving('readystatechange', checkReadyState);
+      fireContentLoadedEvent();
+    }
+  }
+
+  function pollDoScroll() {
+    try { document.documentElement.doScroll('left'); }
+    catch(e) {
+      timer = pollDoScroll.defer();
+      return;
+    }
+    fireContentLoadedEvent();
+  }
+
+  if (document.addEventListener) {
+    document.addEventListener('DOMContentLoaded', fireContentLoadedEvent, false);
+  } else {
+    document.observe('readystatechange', checkReadyState);
+    if (window == top)
+      timer = pollDoScroll.defer();
+  }
+
+  Event.observe(window, 'load', fireContentLoadedEvent);
+})();
+
+Element.addMethods();
+
+/*------------------------------- DEPRECATED -------------------------------*/
+
+Hash.toQueryString = Object.toQueryString;
+
+var Toggle = { display: Element.toggle };
+
+Element.Methods.childOf = Element.Methods.descendantOf;
+
+var Insertion = {
+  Before: function(element, content) {
+    return Element.insert(element, {before:content});
+  },
+
+  Top: function(element, content) {
+    return Element.insert(element, {top:content});
+  },
+
+  Bottom: function(element, content) {
+    return Element.insert(element, {bottom:content});
+  },
+
+  After: function(element, content) {
+    return Element.insert(element, {after:content});
+  }
+};
+
+var $continue = new Error('"throw $continue" is deprecated, use "return" instead');
+
+var Position = {
+  includeScrollOffsets: false,
+
+  prepare: function() {
+    this.deltaX =  window.pageXOffset
+                || document.documentElement.scrollLeft
+                || document.body.scrollLeft
+                || 0;
+    this.deltaY =  window.pageYOffset
+                || document.documentElement.scrollTop
+                || document.body.scrollTop
+                || 0;
+  },
+
+  within: function(element, x, y) {
+    if (this.includeScrollOffsets)
+      return this.withinIncludingScrolloffsets(element, x, y);
+    this.xcomp = x;
+    this.ycomp = y;
+    this.offset = Element.cumulativeOffset(element);
+
+    return (y >= this.offset[1] &&
+            y <  this.offset[1] + element.offsetHeight &&
+            x >= this.offset[0] &&
+            x <  this.offset[0] + element.offsetWidth);
+  },
+
+  withinIncludingScrolloffsets: function(element, x, y) {
+    var offsetcache = Element.cumulativeScrollOffset(element);
+
+    this.xcomp = x + offsetcache[0] - this.deltaX;
+    this.ycomp = y + offsetcache[1] - this.deltaY;
+    this.offset = Element.cumulativeOffset(element);
+
+    return (this.ycomp >= this.offset[1] &&
+            this.ycomp <  this.offset[1] + element.offsetHeight &&
+            this.xcomp >= this.offset[0] &&
+            this.xcomp <  this.offset[0] + element.offsetWidth);
+  },
+
+  overlap: function(mode, element) {
+    if (!mode) return 0;
+    if (mode == 'vertical')
+      return ((this.offset[1] + element.offsetHeight) - this.ycomp) /
+        element.offsetHeight;
+    if (mode == 'horizontal')
+      return ((this.offset[0] + element.offsetWidth) - this.xcomp) /
+        element.offsetWidth;
+  },
+
+
+  cumulativeOffset: Element.Methods.cumulativeOffset,
+
+  positionedOffset: Element.Methods.positionedOffset,
+
+  absolutize: function(element) {
+    Position.prepare();
+    return Element.absolutize(element);
+  },
+
+  relativize: function(element) {
+    Position.prepare();
+    return Element.relativize(element);
+  },
+
+  realOffset: Element.Methods.cumulativeScrollOffset,
+
+  offsetParent: Element.Methods.getOffsetParent,
+
+  page: Element.Methods.viewportOffset,
+
+  clone: function(source, target, options) {
+    options = options || { };
+    return Element.clonePosition(target, source, options);
+  }
+};
+
+/*--------------------------------------------------------------------------*/
+
+if (!document.getElementsByClassName) document.getElementsByClassName = function(instanceMethods){
+  function iter(name) {
+    return name.blank() ? null : "[contains(concat(' ', @class, ' '), ' " + name + " ')]";
+  }
+
+  instanceMethods.getElementsByClassName = Prototype.BrowserFeatures.XPath ?
+  function(element, className) {
+    className = className.toString().strip();
+    var cond = /\s/.test(className) ? $w(className).map(iter).join('') : iter(className);
+    return cond ? document._getElementsByXPath('.//*' + cond, element) : [];
+  } : function(element, className) {
+    className = className.toString().strip();
+    var elements = [], classNames = (/\s/.test(className) ? $w(className) : null);
+    if (!classNames && !className) return elements;
+
+    var nodes = $(element).getElementsByTagName('*');
+    className = ' ' + className + ' ';
+
+    for (var i = 0, child, cn; child = nodes[i]; i++) {
+      if (child.className && (cn = ' ' + child.className + ' ') && (cn.include(className) ||
+          (classNames && classNames.all(function(name) {
+            return !name.toString().blank() && cn.include(' ' + name + ' ');
+          }))))
+        elements.push(Element.extend(child));
+    }
+    return elements;
+  };
+
+  return function(className, parentElement) {
+    return $(parentElement || document.body).getElementsByClassName(className);
+  };
+}(Element.Methods);
+
+/*--------------------------------------------------------------------------*/
+
+Element.ClassNames = Class.create();
+Element.ClassNames.prototype = {
+  initialize: function(element) {
+    this.element = $(element);
+  },
+
+  _each: function(iterator) {
+    this.element.className.split(/\s+/).select(function(name) {
+      return name.length > 0;
+    })._each(iterator);
+  },
+
+  set: function(className) {
+    this.element.className = className;
+  },
+
+  add: function(classNameToAdd) {
+    if (this.include(classNameToAdd)) return;
+    this.set($A(this).concat(classNameToAdd).join(' '));
+  },
+
+  remove: function(classNameToRemove) {
+    if (!this.include(classNameToRemove)) return;
+    this.set($A(this).without(classNameToRemove).join(' '));
+  },
+
+  toString: function() {
+    return $A(this).join(' ');
+  }
+};
+
+Object.extend(Element.ClassNames.prototype, Enumerable);
+
+/*--------------------------------------------------------------------------*/
+
+(function() {
+  window.Selector = Class.create({
+    initialize: function(expression) {
+      this.expression = expression.strip();
+    },
+
+    findElements: function(rootElement) {
+      return Prototype.Selector.select(this.expression, rootElement);
+    },
+
+    match: function(element) {
+      return Prototype.Selector.match(element, this.expression);
+    },
+
+    toString: function() {
+      return this.expression;
+    },
+
+    inspect: function() {
+      return "#<Selector: " + this.expression + ">";
+    }
+  });
+
+  Object.extend(Selector, {
+    matchElements: function(elements, expression) {
+      var match = Prototype.Selector.match,
+          results = [];
+
+      for (var i = 0, length = elements.length; i < length; i++) {
+        var element = elements[i];
+        if (match(element, expression)) {
+          results.push(Element.extend(element));
+        }
+      }
+      return results;
+    },
+
+    findElement: function(elements, expression, index) {
+      index = index || 0;
+      var matchIndex = 0, element;
+      for (var i = 0, length = elements.length; i < length; i++) {
+        element = elements[i];
+        if (Prototype.Selector.match(element, expression) && index === matchIndex++) {
+          return Element.extend(element);
+        }
+      }
+    },
+
+    findChildElements: function(element, expressions) {
+      var selector = expressions.toArray().join(', ');
+      return Prototype.Selector.select(selector, element || document);
+    }
+  });
+})();
diff --git a/minsrc/ricoCalendar.js b/minsrc/ricoCalendar.js
new file mode 100644 (file)
index 0000000..64b099a
--- /dev/null
@@ -0,0 +1,548 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+//  Inspired by code originally written by Tan Ling Wee on 2 Dec 2001
+
+Rico.CalendarControl = function(id,options) {
+  this.initialize(id,options);
+};
+
+Rico.CalendarControl.prototype = {
+/**
+ * @class Implements a pop-up Gregorian calendar.
+ * Dates of adoption of the Gregorian calendar vary by country - accurate as a US & British calendar from 14 Sept 1752 to present.
+ * Mark special dates with calls to addHoliday()
+ * @extends Rico.Popup
+ * @constructs
+ * @param id unique identifier
+ * @param options object may contain any of the following:<dl>
+ *   <dt>startAt       </dt><dd> week starts with 0=sunday, 1=monday? default=0</dd>
+ *   <dt>showWeekNumber</dt><dd> show week number in first column? default=0</dd>
+ *   <dt>showToday     </dt><dd> show "Today is..." in footer? default=1</dd>
+ *   <dt>dateFmt       </dt><dd> date format for return value (one of values accepted by {@link Date#formatDate}), default=ISO8601</dd>
+ *   <dt>minDate       </dt><dd> earliest selectable date? default=today-50 years</dd>
+ *   <dt>maxDate       </dt><dd> last selectable date? default=today+50 years</dd>
+ *</dl>
+ */
+  initialize: function(id,options) {
+    this.id=id;
+    var today=new Date();
+    Rico.extend(this, new Rico.Popup());
+    Rico.extend(this.options, {
+      ignoreClicks:true,
+      startAt : 0,
+      showWeekNumber : 0,
+      showToday : 1,
+      dateFmt : 'ISO8601',
+      minDate : new Date(today.getFullYear()-50,0,1),
+      maxDate : new Date(today.getFullYear()+50,11,31)
+    });
+    Rico.extend(this.options, options || {});
+    /**
+     * alias for closePopup
+     * @function
+     */
+    this.close=this.closePopup;
+    this.bPageLoaded=false;
+    this.img=[];
+    this.Holidays={};
+    this.weekString=Rico.getPhraseById("calWeekHdg");
+    this.re=/^\s*(\w+)(\W)(\w+)(\W)(\w+)/i;
+    this.setDateFmt(this.options.dateFmt);
+  },
+
+
+  setDateFmt: function(fmt) {
+    this.dateFmt=(fmt=='rico') ? Rico.dateFmt : fmt;
+    Rico.log(this.id+' date format set to '+this.dateFmt);
+    this.dateParts={};
+    if (this.re.exec(this.dateFmt)) {
+      this.dateParts[RegExp.$1]=0;
+      this.dateParts[RegExp.$3]=1;
+      this.dateParts[RegExp.$5]=2;
+    }
+  },
+  
+/**
+ * Call before displaying calendar to highlight special days
+ * @param d day (1-31)
+ * @param m month (1-12)
+ * @param y year (0 implies a repeating holiday)
+ * @param desc description
+ * @param bgColor background color for cell displaying this day (CSS value, defaults to '#DDF')
+ * @param txtColor text color for cell displaying this day (CSS value), if not specified it is displayed with the same color as other days
+ */
+  addHoliday : function(d, m, y, desc, bgColor, txtColor) {
+    this.Holidays[this.holidayKey(y,m-1,d)]={desc:desc, txtColor:txtColor, bgColor:bgColor || '#DDF'};
+  },
+  
+/** @private */
+  holidayKey : function(y,m,d) {
+    return 'h'+Rico.zFill(y,4)+Rico.zFill(m,2)+Rico.zFill(d,2);
+  },
+
+  atLoad : function() {
+    Rico.log('Calendar#atLoad: '+this.id);
+    this.createContainer();
+    this.container.id=this.id;
+    Rico.addClass(this.container, Rico.theme.calendar || 'ricoCalContainer');
+
+    this.maintab=document.createElement("table");
+    this.maintab.cellSpacing=2;
+    this.maintab.cellPadding=0;
+    this.maintab.border=0;
+    this.maintab.style.borderCollapse='separate';
+    this.maintab.className='ricoCalTab';
+    if (Rico.theme.calendarTable) Rico.addClass(this.maintab,Rico.theme.calendarTable)
+    this.tbody=Rico.getTBody(this.maintab);
+
+    var r,c,d,i,j,img,dow,a,s,tab;
+    this.colStart=this.options.showWeekNumber ? 1 : 0;
+    for (i=0; i<7; i++) {
+      r=this.tbody.insertRow(-1);
+      r.className='row'+i;
+      for (c=0; c<7+this.colStart; c++) {
+        r.insertCell(-1);
+      }
+    }
+    r=this.tbody.rows[0];
+    r.className='ricoCalDayNames';
+    if (this.options.showWeekNumber) {
+      r.cells[0].innerHTML=this.weekString;
+      for (i=0; i<7; i++) {
+        this.tbody.rows[i].cells[0].className='ricoCalWeekNum';
+      }
+    }
+    this.styles=[];
+    for (i=0; i<7; i++) {
+      dow=(i+this.options.startAt) % 7;
+      r.cells[i+this.colStart].innerHTML=Rico.dayAbbr(dow);
+      this.styles[i]='ricoCal'+dow;
+    }
+    
+    // table header (navigation controls)
+    this.thead=this.maintab.createTHead();
+    r=this.thead.insertRow(-1);
+    c=r.appendChild(document.createElement("th"));
+    c.colSpan=7+this.colStart;
+    d=c.appendChild(document.createElement("div"));
+    //d.style.padding='3px';
+    d.className=Rico.theme.calendarHeading || 'RicoCalHeading';
+    
+    d.appendChild(this._createTitleSection('Month'));
+    d.appendChild(this._createTitleSection('Year'));
+    new Rico.HoverSet(d.getElementsByTagName('a'));
+    new Rico.HoverSet(this.tbody.getElementsByTagName('td'),{ hoverNodes: function(e) { return e.innerHTML.match(/^\d+$/) ? [e] : []; } });
+    d.appendChild(Rico.closeButton(Rico.eventHandle(this,'close')));
+
+    // table footer (today)
+    if (this.options.showToday) {
+      this.tfoot=this.maintab.createTFoot();
+      r=this.tfoot.insertRow(-1);
+      this.todayCell=r.insertCell(-1);
+      this.todayCell.colSpan=7+this.colStart;
+      if (Rico.theme.calendarFooter) Rico.addClass(this.todayCell,Rico.theme.calendarFooter);
+      Rico.eventBind(this.todayCell,"click", Rico.eventHandle(this,'selectNow'), false);
+    }
+    this.content.appendChild(this.maintab);
+    var ie6=Rico.isIE && Rico.ieVersion < 7;
+    var selectOptions={shadow: !ie6};
+    
+    // month selector
+    this.monthPopup=new Rico.Popup(document.createElement("div"),selectOptions);
+    this.monthPopup.closePopup();
+    tab=document.createElement("table");
+    tab.className='ricoCalMenu';
+    if (Rico.theme.calendarPopdown) Rico.addClass(tab,Rico.theme.calendarPopdown);
+    tab.cellPadding=2;
+    tab.cellSpacing=0;
+    tab.border=0;
+    tab.style.borderCollapse='separate';
+    tab.style.margin='0px';
+    for (i=0; i<4; i++) {
+      r=tab.insertRow(-1);
+      for (j=0; j<3; j++) {
+        c=r.insertCell(-1);
+        a=document.createElement("a");
+        a.innerHTML=Rico.monthAbbr(i*3+j);
+        a.name=i*3+j;
+        if (Rico.theme.calendarDay) Rico.addClass(a,Rico.theme.calendarDay);
+        c.appendChild(a);
+        Rico.eventBind(a,"click", Rico.eventHandle(this,'selectMonth'), false);
+      }
+    }
+    new Rico.HoverSet(tab.getElementsByTagName('a'));
+    this.monthPopup.content.appendChild(tab);
+    this.content.appendChild(this.monthPopup.container);
+    
+    // year selector
+    this.yearPopup=new Rico.Popup(document.createElement("div"),selectOptions);
+    this.yearPopup.closePopup();
+    this.yearPopup.content.className='ricoCalYearPrompt';
+    if (Rico.theme.calendarPopdown) Rico.addClass(this.yearPopup.content,Rico.theme.calendarPopdown);
+    var tab=document.createElement("table");
+    tab.cellPadding=2;
+    tab.cellSpacing=0;
+    tab.border=0;
+    tab.style.borderCollapse='separate';
+    tab.style.margin='0px';
+    r=tab.insertRow(-1);
+    this.yearLabel=r.insertCell(-1);
+    this.yearLabel.colSpan=3;
+    r=tab.insertRow(-1);
+    c=r.insertCell(-1);
+    this.yearInput=c.appendChild(document.createElement("input"));
+    this.yearInput.maxlength=4;
+    this.yearInput.size=4;
+    Rico.eventBind(this.yearInput,"keypress", Rico.eventHandle(this,'yearKey'), false);
+    c=r.insertCell(-1);
+    var a=Rico.floatButton('Checkmark', Rico.eventHandle(this,'processPopUpYear'));
+    Rico.setStyle(a.firstChild,{ margin:"0px", padding:"0px", border:"none" });
+    c.appendChild(a);
+    c=r.insertCell(-1);
+    a=Rico.floatButton('Cancel', Rico.eventHandle(this,'popDownYear'));
+    Rico.setStyle(a.firstChild,{ margin:"0px", padding:"0px", border:"none" });
+    c.appendChild(a);
+    this.yearPopup.content.appendChild(tab);
+    this.content.appendChild(this.yearPopup.container);
+
+    //this.yearLabel.className='ricoCalYearPromptText';
+
+    // fix anchors so they work in IE6
+    a=this.content.getElementsByTagName('a');
+    for (i=0; i<a.length; i++) {
+      a[i].href='javascript:void(0)';
+    }
+    
+    Rico.eventBind(this.tbody,"click", Rico.eventHandle(this,'saveAndClose'));
+    this.close();
+    this.bPageLoaded=true;
+  },
+
+  _createTitleSection : function(section) {
+    var s=document.createElement("span");
+    s.className='RicoCal'+section+'Heading';
+
+    var a=s.appendChild(document.createElement("a"));
+    a.className='Rico_leftArrow';
+    if (Rico.theme.leftArrowAnchor) Rico.addClass(a,Rico.theme.leftArrowAnchor);
+    a.appendChild(this.createNavArrow('dec'+section,'left'));
+
+    a=s.appendChild(document.createElement("a"));
+    a.style.display='inline';
+    Rico.eventBind(a,"click", Rico.eventHandle(this,'popUp'+section), false);
+    this['title'+section]=a;
+
+    a=s.appendChild(document.createElement("a"));
+    a.className='Rico_rightArrow';
+    if (Rico.theme.rightArrowAnchor) Rico.addClass(a,Rico.theme.rightArrowAnchor);
+    a.appendChild(this.createNavArrow('inc'+section,'right'));
+    return s
+  },
+  
+  selectNow : function() {
+    var today = new Date();
+    this.dateNow  = today.getDate();
+    this.monthNow = today.getMonth();
+    this.yearNow  = today.getFullYear();
+    this.monthSelected=this.monthNow;
+    this.yearSelected=this.yearNow;
+    this.constructCalendar();
+  },
+  
+/** @private */
+  createNavArrow: function(funcname,direction) {
+    var span=document.createElement("span");
+    span.className=Rico.theme[direction+'Arrow'] || 'rico-icon Rico_'+direction+'Arrow';
+    Rico.eventBind(span,"click", Rico.eventHandle(this,funcname), false);
+    return span;
+  },
+
+/**
+ * @returns true if yr/mo is within minDate/MaxDate
+ */
+  isValidMonth : function(yr,mo) {
+    if (yr < this.options.minDate.getFullYear()) return false;
+    if (yr == this.options.minDate.getFullYear() && mo < this.options.minDate.getMonth()) return false;
+    if (yr > this.options.maxDate.getFullYear()) return false;
+    if (yr == this.options.maxDate.getFullYear() && mo > this.options.maxDate.getMonth()) return false;
+    return true;
+  },
+
+  incMonth : function() {
+    var newMonth=this.monthSelected+1;
+    var newYear=this.yearSelected;
+    if (newMonth>11) {
+      newMonth=0;
+      newYear++;
+    }
+    if (!this.isValidMonth(newYear,newMonth)) return;
+    this.monthSelected=newMonth;
+    this.yearSelected=newYear;
+    this.constructCalendar();
+  },
+
+  decMonth : function() {
+    var newMonth=this.monthSelected-1;
+    var newYear=this.yearSelected;
+    if (newMonth<0) {
+      newMonth=11;
+      newYear--;
+    }
+    if (!this.isValidMonth(newYear,newMonth)) return;
+    this.monthSelected=newMonth;
+    this.yearSelected=newYear;
+    this.constructCalendar();
+  },
+  
+/** @private */
+  selectMonth : function(e) {
+    var el=Rico.eventElement(e);
+    this.monthSelected=parseInt(el.name,10);
+    this.constructCalendar();
+    Rico.eventStop(e);
+  },
+
+  popUpMonth : function(e) {
+    if (this.monthPopup.visible()) {
+      this.popDownMonth();
+      return;
+    }
+    this.popDownYear();
+    this.monthPopup.openPopup(this.titleMonth.parentNode.offsetLeft, this.thead.offsetHeight+2);
+    Rico.eventStop(e);
+    return false;
+  },
+
+  popDownMonth : function() {
+    this.monthPopup.closePopup();
+  },
+
+  popDownYear : function() {
+    this.yearPopup.closePopup();
+    this.yearInput.disabled=true;  // make sure this does not get submitted
+  },
+
+/**
+ * Prompt for year
+ */
+  popUpYear : function(e) {
+    if (this.yearPopup.visible()) {
+      this.popDownYear();
+      return;
+    }
+    this.popDownMonth();
+    this.yearPopup.openPopup(90, this.thead.offsetHeight+2);
+    this.yearLabel.innerHTML=Rico.getPhraseById("calYearRange",this.options.minDate.getFullYear(),this.options.maxDate.getFullYear());
+    this.yearInput.disabled=false;
+    this.yearInput.value='';   // this.yearSelected
+    this.yearInput.focus();
+    Rico.eventStop(e);
+    return false;
+  },
+  
+  yearKey : function(e) {
+    switch (Rico.eventKey(e)) {
+      case 27: this.popDownYear(); Rico.eventStop(e); return false;
+      case 13: this.processPopUpYear(); Rico.eventStop(e); return false;
+    }
+    return true;
+  },
+  
+  processPopUpYear : function() {
+    var newYear=this.yearInput.value;
+    newYear=parseInt(newYear,10);
+    if (isNaN(newYear) || newYear<this.options.minDate.getFullYear() || newYear>this.options.maxDate.getFullYear()) {
+      alert(Rico.getPhraseById("calInvalidYear"));
+    } else {
+      this.yearSelected=newYear;
+      this.popDownYear();
+      this.constructCalendar();
+    }
+  },
+  
+  incYear : function() {
+    if (this.yearSelected>=this.options.maxDate.getFullYear()) return;
+    this.yearSelected++;
+    this.constructCalendar();
+  },
+
+  decYear : function() {
+    if (this.yearSelected<=this.options.minDate.getFullYear()) return;
+    this.yearSelected--;
+    this.constructCalendar();
+  },
+
+  // tried a number of different week number functions posted on the net
+  // this is the only one that produced consistent results when comparing week numbers for December and the following January
+  WeekNbr : function(year,month,day) {
+    var when = new Date(year,month,day);
+    var newYear = new Date(year,0,1);
+    var offset = 7 + 1 - newYear.getDay();
+    if (offset == 8) offset = 1;
+    var daynum = ((Date.UTC(year,when.getMonth(),when.getDate(),0,0,0) - Date.UTC(year,0,1,0,0,0)) /1000/60/60/24) + 1;
+    var weeknum = Math.floor((daynum-offset+7)/7);
+    if (weeknum == 0) {
+      year--;
+      var prevNewYear = new Date(year,0,1);
+      var prevOffset = 7 + 1 - prevNewYear.getDay();
+      weeknum = (prevOffset == 2 || prevOffset == 8) ? 53 : 52;
+    }
+    return weeknum;
+  },
+
+  constructCalendar : function() {
+    var aNumDays = [31,0,31,30,31,30,31,31,30,31,30,31];
+    var startDate = new Date (this.yearSelected,this.monthSelected,1);
+    var endDate,numDaysInMonth,i,colnum;
+
+    if (typeof this.monthSelected!='number' || this.monthSelected>=12 || this.monthSelected<0) {
+      alert('ERROR in calendar: monthSelected='+this.monthSelected);
+      return;
+    }
+
+    if (this.monthSelected==1) {
+      endDate = new Date (this.yearSelected,this.monthSelected+1,1);
+      endDate = new Date (endDate - (24*60*60*1000));
+      numDaysInMonth = endDate.getDate();
+    } else {
+      numDaysInMonth = aNumDays[this.monthSelected];
+    }
+    var dayPointer = startDate.getDay() - this.options.startAt;
+    if (dayPointer<0) dayPointer+=7;
+    this.popDownMonth();
+    this.popDownYear();
+
+    //this.bgcolor=Rico.getStyle(this.tbody,'background-color');
+    //this.bgcolor=this.bgcolor.replace(/\"/g,'');
+    if (this.options.showWeekNumber) {
+      for (i=1; i<7; i++) {
+        this.tbody.rows[i].cells[0].innerHTML='&nbsp;';
+      }
+    }
+    for ( i=0; i<dayPointer; i++ ) {
+      this.resetCell(this.tbody.rows[1].cells[i+this.colStart]);
+    }
+
+    for ( var datePointer=1,r=1; datePointer<=numDaysInMonth; datePointer++,dayPointer++ ) {
+      colnum=dayPointer % 7;
+      if (this.options.showWeekNumber && colnum==0) {
+        this.tbody.rows[r].cells[0].innerHTML=this.WeekNbr(this.yearSelected,this.monthSelected,datePointer);
+      }
+      var c=this.tbody.rows[r].cells[colnum+this.colStart];
+      c.innerHTML=datePointer;
+      c.className=this.styles[colnum];
+      if ((datePointer==this.dateNow)&&(this.monthSelected==this.monthNow)&&(this.yearSelected==this.yearNow)) {
+        Rico.addClass(c,Rico.theme.calendarToday || 'ricoCalToday');
+      }
+      if (Rico.theme.calendarDay) Rico.addClass(c,Rico.theme.calendarDay);
+      if ((datePointer==this.odateSelected) && (this.monthSelected==this.omonthSelected) && (this.yearSelected==this.oyearSelected)) {
+        Rico.addClass(c,Rico.theme.calendarSelectedDay || 'ricoSelectedDay');
+      }
+      var h=this.Holidays[this.holidayKey(this.yearSelected,this.monthSelected,datePointer)];
+      if (!h)  {
+        h=this.Holidays[this.holidayKey(0,this.monthSelected,datePointer)];
+      }
+      c.style.color=h ? h.txtColor : '';
+      c.style.backgroundColor=h ? h.bgColor : '';
+      c.title=h ? h.desc : '';
+      if (colnum==6) r++;
+    }
+    while (dayPointer<42) {
+      colnum=dayPointer % 7;
+      this.resetCell(this.tbody.rows[r].cells[colnum+this.colStart]);
+      dayPointer++;
+      if (colnum==6) r++;
+    }
+
+    this.titleMonth.innerHTML = Rico.monthAbbr(this.monthSelected);
+    this.titleYear.innerHTML = this.yearSelected;
+    if (this.todayCell) {
+      this.todayCell.innerHTML = Rico.getPhraseById("calToday",this.dateNow,Rico.monthAbbr(this.monthNow),this.yearNow,this.monthNow+1);
+    }
+  },
+  
+/** @private */
+  resetCell: function(c) {
+    c.innerHTML="&nbsp;";
+    c.className='ricoCalEmpty';
+    c.style.color='';
+    c.style.backgroundColor='';
+    c.title='';
+  },
+  
+/** @private */
+  saveAndClose : function(e) {
+    Rico.eventStop(e);
+    var el=Rico.eventElement(e);
+    var s=el.innerHTML.replace(/&nbsp;/g,'');
+    if (s=='' || el.className=='ricoCalWeekNum') return;
+    var day=parseInt(s,10);
+    if (isNaN(day)) return;
+    var d=new Date(this.yearSelected,this.monthSelected,day);
+    var dateStr=Rico.formatDate(d,this.dateFmt=='ISO8601' ? 'yyyy-mm-dd' : this.dateFmt);
+    if (this.returnValue) {
+      this.returnValue(dateStr);
+      this.close();
+    }
+  },
+
+  open : function(curval) {
+    if (!this.bPageLoaded) return;
+    var today = new Date();
+    this.dateNow  = today.getDate();
+    this.monthNow = today.getMonth();
+    this.yearNow  = today.getFullYear();
+    this.oyearSelected = -1;
+    if (typeof curval=='object') {
+      this.odateSelected  = curval.getDate();
+      this.omonthSelected = curval.getMonth();
+      this.oyearSelected  = curval.getFullYear();
+    } else if (this.dateFmt=='ISO8601') {
+      var d=Rico.setISO8601(curval);
+      if (d) {
+        this.odateSelected  = d.getDate();
+        this.omonthSelected = d.getMonth();
+        this.oyearSelected  = d.getFullYear();
+      }
+    } else if (this.re.exec(curval)) {
+      var aDate = [ RegExp.$1, RegExp.$3, RegExp.$5 ];
+      this.odateSelected  = parseInt(aDate[this.dateParts.dd], 10);
+      this.omonthSelected = parseInt(aDate[this.dateParts.mm], 10) - 1;
+      this.oyearSelected  = parseInt(aDate[this.dateParts.yyyy], 10);
+      if (this.oyearSelected < 100) {
+        // apply a century to 2-digit years
+        this.oyearSelected+=this.yearNow - (this.yearNow % 100);
+        var maxyr=this.options.maxDate.getFullYear();
+        while (this.oyearSelected > maxyr) this.oyearSelected-=100;
+      }
+    } else {
+      if (curval) {
+        alert('ERROR: invalid date passed to calendar ('+curval+')');
+      }
+    }
+    if (this.oyearSelected > 0) {
+      this.dateSelected=this.odateSelected;
+      this.monthSelected=this.omonthSelected;
+      this.yearSelected=this.oyearSelected;
+    } else {
+      this.dateSelected=this.dateNow;
+      this.monthSelected=this.monthNow;
+      this.yearSelected=this.yearNow;
+    }
+    this.constructCalendar();
+    this.openPopup();
+  }
+};
diff --git a/minsrc/ricoColorPicker.js b/minsrc/ricoColorPicker.js
new file mode 100644 (file)
index 0000000..36c2a3e
--- /dev/null
@@ -0,0 +1,110 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+// ===================================================================
+// Adapted to Rico by Matt Brown from code 
+//   published by Matt Kruse http://www.mattkruse.com/
+// ===================================================================
+
+Rico.ColorPicker = function(id,options) {
+  this.initialize(id,options);
+};
+
+Rico.ColorPicker.prototype = {
+/**
+ * @class Implements a pop-up color picker control.
+ * @extends Rico.Popup
+ * @constructs
+ * @param id unique identifier
+ * @param options object may contain any of the following:<dl>
+ *   <dt>showColorCode</dt><dd> show hex color code as user hovers over color grid? default=false</dd>
+ *   <dt>cellsPerRow  </dt><dd> number of colors per row in the grid? default=18</dd>
+ *   <dt>palette      </dt><dd> array of 6 digit hex values, default=216 "web safe" colors</dd>
+ *</dl>
+ */
+  initialize: function(id,options) {
+    this.id=id;
+    this.currentValue = "#FFFFFF";
+    Rico.extend(this, new Rico.Popup());
+    Rico.extend(this.options, {
+      showColorCode : false,
+      cellsPerRow   : 18,
+      palette       : []
+    });
+    var hexvals=['00','33','66','99','CC','FF'];
+    for (var g=0; g<hexvals.length; g++) {
+      for (var r=0; r<hexvals.length; r++) {
+        for (var b=0; b<hexvals.length; b++) {
+          this.options.palette.push(hexvals[r]+hexvals[g]+hexvals[b]);
+        }
+      }
+    }
+    Rico.extend(this.options, options || {});
+  },
+
+  atLoad : function() {
+    this.createContainer();
+    this.content.className='ricoColorPicker';
+    var width = this.options.cellsPerRow;
+    var cp_contents = "<TABLE BORDER='1' CELLSPACING='1' CELLPADDING='0'>";
+    for (var i=0; i<this.options.palette.length; i++) {
+      if ((i % width) == 0) { cp_contents += "<TR>"; }
+      cp_contents += '<TD BGCOLOR="#'+this.options.palette[i]+'">&nbsp;</TD>';
+      if ( ((i+1)>=this.options.palette.length) || (((i+1) % width) == 0)) {
+        cp_contents += "</TR>";
+      }
+    }
+    var halfwidth = Math.floor(width/2);
+    if (this.options.showColorCode) {
+      cp_contents += "<TR><TD COLSPAN='"+halfwidth+"' ID='colorPickerSelectedColor'>&nbsp;</TD><TD COLSPAN='"+(width-halfwidth)+"' ALIGN='CENTER' ID='colorPickerSelectedColorValue'>#FFFFFF</TD></TR>";
+    } else {
+      cp_contents += "<TR><TD COLSPAN='"+width+"' ID='colorPickerSelectedColor'>&nbsp;</TD></TR>";
+    }
+    cp_contents += "</TABLE>";
+    this.content.innerHTML=cp_contents;
+    /**
+     * alias for openPopup
+     * @function
+     */
+    this.open=this.openPopup;
+    /**
+     * alias for closePopup
+     * @function
+     */
+    this.close=this.closePopup;
+    Rico.eventBind(this.container,"mouseover", Rico.eventHandle(this,'highlightColor'), false);
+    Rico.eventBind(this.container,"click", Rico.eventHandle(this,'selectColor'), false);
+    this.close();
+  },
+
+/** @private */
+  selectColor: function(e) {
+    Rico.eventStop(e);
+    if (this.returnValue) this.returnValue(this.currentValue);
+    this.close();
+  },
+
+/* This function runs when you move your mouse over a color block */
+/** @private */
+  highlightColor: function(e) {
+    var elem = Rico.eventElement(e);
+    if (!elem.tagName || elem.tagName.toLowerCase() != 'td') return;
+    var c=Rico.Color.createColorFromBackground(elem).toString();
+    this.currentValue = c;
+    Rico.setStyle('colorPickerSelectedColor', {backgroundColor:c});
+    var d = Rico.$("colorPickerSelectedColorValue");
+    if (d) d.innerHTML = c;
+  }
+};
diff --git a/minsrc/ricoDragDrop.js b/minsrc/ricoDragDrop.js
new file mode 100644 (file)
index 0000000..5eafd1a
--- /dev/null
@@ -0,0 +1,577 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+Rico.dndMgrList = [];
+
+Rico.registerDraggable = function(aDraggable, mgrIdx) {
+  if (typeof mgrIdx != 'number') mgrIdx=0;
+  if (typeof Rico.dndMgrList[mgrIdx] != 'object')
+    Rico.dndMgrList[mgrIdx] = new Rico.dndMgr();
+  Rico.dndMgrList[mgrIdx].registerDraggable(aDraggable);
+};
+
+Rico.registerDropZone = function(aDropZone, mgrIdx) {
+  if (typeof mgrIdx != 'number') mgrIdx=0;
+  if (typeof Rico.dndMgrList[mgrIdx] != 'object')
+    Rico.dndMgrList[mgrIdx] = new Rico.dndMgr();
+  Rico.dndMgrList[mgrIdx].registerDropZone(aDropZone);
+};
+
+Rico.dndMgr = function() {
+  this.initialize();
+};
+
+Rico.dndMgr.prototype = {
+/**
+ * @class Implements drag-n-drop manager -- a group of linked draggables and drop zones
+ * @constructs
+ */
+   initialize: function() {
+      this.dropZones                = [];
+      this.draggables               = [];
+      this.currentDragObjects       = [];
+      this.dragElement              = null;
+      this.lastSelectedDraggable    = null;
+      this.currentDragObjectVisible = false;
+      this.interestedInMotionEvents = false;
+      this._mouseDown = Rico.eventHandle(this,'_mouseDownHandler');
+      this._mouseMove = Rico.eventHandle(this,'_mouseMoveHandler');
+      this._mouseUp = Rico.eventHandle(this,'_mouseUpHandler');
+   },
+
+   registerDropZone: function(aDropZone) {
+      this.dropZones[ this.dropZones.length ] = aDropZone;
+   },
+
+   deregisterDropZone: function(aDropZone) {
+      var newDropZones = new Array();
+      var j = 0;
+      for ( var i = 0 ; i < this.dropZones.length ; i++ ) {
+         if ( this.dropZones[i] != aDropZone )
+            newDropZones[j++] = this.dropZones[i];
+      }
+
+      this.dropZones = newDropZones;
+   },
+
+   clearDropZones: function() {
+      this.dropZones = new Array();
+   },
+
+   registerDraggable: function( aDraggable ) {
+      this.draggables[ this.draggables.length ] = aDraggable;
+      var htmlElement = aDraggable.getMouseDownHTMLElement();
+      if ( htmlElement != null ) {
+         htmlElement.ricoDraggable = aDraggable;
+         Rico.eventBind(htmlElement, "mousedown", Rico.eventHandle(this,'_attachEvents'));
+         Rico.eventBind(htmlElement, "mousedown", this._mouseDown);
+      }
+   },
+
+   clearSelection: function() {
+      for ( var i = 0 ; i < this.currentDragObjects.length ; i++ )
+         this.currentDragObjects[i].deselect();
+      this.currentDragObjects = new Array();
+      this.lastSelectedDraggable = null;
+   },
+
+   hasSelection: function() {
+      return this.currentDragObjects.length > 0;
+   },
+
+   setStartDragFromElement: function( e, mouseDownElement ) {
+      this.origPos = Rico.cumulativeOffset(mouseDownElement);
+      var coord=Rico.eventClient(e);
+      this.startx = coord.x - this.origPos.left;
+      this.starty = coord.y - this.origPos.top;
+
+      this.interestedInMotionEvents = this.hasSelection();
+      Rico.eventStop(e);
+   },
+
+   updateSelection: function( draggable, extendSelection ) {
+      if ( ! extendSelection )
+         this.clearSelection();
+
+      if ( draggable.isSelected() ) {
+         this.currentDragObjects=this.currentDragObjects.without(draggable);
+         draggable.deselect();
+         if ( draggable == this.lastSelectedDraggable )
+            this.lastSelectedDraggable = null;
+      }
+      else {
+         draggable.select();
+         if ( draggable.isSelected() ) {
+           this.currentDragObjects.push(draggable);
+           this.lastSelectedDraggable = draggable;
+         }
+      }
+   },
+
+   _mouseDownHandler: function(e) {
+      // if not button 1 ignore it...
+      if (!Rico.eventLeftClick(e)) return;
+
+      var eventTarget      = Rico.eventElement(e);
+      var draggableObject  = eventTarget.ricoDraggable;
+
+      var candidate = eventTarget;
+      while (draggableObject == null && candidate.parentNode) {
+         candidate = candidate.parentNode;
+         draggableObject = candidate.ricoDraggable;
+      }
+
+      if ( draggableObject == null ) return;
+
+      this.updateSelection( draggableObject, e.ctrlKey );
+
+      // clear the drop zones postion cache...
+      if ( this.hasSelection() ) {
+         for ( var i = 0 ; i < this.dropZones.length ; i++ )
+            this.dropZones[i].clearPositionCache();
+      }
+      this.setStartDragFromElement( e, draggableObject.getMouseDownHTMLElement() );
+   },
+
+
+   _mouseMoveHandler: function(e) {
+      if ( !this.interestedInMotionEvents ) {
+         return;
+      }
+
+      if ( ! this.hasSelection() )
+         return;
+
+      if ( ! this.currentDragObjectVisible )
+         this._startDrag(e);
+
+      if ( !this.activatedDropZones )
+         this._activateRegisteredDropZones();
+
+      this._updateDraggableLocation(e);
+      this._updateDropZonesHover(e);
+
+      Rico.eventStop(e);
+   },
+
+   _makeDraggableObjectVisible: function(e) {
+      if ( !this.hasSelection() )
+         return;
+
+      var dragElement;
+      if ( this.currentDragObjects.length > 1 )
+         dragElement = this.currentDragObjects[0].getMultiObjectDragGUI(this.currentDragObjects);
+      else
+         dragElement = this.currentDragObjects[0].getSingleObjectDragGUI();
+
+      // go ahead and absolute position it...
+      this.dragElemPosition=Rico.getStyle(dragElement, "position");
+      if (this.dragElemPosition != "absolute")
+         dragElement.style.position = "absolute";
+
+      // need to parent him into the document...
+      if ( dragElement.parentNode == null || dragElement.parentNode.nodeType == 11 )
+         document.body.appendChild(dragElement);
+
+      this.dragElement = dragElement;
+      this._updateDraggableLocation(e);
+
+      this.currentDragObjectVisible = true;
+   },
+
+   _leftOffset: function(e) {
+          return e.offsetX ? document.body.scrollLeft : 0;
+       },
+
+   _topOffset: function(e) {
+          return e.offsetY ? document.body.scrollTop : 0;
+       },
+
+
+   _updateDraggableLocation: function(e) {
+      var dragObjectStyle = this.dragElement.style;
+      var coord=Rico.eventClient(e);
+      dragObjectStyle.left = (coord.x + this._leftOffset(e) - this.startx) + "px";
+      dragObjectStyle.top  = (coord.y + this._topOffset(e) - this.starty) + "px";
+   },
+
+   _updateDropZonesHover: function(e) {
+      var i,n = this.dropZones.length;
+      for ( i = 0 ; i < n ; i++ ) {
+         if ( ! this._mousePointInDropZone( e, this.dropZones[i] ) )
+            this.dropZones[i].hideHover();
+      }
+
+      for ( i = 0 ; i < n ; i++ ) {
+         if ( this._mousePointInDropZone( e, this.dropZones[i] ) ) {
+            if ( this.dropZones[i].canAccept(this.currentDragObjects) )
+               this.dropZones[i].showHover();
+         }
+      }
+   },
+
+   _startDrag: function(e) {
+      for ( var i = 0 ; i < this.currentDragObjects.length ; i++ )
+         this.currentDragObjects[i].startDrag();
+      this._makeDraggableObjectVisible(e);
+   },
+
+   _mouseUpHandler: function(e) {
+      if ( ! this.hasSelection() ) return;
+      if (!Rico.eventLeftClick(e)) return;
+
+      this.interestedInMotionEvents = false;
+
+      if ( this._placeDraggableInDropZone(e) )
+         this._completeDropOperation(e);
+      else if (this.dragElement != null) {
+         Rico.eventStop(e);
+         var self=this;
+         Rico.animate(this.dragElement,
+                      {duration: 300, onEnd: function() { self._doCancelDragProcessing(); } },
+                      {left:this.origPos.left, top:this.origPos.top});
+      }
+
+     Rico.eventUnbind(document.body, "mousemove", this._mouseMove);
+     Rico.eventUnbind(document.body, "mouseup",  this._mouseUp);
+   },
+
+   _retTrue: function () {
+      return true;
+   },
+
+   _completeDropOperation: function(e) {
+      if ( this.dragElement != this.currentDragObjects[0].getMouseDownHTMLElement() ) {
+         if ( this.dragElement.parentNode != null )
+            this.dragElement.parentNode.removeChild(this.dragElement);
+      }
+
+      this._deactivateRegisteredDropZones();
+      this._endDrag();
+      this.clearSelection();
+      this.dragElement = null;
+      this.currentDragObjectVisible = false;
+      Rico.eventStop(e);
+   },
+
+   _doCancelDragProcessing: function() {
+      this._cancelDrag();
+      if ( this.dragElement == this.currentDragObjects[0].getMouseDownHTMLElement() ) {
+         this.dragElement.style.position=this.dragElemPosition;
+      } else {
+         if ( this.dragElement && this.dragElement.parentNode != null )
+            this.dragElement.parentNode.removeChild(this.dragElement);
+      }
+      this._deactivateRegisteredDropZones();
+      this.dragElement = null;
+      this.currentDragObjectVisible = false;
+   },
+
+   _placeDraggableInDropZone: function(e) {
+      var foundDropZone = false;
+      var n = this.dropZones.length;
+      for ( var i = 0 ; i < n ; i++ ) {
+         if ( this._mousePointInDropZone( e, this.dropZones[i] ) ) {
+            if ( this.dropZones[i].canAccept(this.currentDragObjects) ) {
+               this.dropZones[i].hideHover();
+               this.dropZones[i].accept(this.currentDragObjects);
+               foundDropZone = true;
+               break;
+            }
+         }
+      }
+
+      return foundDropZone;
+   },
+
+   _cancelDrag: function() {
+      for ( var i = 0 ; i < this.currentDragObjects.length ; i++ )
+         this.currentDragObjects[i].cancelDrag();
+   },
+
+   _endDrag: function() {
+      for ( var i = 0 ; i < this.currentDragObjects.length ; i++ )
+         this.currentDragObjects[i].endDrag();
+   },
+
+   _mousePointInDropZone: function( e, dropZone ) {
+
+      var absoluteRect = dropZone.getAbsoluteRect();
+      var coord=Rico.eventClient(e);
+
+      return coord.x  > absoluteRect.left + this._leftOffset(e) &&
+             coord.x  < absoluteRect.right + this._leftOffset(e) &&
+             coord.y  > absoluteRect.top + this._topOffset(e)   &&
+             coord.y  < absoluteRect.bottom + this._topOffset(e);
+   },
+
+   _activateRegisteredDropZones: function() {
+      var n = this.dropZones.length;
+      for ( var i = 0 ; i < n ; i++ ) {
+         var dropZone = this.dropZones[i];
+         if ( dropZone.canAccept(this.currentDragObjects) )
+            dropZone.activate();
+      }
+
+      this.activatedDropZones = true;
+   },
+
+   _deactivateRegisteredDropZones: function() {
+      var n = this.dropZones.length;
+      for ( var i = 0 ; i < n ; i++ )
+         this.dropZones[i].deactivate();
+      this.activatedDropZones = false;
+   },
+
+   _attachEvents: function () {
+     Rico.eventBind(document.body, "mousemove", this._mouseMove);
+     Rico.eventBind(document.body, "mouseup",  this._mouseUp);
+   }
+
+};
+
+
+Rico.Draggable = function(type, htmlElement) {
+  this.initialize(type, htmlElement);
+};
+
+Rico.Draggable.prototype = {
+/**
+ * @class Implements behavior for a draggable element
+ * @constructs
+ */
+   initialize: function( type, htmlElement ) {
+      this.type          = type;
+      this.htmlElement   = Rico.$(htmlElement);
+      this.selected      = false;
+   },
+
+   /**
+    *   Returns the HTML element that should have a mouse down event
+    *   added to it in order to initiate a drag operation
+    **/
+   getMouseDownHTMLElement: function() {
+      return this.htmlElement;
+   },
+
+   select: function() {
+      this._select();
+   },
+
+   _select: function() {
+      this.selected = true;
+      if (this.showingSelected) return;
+      this.showingSelected = true;
+
+      var htmlElement = this.getMouseDownHTMLElement();
+      var color = Rico.Color.createColorFromBackground(htmlElement);
+      color.isBright() ? color.darken(0.033) : color.brighten(0.033);
+      this.saveBackground = Rico.getStyle(htmlElement, "backgroundColor", "background-color");
+      htmlElement.style.backgroundColor = color.asHex();
+   },
+
+   deselect: function() {
+      this.selected = false;
+      if (!this.showingSelected) return;
+      var htmlElement = this.getMouseDownHTMLElement();
+      htmlElement.style.backgroundColor = this.saveBackground;
+      this.showingSelected = false;
+   },
+
+   isSelected: function() {
+      return this.selected;
+   },
+
+   startDrag: function() {
+   },
+
+   cancelDrag: function() {
+   },
+
+   endDrag: function() {
+   },
+
+   getSingleObjectDragGUI: function() {
+      return this.htmlElement;
+   },
+
+   getMultiObjectDragGUI: function( draggables ) {
+      return this.htmlElement;
+   },
+
+   getDroppedGUI: function() {
+      return this.htmlElement;
+   },
+
+   toString: function() {
+      return this.type + ":" + this.htmlElement + ":";
+   }
+
+};
+
+
+Rico.LiveGridDraggable = function(grid, rownum, colnum) {
+  this.initialize(grid, rownum, colnum);
+};
+
+Rico.LiveGridDraggable.prototype = Rico.extend(new Rico.Draggable(), {
+/**
+ * @class Enables draggable behavior for LiveGrid cells.
+ * Called by LiveGrid#appendBlankRow for columns where canDrag is true.
+ * @extends Rico.Draggable
+ * @constructs
+ */
+  initialize: function( grid, rownum, colnum) {\r
+    this.type        = 'RicoCell';\r
+    this.htmlElement = grid.cell(rownum,colnum);\r
+    this.liveGrid    = grid;\r
+    this.dragRow     = rownum;\r
+    this.dragCol     = colnum;\r
+  },\r
+\r
+  select: function() {
+    if (this.dragRow >= this.liveGrid.buffer.totalRows) return;
+    this.selected = true;
+    this.showingSelected = true;
+  },
+
+  deselect: function() {
+    this.selected = false;
+    this.showingSelected = false;
+  },
+
+  getSingleObjectDragGUI: function() {\r
+    var div = document.createElement("div");\r
+    div.className = 'LiveGridDraggable';\r
+    div.style.width = (this.htmlElement.offsetWidth - 10) + "px";\r
+    div.innerHTML=this.htmlElement.innerHTML;
+    return div;\r
+  }\r
+});\r
+
+
+Rico.Dropzone = function(htmlElement) {
+  this.initialize(htmlElement);
+};
+
+Rico.Dropzone.prototype = {
+/**
+ * @class Implements behavior for a drop zone
+ * @constructs
+ */
+   initialize: function( htmlElement ) {
+      this.htmlElement  = Rico.$(htmlElement);
+      this.absoluteRect = null;
+   },
+
+   getHTMLElement: function() {
+      return this.htmlElement;
+   },
+
+   clearPositionCache: function() {
+      this.absoluteRect = null;
+   },
+
+   getAbsoluteRect: function() {
+      if ( this.absoluteRect == null ) {
+         var htmlElement = this.getHTMLElement();
+         var pos = Rico.viewportOffset(htmlElement);
+
+         this.absoluteRect = {
+            top:    pos.top,
+            left:   pos.left,
+            bottom: pos.top + htmlElement.offsetHeight,
+            right:  pos.left + htmlElement.offsetWidth
+         };
+      }
+      return this.absoluteRect;
+   },
+
+   activate: function() {
+      var htmlElement = this.getHTMLElement();
+      if (htmlElement == null  || this.showingActive)
+         return;
+
+      this.showingActive = true;
+      this.saveBackgroundColor = htmlElement.style.backgroundColor;
+
+      var fallbackColor = "#ffea84";
+      var currentColor = Rico.Color.createColorFromBackground(htmlElement);
+      if ( currentColor == null )
+         htmlElement.style.backgroundColor = fallbackColor;
+      else {
+         currentColor.isBright() ? currentColor.darken(0.2) : currentColor.brighten(0.2);
+         htmlElement.style.backgroundColor = currentColor.asHex();
+      }
+   },
+
+   deactivate: function() {
+      var htmlElement = this.getHTMLElement();
+      if (htmlElement == null || !this.showingActive)
+         return;
+
+      htmlElement.style.backgroundColor = this.saveBackgroundColor;
+      this.showingActive = false;
+      this.saveBackgroundColor = null;
+   },
+
+   showHover: function() {
+      var htmlElement = this.getHTMLElement();
+      if ( htmlElement == null || this.showingHover )
+         return;
+
+      this.saveBorderWidth = htmlElement.style.borderWidth;
+      this.saveBorderStyle = htmlElement.style.borderStyle;
+      this.saveBorderColor = htmlElement.style.borderColor;
+
+      this.showingHover = true;
+      htmlElement.style.borderWidth = "1px";
+      htmlElement.style.borderStyle = "solid";
+      //htmlElement.style.borderColor = "#ff9900";
+      htmlElement.style.borderColor = "#ffff00";
+   },
+
+   hideHover: function() {
+      var htmlElement = this.getHTMLElement();
+      if ( htmlElement == null || !this.showingHover )
+         return;
+
+      htmlElement.style.borderWidth = this.saveBorderWidth;
+      htmlElement.style.borderStyle = this.saveBorderStyle;
+      htmlElement.style.borderColor = this.saveBorderColor;
+      this.showingHover = false;
+   },
+
+   canAccept: function(draggableObjects) {
+      return true;
+   },
+
+   accept: function(draggableObjects) {
+      var htmlElement = this.getHTMLElement();
+      if ( htmlElement == null )
+         return;
+
+      var n = draggableObjects.length;
+      for ( var i = 0 ; i < n ; i++ ) {
+         var theGUI = draggableObjects[i].getDroppedGUI();
+         if ( Rico.getStyle( theGUI, "position" ) == "absolute" ) {
+            theGUI.style.position = "static";
+            theGUI.style.top = "";
+            theGUI.style.top = "";
+         }
+         htmlElement.appendChild(theGUI);
+      }
+   }
+};
diff --git a/minsrc/ricoGridCommon.js b/minsrc/ricoGridCommon.js
new file mode 100644 (file)
index 0000000..7ecaa99
--- /dev/null
@@ -0,0 +1,980 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+if(typeof Rico=='undefined') throw("GridCommon requires the Rico JavaScript framework");
+
+/**
+ * Define methods that are common to both SimpleGrid and LiveGrid
+ */
+Rico.GridCommon = {
+
+  baseInit: function() {
+    this.options = {
+      saveColumnInfo   : {width:true, filter:false, sort:false},  // save info in cookies?
+      cookiePrefix     : 'RicoGrid.',
+      allowColResize   : true,      // allow user to resize columns
+      windowResize     : true,      // Resize grid on window.resize event? Set to false when embedded in an accordian.
+      click            : null,
+      dblclick         : null,
+      contextmenu      : null,
+      menuEvent        : 'dblclick',  // event that triggers menus - click, dblclick, contextmenu, or none (no menus)
+      defaultWidth     : -1,          // if -1, then use unformatted column width
+      scrollBarWidth   : 19,          // this is the value used in positioning calculations, it does not actually change the width of the scrollbar
+      minScrollWidth   : 100,         // min scroll area width when width of frozen columns exceeds window width
+      frozenColumns    : 0,
+      exportWindow     : "height=400,width=500,scrollbars=1,menubar=1,resizable=1",
+      exportStyleList  : ['background-color','color','text-align','font-weight','font-size','font-family'],
+      exportImgTags    : false,       // applies to grid header and to SimpleGrid cells (not LiveGrid cells)
+      exportFormFields : true,
+      FilterLocation   : null,        // heading row number to place filters. -1=add a new heading row.
+      FilterAllToken   : '___ALL___', // select box value to use to indicate ALL
+      columnSpecs      : []
+    };
+    this.colWidths = [];
+    this.hdrCells=[];
+    this.headerColCnt=0;
+    this.headerRowIdx=0;       // row in header which gets resizers (no colspan's in this row)
+    this.tabs=new Array(2);
+    this.thead=new Array(2);
+    this.tbody=new Array(2);
+  },
+
+  attachMenuEvents: function() {
+    var i;
+    if (!this.options.menuEvent || this.options.menuEvent=='none') return;
+    this.hideScroll=navigator.userAgent.match(/Macintosh\b.*\b(Firefox|Camino)\b/i) || (Rico.isOpera && parseFloat(window.opera.version())<9.5);
+    this.options[this.options.menuEvent]=Rico.eventHandle(this,'handleMenuClick');
+    if (this.highlightDiv) {
+      switch (this.options.highlightElem) {
+        case 'cursorRow':
+          this.attachMenu(this.highlightDiv[0]);
+          break;
+        case 'cursorCell':
+          for (i=0; i<2; i++) {
+            this.attachMenu(this.highlightDiv[i]);
+          }
+          break;
+      }
+    }
+    for (i=0; i<2; i++) {
+      this.attachMenu(this.tbody[i]);
+    }
+  },
+
+  attachMenu: function(elem) {
+    if (this.options.click)
+      Rico.eventBind(elem, 'click', this.options.click, false);
+    if (this.options.dblclick) {
+      if (Rico.isWebKit || Rico.isOpera)
+        Rico.eventBind(elem, 'click', Rico.eventHandle(this,'handleDblClick'), false);
+      else
+        Rico.eventBind(elem, 'dblclick', this.options.dblclick, false);
+    }
+    if (this.options.contextmenu) {
+      if (Rico.isOpera || Rico.isKonqueror)
+        Rico.eventBind(elem, 'click', Rico.eventHandle(this,'handleContextMenu'), false);
+      else
+        Rico.eventBind(elem, 'contextmenu', this.options.contextmenu, false);
+    }
+  },
+
+/**
+ * implement double-click for browsers that don't support a double-click event (e.g. Safari)
+ */
+  handleDblClick: function(e) {
+    var elem=Rico.eventElement(e);
+    if (this.dblClickElem == elem) {
+      this.options.dblclick(e);
+    } else {
+      this.dblClickElem = elem;
+      this.safariTimer=Rico.runLater(300,this,'clearDblClick');
+    }
+  },
+
+  clearDblClick: function() {
+    this.dblClickElem=null;
+  },
+
+/**
+ * implement right-click for browsers that don't support contextmenu event (e.g. Opera, Konqueror)
+ * use control-click instead
+ */
+  handleContextMenu: function(e) {
+    var b;
+    if( typeof( e.which ) == 'number' )
+      b = e.which; //Netscape compatible
+    else if( typeof( e.button ) == 'number' )
+      b = e.button; //DOM
+    else
+      return;
+    if (b==1 && e.ctrlKey) {
+      this.options.contextmenu(e);
+    }
+  },
+
+  cancelMenu: function() {
+    if (this.menu) this.menu.cancelmenu();
+  },
+
+/**
+ * gather info from original headings
+ */
+  getColumnInfo: function(hdrSrc) {
+    Rico.log('getColumnInfo: len='+hdrSrc.length);
+    if (hdrSrc.length == 0) return 0;
+    this.headerRowCnt=hdrSrc.length;
+    var r,c,colcnt;
+    for (r=0; r<this.headerRowCnt; r++) {
+      var headerRow = hdrSrc[r];
+      var headerCells=headerRow.cells;
+      if (r >= this.hdrCells.length) this.hdrCells[r]=[];
+      for (c=0; c<headerCells.length; c++) {
+        var obj={};
+        obj.cell=headerCells[c];
+        obj.colSpan=headerCells[c].colSpan || 1;  // Safari & Konqueror return default colspan of 0
+        if (this.options.defaultWidth < 0) obj.initWidth=headerCells[c].offsetWidth;
+        this.hdrCells[r].push(obj);
+      }
+      if (headerRow.id.slice(-5)=='_main') {
+        colcnt=this.hdrCells[r].length;
+        this.headerRowIdx=r;
+      }
+    }
+    if (!colcnt) {
+      this.headerRowIdx=this.headerRowCnt-1;
+      colcnt=this.hdrCells[this.headerRowIdx].length;
+    }
+    Rico.log("getColumnInfo: colcnt="+colcnt);
+    return colcnt;
+  },
+
+  addHeadingRow: function(className) {
+    var r=this.headerRowCnt++;
+    this.hdrCells[r]=[];
+    for( var h=0; h < 2; h++ ) {
+      var row = this.thead[h].insertRow(-1);
+      var newClass='ricoLG_hdg '+this.tableId+'_hdg'+r;
+      if (className) newClass+=' '+className;
+      row.className=newClass;
+      var limit= h==0 ? this.options.frozenColumns : this.headerColCnt-this.options.frozenColumns;
+      for( var c=0; c < limit; c++ ) {
+        var hdrCell=row.insertCell(-1);
+        var colDiv=Rico.wrapChildren(hdrCell,'ricoLG_col');
+        Rico.wrapChildren(colDiv,'ricoLG_cell');
+        this.hdrCells[r].push({cell:hdrCell,colSpan:1});
+      }
+    }
+    return r;
+  },
+  
+/**
+ * create column array
+ */
+  createColumnArray: function(columnType) {
+    this.direction=Rico.getStyle(this.outerDiv,'direction').toLowerCase();  // ltr or rtl
+    this.align=this.direction=='rtl' ? ['right','left'] : ['left','right'];
+    Rico.log('createColumnArray: dir='+this.direction);
+    this.columns = [];
+    for (var c=0; c < this.headerColCnt; c++) {
+      Rico.log("createColumnArray: c="+c);
+      var tabidx=c<this.options.frozenColumns ? 0 : 1;
+      var col=new Rico[columnType](this, c, this.hdrCells[this.headerRowIdx][c], tabidx);
+      this.columns.push(col);
+      if (c > 0) this.columns[c-1].next=col;
+    }
+    this.getCookie();
+    Rico.runLater(100,this,'insertResizers');  // avoids peek-a-boo bug in column 1 in IE6/7
+  },
+
+/**
+ * Insert resizing handles
+ */
+  insertResizers: function() {
+    if (!this.options.allowColResize) return;
+    for (var x=0;x<this.columns.length;x++) {
+      this.columns[x].insertResizer();
+    }
+  },
+
+/**
+ * Create div structure
+ */
+  createDivs: function() {
+    Rico.log("createDivs start");
+    this.outerDiv   = this.createDiv("outer");
+    if (Rico.theme.widget) Rico.addClass(this.outerDiv,Rico.theme.widget);
+    if (this.outerDiv.firstChild && this.outerDiv.firstChild.tagName && this.outerDiv.firstChild.tagName.toUpperCase()=='TABLE') {
+      this.structTab=this.outerDiv.firstChild;
+      this.structTabLeft=this.structTab.rows[0].cells[0];
+      this.structTabUR=this.structTab.rows[0].cells[1];
+      this.structTabLR=this.structTab.rows[1].cells[0];
+    } else {
+      this.structTab = document.createElement("table");
+      this.structTab.border=0;
+      this.structTab.cellPadding=0;
+      this.structTab.cellSpacing=0;
+      var tr1=this.structTab.insertRow(-1);
+      tr1.vAlign='top';
+      this.structTabLeft=tr1.insertCell(-1);
+      this.structTabLeft.rowSpan=2;
+      var tr2=this.structTab.insertRow(-1);
+      tr2.vAlign='top';
+      this.structTabUR=tr1.insertCell(-1);
+      this.structTabLR=tr2.insertCell(-1);
+      this.outerDiv.appendChild(this.structTab);
+    }
+    //this.structTabLR.style.overflow='hidden';
+    //if (Rico.isOpera) this.outerDiv.style.overflow="hidden";
+    this.frozenTabs = this.createDiv("frozenTabs",this.structTabLeft);
+    this.innerDiv   = this.createDiv("inner",this.structTabUR);
+    this.scrollDiv  = this.createDiv("scroll",this.structTabLR);
+    this.resizeDiv  = this.createDiv("resize",this.outerDiv,true);
+    this.exportDiv  = this.createDiv("export",this.outerDiv,true);
+
+    this.messagePopup=new Rico.Popup();
+    this.messagePopup.createContainer({hideOnEscape:false, hideOnClick:false, parent:this.outerDiv});
+    this.messagePopup.content.className='ricoLG_messageDiv';
+    if (Rico.theme.gridMessage) Rico.addClass(this.messagePopup.content,Rico.theme.gridMessage);
+
+    this.keywordPopup=new Rico.Window('', {zIndex:-1, parent:this.outerDiv});
+    Rico.addClass(this.keywordPopup.container, 'ricoLG_keywordDiv');
+    var instructions=this.keywordPopup.contentDiv.appendChild(document.createElement("p"));
+    instructions.innerHTML=Rico.getPhraseById("keywordPrompt");
+    this.keywordBox=this.keywordPopup.contentDiv.appendChild(document.createElement("input"));
+    this.keywordBox.size=20;
+    Rico.eventBind(this.keywordBox,"keypress", Rico.eventHandle(this,'keywordKey'), false);
+    this.keywordPopup.contentDiv.appendChild(Rico.floatButton('Checkmark', Rico.eventHandle(this,'processKeyword')));
+    var s=this.keywordPopup.contentDiv.appendChild(document.createElement("p"));
+    Rico.setStyle(s,{clear:'both'});
+
+    //this.frozenTabs.style[this.align[0]]='0px';
+    //this.innerDiv.style[this.align[0]]='0px';
+    Rico.log("createDivs end");
+  },
+  
+  keywordKey: function(e) {
+    switch (Rico.eventKey(e)) {
+      case 27: this.closeKeyword(); Rico.eventStop(e); return false;
+      case 13: this.processKeyword(); Rico.eventStop(e); return false;
+    }
+    return true;
+  },
+  
+  openKeyword: function(colnum) {
+    this.keywordCol=colnum;
+    this.keywordBox.value='';
+    this.keywordPopup.setTitle(this.columns[colnum].displayName);
+    this.keywordPopup.centerPopup();
+    this.keywordBox.focus();
+  },
+  
+  closeKeyword: function() {
+    this.keywordPopup.closePopup();
+    this.cancelMenu();
+  },
+  
+  processKeyword: function() {
+    var keyword=this.keywordBox.value;
+    this.closeKeyword();
+    this.columns[this.keywordCol].setFilterKW(keyword);
+  },
+
+/**
+ * Create a div and give it a standardized id and class name.
+ * If the div already exists, then just assign the class name.
+ */
+  createDiv: function(elemName,elemParent,hidden) {
+    var id=this.tableId+"_"+elemName+"Div";
+    var newdiv=document.getElementById(id);
+    if (!newdiv) {
+      newdiv = document.createElement("div");
+      newdiv.id = id;
+      if (elemParent) elemParent.appendChild(newdiv);
+    }
+    newdiv.className = "ricoLG_"+elemName+"Div";
+    if (hidden) Rico.hide(newdiv);
+    return newdiv;
+  },
+
+/**
+ * Common code used to size & position divs in both SimpleGrid & LiveGrid
+ */
+  baseSizeDivs: function() {
+    this.setOtherHdrCellWidths();
+
+    if (this.options.frozenColumns) {
+      Rico.show(this.tabs[0]);
+      Rico.show(this.frozenTabs);
+      // order of next 3 lines is critical in IE6
+      this.hdrHt=Math.max(Rico.nan2zero(this.thead[0].offsetHeight),this.thead[1].offsetHeight);
+      this.dataHt=Math.max(Rico.nan2zero(this.tbody[0].offsetHeight),this.tbody[1].offsetHeight);
+      this.frzWi=this.borderWidth(this.tabs[0]);
+    } else {
+      Rico.hide(this.tabs[0]);
+      Rico.hide(this.frozenTabs);
+      this.frzWi=0;
+      this.hdrHt=this.thead[1].offsetHeight;
+      this.dataHt=this.tbody[1].offsetHeight;
+    }
+
+    var wiLimit,i;
+    var borderWi=this.borderWidth(this.columns[0].dataCell);
+    Rico.log('baseSizeDivs '+this.tableId+': hdrHt='+this.hdrHt+' dataHt='+this.dataHt);
+    Rico.log(this.tableId+' frzWi='+this.frzWi+' borderWi='+borderWi);
+    for (i=0; i<this.options.frozenColumns; i++) {
+      if (this.columns[i].visible) this.frzWi+=parseInt(this.columns[i].colWidth,10)+borderWi;
+    }
+    this.scrTabWi=this.borderWidth(this.tabs[1]);
+    this.scrTabWi0=this.scrTabWi;
+    Rico.log('scrTabWi: '+this.scrTabWi);
+    for (i=this.options.frozenColumns; i<this.columns.length; i++) {
+      if (this.columns[i].visible) this.scrTabWi+=parseInt(this.columns[i].colWidth,10)+borderWi;
+    }
+    this.scrWi=this.scrTabWi+this.options.scrollBarWidth;
+    if (this.sizeTo=='parent') {
+      if (Rico.isIE) Rico.hide(this.outerDiv);
+      wiLimit=this.outerDiv.parentNode.offsetWidth;
+      if (Rico.isIE) Rico.show(this.outerDiv);
+    }  else {
+      wiLimit=Rico.windowWidth()-this.options.scrollBarWidth-8;
+    }
+    if (this.outerDiv.parentNode.clientWidth > 0)
+      wiLimit=Math.min(this.outerDiv.parentNode.clientWidth, wiLimit);
+    var overage=this.frzWi+this.scrWi-wiLimit;
+    Rico.log('baseSizeDivs '+this.tableId+': scrWi='+this.scrWi+' wiLimit='+wiLimit+' overage='+overage+' clientWidth='+this.outerDiv.parentNode.clientWidth);
+    if (overage > 0 && this.options.frozenColumns < this.columns.length)
+      this.scrWi=Math.max(this.scrWi-overage, this.options.minScrollWidth);
+    this.scrollDiv.style.width=this.scrWi+'px';
+    //this.scrollDiv.style.top=this.hdrHt+'px';
+    //this.frozenTabs.style.width=this.scrollDiv.style[this.align[0]]=this.innerDiv.style[this.align[0]]=this.frzWi+'px';
+    this.frozenTabs.style.width=this.frzWi+'px';
+    this.outerDiv.style.width=(this.frzWi+this.scrWi)+'px';
+  },
+
+/**
+ * Returns the sum of the left & right border widths of an element
+ */
+  borderWidth: function(elem) {
+    var l=Rico.nan2zero(Rico.getStyle(elem,'borderLeftWidth'));
+    var r=Rico.nan2zero(Rico.getStyle(elem,'borderRightWidth'));
+    Rico.log((elem.id || elem.tagName)+' borderWidth: L='+l+', R='+r);
+    return l + r;
+//    return Rico.nan2zero(Rico.getStyle(elem,'borderLeftWidth')) + Rico.nan2zero(Rico.getStyle(elem,'borderRightWidth'));
+  },
+
+  setOtherHdrCellWidths: function() {
+    var c,i,j,r,w,hdrcell,cell,origSpan,newSpan,divs;
+    for (r=0; r<this.hdrCells.length; r++) {
+      if (r==this.headerRowIdx) continue;
+      Rico.log('setOtherHdrCellWidths: r='+r);
+      c=i=0;
+      while (i<this.headerColCnt && c<this.hdrCells[r].length) {
+        hdrcell=this.hdrCells[r][c];
+        cell=hdrcell.cell;
+        origSpan=newSpan=hdrcell.colSpan;
+        for (w=j=0; j<origSpan; j++, i++) {
+          if (this.columns[i].hdrCell.style.display=='none')
+            newSpan--;
+          else if (this.columns[i].hdrColDiv.style.display!='none')
+            w+=parseInt(this.columns[i].colWidth,10);
+        }
+        if (!hdrcell.hdrColDiv || !hdrcell.hdrCellDiv) {
+          divs=cell.getElementsByTagName('div');
+          hdrcell.hdrColDiv=(divs.length<1) ? Rico.wrapChildren(cell,'ricoLG_col') : divs[0];
+          hdrcell.hdrCellDiv=(divs.length<2) ? Rico.wrapChildren(hdrcell.hdrColDiv,'ricoLG_cell') : divs[1];
+        }
+        if (newSpan==0) {
+          cell.style.display='none';
+        } else if (w==0) {
+          hdrcell.hdrColDiv.style.display='none';
+          cell.colSpan=newSpan;
+        } else {
+          cell.style.display='';
+          hdrcell.hdrColDiv.style.display='';
+          cell.colSpan=newSpan;
+          hdrcell.hdrColDiv.style.width=w+'px';
+        }
+        c++;
+      }
+    }
+  },
+
+  initFilterImage: function(filterRowNum){
+    this.filterAnchor=document.getElementById(this.tableId+'_filterLink');
+    if (!this.filterAnchor) return;
+    this.filterRows=Rico.select('tr.'+this.tableId+'_hdg'+filterRowNum);
+    if (this.filterRows.length!=2) return;
+    for (var i=0, r=[]; i<2; i++) r[i]=Rico.select('.ricoLG_cell',this.filterRows[i]);
+    this.filterElements=r[0].concat(r[1]);
+    this.saveHeight = this.filterElements[0].offsetHeight;
+    var pt=Rico.getStyle(this.filterElements[0],'paddingTop');
+    var pb=Rico.getStyle(this.filterElements[0],'paddingBottom');
+    if (pt) this.saveHeight-=parseInt(pt,10);
+    if (pb) this.saveHeight-=parseInt(pb,10);
+    this.rowNum = filterRowNum;
+    this.setFilterImage(false);
+    //Rico.eventBind(this.filterAnchor, 'click', Rico.eventHandle(this,'toggleFilterRow'), false);
+  },
+
+  toggleFilterRow: function() {
+    if ( Rico.visible(this.filterRows[0]) )
+      this.slideFilterUp();
+    else
+      this.slideFilterDown();
+  },
+
+  setFilterImage: function(expandFlag) {
+    var altText=Rico.getPhraseById((expandFlag ? 'show' : 'hide')+'FilterRow');
+    this.filterAnchor.innerHTML = '<img src="'+Rico.imgDir+'tableFilter'+(expandFlag ? 'Expand' : 'Collapse')+'.gif" alt="'+altText+'" border="0">';
+  },
+
+/**
+ * Returns a div for the cell at the specified row and column index.
+ * In SimpleGrid, r can refer to any row in the grid.
+ * In LiveGrid, r refers to a visible row (row 0 is the first visible row).
+ */
+  cell: function(r,c) {
+    return (0<=c && c<this.columns.length && r>=0) ? this.columns[c].cell(r) : null;
+  },
+
+/**
+ * Returns the screen height available for a grid
+ */
+  availHt: function() {
+    var divPos=Rico.cumulativeOffset(this.outerDiv);
+    return Rico.windowHeight()-divPos.top-2*this.options.scrollBarWidth-15;  // allow for scrollbar and some margin
+  },
+
+  setHorizontalScroll: function() {
+    var newLeft=(-this.scrollDiv.scrollLeft)+'px';
+    this.hdrTabs[1].style.marginLeft=newLeft;
+  },
+
+  pluginScroll: function() {
+     if (this.scrollPluggedIn) return;
+     Rico.eventBind(this.scrollDiv,"scroll",this.scrollEventFunc, false);
+     this.scrollPluggedIn=true;
+  },
+
+  unplugScroll: function() {
+     Rico.eventUnbind(this.scrollDiv,"scroll", this.scrollEventFunc , false);
+     this.scrollPluggedIn=false;
+  },
+
+  hideMsg: function() {
+    this.messagePopup.closePopup();
+  },
+
+  showMsg: function(msg) {
+    this.messagePopup.setContent(msg);
+    this.messagePopup.centerPopup();
+    Rico.log("showMsg: "+msg);
+  },
+
+/**
+ * @return array of column objects which have invisible status
+ */
+  listInvisible: function(attr) {
+    var hiddenColumns=[];
+    for (var x=0;x<this.columns.length;x++) {
+      if (!this.columns[x].visible)
+        hiddenColumns.push(attr ? this.columns[x][attr] : this.columns[x]);
+    }
+    return hiddenColumns;
+  },
+
+/**
+ * @return index of left-most visibile column, or -1 if there are no visible columns
+ */
+  firstVisible: function() {
+    for (var x=0;x<this.columns.length;x++) {
+      if (this.columns[x].visible) return x;
+    }
+    return -1;
+  },
+
+/**
+ * Show all columns
+ */
+  showAll: function() {
+    var invisible=this.listInvisible();
+    for (var x=0;x<invisible.length;x++)
+      invisible[x].showColumn();
+  },
+  
+  chooseColumns: function() {
+    this.menu.cancelmenu();
+    var x,z,col,itemDiv,span,contentDiv;\r
+    if (!this.columnChooser) {
+      Rico.log('creating columnChooser');
+      z=Rico.getStyle(this.outerDiv.offsetParent,'zIndex');
+      if (typeof z!='number') z=0;
+      this.columnChooser=new Rico.Window(Rico.getPhraseById('gridChooseCols'), {zIndex:z+2, parent:this.outerDiv});
+      Rico.addClass(this.columnChooser.container, 'ricoLG_chooserDiv');
+      contentDiv=this.columnChooser.contentDiv;
+      for (x=0;x<this.columns.length;x++) {
+        col=this.columns[x];
+        itemDiv=contentDiv.appendChild(document.createElement('div'));
+        col.ChooserBox=Rico.createFormField(itemDiv,'input','checkbox');
+        span=itemDiv.appendChild(document.createElement('span'));
+        span.innerHTML=col.displayName;
+        Rico.eventBind(col.ChooserBox, 'click', Rico.eventHandle(col,'chooseColumn'), false);
+      }
+    }
+    Rico.log('opening columnChooser');
+    this.columnChooser.openPopup(1,this.hdrHt);
+    for (x=0;x<this.columns.length;x++) {
+      this.columns[x].ChooserBox.checked=this.columns[x].visible;
+      this.columns[x].ChooserBox.disabled = !this.columns[x].canHideShow();
+    }
+  },
+
+  blankRow: function(r) {
+    for (var c=0; c < this.columns.length; c++) {
+      this.columns[c].clearCell(r);
+    }
+  },
+  
+  getExportStyles: function(chkelem) {
+    var exportStyles=this.options.exportStyleList;
+    var bgImg=Rico.getStyle(chkelem,'backgroundImage');
+    if (!bgImg || bgImg=='none') return exportStyles;
+    for (var styles=[],i=0; i<exportStyles.length; i++)
+      if (exportStyles[i]!='background-color' && exportStyles[i]!='color') styles.push(exportStyles[i]);
+    return styles;
+  },
+
+/**
+ * Support function for printVisible()
+ */
+  exportStart: function() {
+    var r,c,i,j,hdrcell,newSpan,divs,cell;
+    var exportStyles=this.getExportStyles(this.thead[0]);
+    //alert(exportStyles.join('\n'));
+    this.exportRows=[];
+    this.exportText="<table border='1' cellspacing='0'>";\r
+    for (c=0; c<this.columns.length; c++) {\r
+      if (this.columns[c].visible) this.exportText+="<col width='"+parseInt(this.columns[c].colWidth,10)+"'>";
+    }\r
+    this.exportText+="<thead style='display: table-header-group;'>";
+    if (this.exportHeader) this.exportText+=this.exportHeader;
+    for (r=0; r<this.hdrCells.length; r++) {
+      if (this.hdrCells[r].length==0 || !Rico.visible(this.hdrCells[r][0].cell.parentNode)) continue;
+      this.exportText+="<tr>";
+      for (c=0,i=0; c<this.hdrCells[r].length; c++) {
+        hdrcell=this.hdrCells[r][c];
+        newSpan=hdrcell.colSpan;
+        for (j=0; j<hdrcell.colSpan; j++, i++) {
+          if (!this.columns[i].visible) newSpan--;
+        }
+        if (newSpan > 0) {
+          divs=Rico.select('.ricoLG_cell',hdrcell.cell);
+          cell=divs && divs.length>0 ? divs[0] : hdrcell.cell;
+          this.exportText+="<td style='"+this.exportStyle(cell,exportStyles)+"'";
+          if (hdrcell.colSpan > 1) this.exportText+=" colspan='"+newSpan+"'";
+          this.exportText+=">"+Rico.getInnerText(cell,!this.options.exportImgTags, !this.options.exportFormFields, 'NoExport')+"</td>";
+        }
+      }
+      this.exportText+="</tr>";
+    }
+    this.exportText+="</thead><tbody>";
+  },
+
+/**
+ * Support function for printVisible().
+ * exportType is optional and defaults 'plain'; 'owc' can be used for IE users with Office Web Components.
+ */
+  exportFinish: function(exportType) {
+    if (this.hideMsg) this.hideMsg();
+    window.status=Rico.getPhraseById('exportComplete');
+    if (this.exportRows.length > 0) this.exportText+='<tr>'+this.exportRows.join('</tr><tr>')+'</tr>';
+    if (this.exportFooter) this.exportText+=this.exportFooter;
+    this.exportText+="</tbody></table>";
+    this.exportDiv.innerHTML=this.exportText;
+    this.exportText=undefined;
+    this.exportRows=undefined;
+    if (this.cancelMenu) this.cancelMenu();
+    var w=window.open(Rico.htmDir+'export_'+(exportType || 'plain')+'.html?'+this.exportDiv.id,'',this.options.exportWindow);
+    if (w == null) alert(Rico.getPhraseById('disableBlocker'));
+  },
+
+/**
+ * Support function for printVisible()
+ */
+  exportStyle: function(elem,styleList) {
+    for (var i=0,s=''; i < styleList.length; i++) {
+      try {
+        var curstyle=Rico.getStyle(elem,styleList[i]);
+        if (curstyle) s+=styleList[i]+':'+curstyle+';';
+      } catch(e) {};
+    }
+    return s;
+  },
+
+/**
+ * Gets the value of the grid cookie and interprets the contents.
+ * All information for a particular grid is stored in a single cookie.
+ * This may include column widths, column hide/show status, current sort, and any column filters.
+ */
+  getCookie: function() {
+    var c=Rico.getCookie(this.options.cookiePrefix+this.tableId);
+    if (!c) return;
+    var cookieVals=c.split(',');
+    for (var i=0; i<cookieVals.length; i++) {
+      var v=cookieVals[i].split(':');
+      if (v.length!=2) continue;
+      var colnum=parseInt(v[0].slice(1),10);
+      if (colnum < 0 || colnum >= this.columns.length) continue;
+      var col=this.columns[colnum];
+      switch (v[0].charAt(0)) {
+        case 'w':
+          col.setColWidth(v[1]);
+          col.customWidth=true;
+          break;
+        case 'h':
+          if (v[1].toLowerCase()=='true')
+            col.hideshow(true,true);
+          else
+            col.hideshow(false,true);
+          break;
+        case 's':
+          if (!this.options.saveColumnInfo.sort || !col.sortable) break;
+          col.setSorted(v[1]);
+          break;
+        case 'f':
+          if (!this.options.saveColumnInfo.filter || !col.filterable) break;
+          var filterTemp=v[1].split('~');
+          col.filterOp=filterTemp.shift();
+          col.filterValues = [];
+          col.filterType = Rico.ColumnConst.USERFILTER;
+          for (var j=0; j<filterTemp.length; j++)
+            col.filterValues.push(unescape(filterTemp[j]));
+          break;
+      }
+    }
+  },
+
+/**
+ * Sets the grid cookie.
+ * All information for a particular grid is stored in a single cookie.
+ * This may include column widths, column hide/show status, current sort, and any column filters.
+ */
+  setCookie: function() {
+    var cookieVals=[];
+    for (var i=0; i<this.columns.length; i++) {
+      var col=this.columns[i];
+      if (this.options.saveColumnInfo.width) {
+        if (col.customWidth) cookieVals.push('w'+i+':'+col.colWidth);
+        if (col.customVisible) cookieVals.push('h'+i+':'+col.visible);
+      }
+      if (this.options.saveColumnInfo.sort) {
+        if (col.currentSort != Rico.ColumnConst.UNSORTED)
+          cookieVals.push('s'+i+':'+col.currentSort);
+      }
+      if (this.options.saveColumnInfo.filter && col.filterType == Rico.ColumnConst.USERFILTER) {
+        var filterTemp=[col.filterOp];
+        for (var j=0; j<col.filterValues.length; j++)
+          filterTemp.push(escape(col.filterValues[j]));
+        cookieVals.push('f'+i+':'+filterTemp.join('~'));
+      }
+    }
+    Rico.setCookie(this.options.cookiePrefix+this.tableId, cookieVals.join(','), this.options.cookieDays, this.options.cookiePath, this.options.cookieDomain);
+  }
+
+};
+
+
+Rico.ColumnConst = {
+  UNFILTERED:   0,
+  SYSTEMFILTER: 1,
+  USERFILTER:   2,
+
+  UNSORTED:  0,
+  SORT_ASC:  "ASC",
+  SORT_DESC: "DESC",
+
+  MINWIDTH: 10,
+  DOLLAR:  {type:'number', prefix:'$', decPlaces:2, ClassName:'alignright'},
+  EURO:    {type:'number', prefix:'&euro;', decPlaces:2, ClassName:'alignright'},
+  PERCENT: {type:'number', suffix:'%', decPlaces:2, multiplier:100, ClassName:'alignright'},
+  QTY:     {type:'number', decPlaces:0, ClassName:'alignright'},
+  DEFAULT: {type:"showTags"}
+}
+
+
+/**
+ * @class Define methods that are common to columns in both SimpleGrid and LiveGrid
+ */
+Rico.TableColumnBase = function() {};
+
+Rico.TableColumnBase.prototype = {
+
+/**
+ * Common code used to initialize the column in both SimpleGrid & LiveGrid
+ */
+  baseInit: function(liveGrid,colIdx,hdrInfo,tabIdx) {
+    Rico.log("TableColumnBase.init index="+colIdx+" tabIdx="+tabIdx);
+    this.liveGrid  = liveGrid;
+    this.index     = colIdx;
+    this.hideWidth = Rico.isKonqueror || Rico.isWebKit || liveGrid.headerRowCnt>1 ? 5 : 2;  // column width used for "hidden" columns. Anything less than 5 causes problems with Konqueror. Best to keep this greater than padding used inside cell.
+    this.options   = liveGrid.options;
+    this.tabIdx    = tabIdx;
+    this.hdrCell   = hdrInfo.cell;
+    this.body = document.getElementsByTagName("body")[0];
+    this.displayName  = this.getDisplayName(this.hdrCell);
+    var divs=this.hdrCell.getElementsByTagName('div');
+    this.hdrColDiv=(divs.length<1) ? Rico.wrapChildren(this.hdrCell,'ricoLG_col') : divs[0];
+    this.hdrCellDiv=(divs.length<2) ? Rico.wrapChildren(this.hdrColDiv,'ricoLG_cell') : divs[1];
+    var sectionIndex= tabIdx==0 ? colIdx : colIdx-liveGrid.options.frozenColumns;
+    this.dataCell = liveGrid.tbody[tabIdx].rows[0].cells[sectionIndex];
+    divs=this.dataCell.getElementsByTagName('div');
+    this.dataColDiv=(divs.length<1) ? Rico.wrapChildren(this.dataCell,'ricoLG_col') : divs[0];
+
+    this.mouseDownHandler= Rico.eventHandle(this,'handleMouseDown');
+    this.mouseMoveHandler= Rico.eventHandle(this,'handleMouseMove');
+    this.mouseUpHandler  = Rico.eventHandle(this,'handleMouseUp');
+    this.mouseOutHandler = Rico.eventHandle(this,'handleMouseOut');
+
+    this.fieldName = 'col'+this.index;
+    var spec = liveGrid.options.columnSpecs[colIdx];
+    this.format=Rico.extend( {}, Rico.ColumnConst.DEFAULT);
+    switch (typeof spec) {
+      case 'object':
+        if (typeof spec.format=='string') Rico.extend(this.format, Rico.ColumnConst[spec.format.toUpperCase()]);
+        Rico.extend(this.format, spec);
+        break;
+      case 'string':
+        if (spec.slice(0,4)=='spec') spec=spec.slice(4).toUpperCase();  // for backwards compatibility
+        if (typeof Rico.ColumnConst[spec]=='object') Rico.extend(this.format, Rico.ColumnConst[spec]);
+        break;
+    }
+    Rico.addClass(this.dataColDiv, this.colClassName());
+    this.visible=true;
+    if (typeof this.format.visible=='boolean') this.visible=this.format.visible;
+    Rico.log("TableColumn.init index="+colIdx+" fieldName="+this.fieldName);
+    this.sortable     = typeof this.format.canSort=='boolean' ? this.format.canSort : liveGrid.options.canSortDefault;
+    this.currentSort  = Rico.ColumnConst.UNSORTED;
+    this.filterable   = typeof this.format.canFilter=='boolean' ? this.format.canFilter : liveGrid.options.canFilterDefault;
+    this.filterType   = Rico.ColumnConst.UNFILTERED;
+    this.hideable     = typeof this.format.canHide=='boolean' ? this.format.canHide : liveGrid.options.canHideDefault;
+
+    var wi=(typeof(this.format.width)=='number') ? this.format.width : hdrInfo.initWidth;
+    wi=(typeof(wi)=='number') ? Math.max(wi,Rico.ColumnConst.MINWIDTH) : liveGrid.options.defaultWidth;
+    this.setColWidth(wi);
+    if (!this.visible) this.setDisplay('none');
+  },
+  
+  colClassName: function() {
+    return this.format.ClassName ? this.format.ClassName : this.liveGrid.tableId+'_col'+this.index;
+  },
+
+  insertResizer: function() {
+    //this.hdrCell.style.width='';
+    if (this.format.noResize) return;
+    var resizer=document.createElement('div');
+    resizer.className='ricoLG_Resize';
+    resizer.style[this.liveGrid.align[1]]='0px';
+    this.hdrCellDiv.appendChild(resizer);
+    Rico.eventBind(resizer,"mousedown", this.mouseDownHandler, false);
+  },
+
+/**
+ * get the display name of a column
+ */
+  getDisplayName: function(el) {
+    var anchors=el.getElementsByTagName("A");
+    //Check the existance of A tags
+    if (anchors.length > 0)
+      return anchors[0].innerHTML;
+    else
+      return Rico.stripTags(el.innerHTML);
+  },
+
+  _clear: function(gridCell) {
+    gridCell.innerHTML='&nbsp;';
+  },
+
+  clearCell: function(rowIndex) {
+    var gridCell=this.cell(rowIndex);
+    this._clear(gridCell,rowIndex);
+    if (!this.liveGrid.buffer) return;
+    var acceptAttr=this.liveGrid.buffer.options.acceptAttr;
+    for (var k=0; k<acceptAttr.length; k++) {
+      switch (acceptAttr[k]) {
+        case 'style': gridCell.style.cssText=''; break;
+        case 'class': gridCell.className=''; break;
+        default:      gridCell['_'+acceptAttr[k]]=''; break;
+      }
+    }
+  },
+
+  dataTable: function() {
+    return this.liveGrid.tabs[this.tabIdx];
+  },
+
+  numRows: function() {
+    return this.dataColDiv.childNodes.length;
+  },
+
+  clearColumn: function() {
+    var childCnt=this.numRows();
+    for (var r=0; r<childCnt; r++)
+      this.clearCell(r);
+  },
+
+  cell: function(r) {
+    return this.dataColDiv.childNodes[r];
+  },
+
+  getFormattedValue: function(r,xImg,xForm,xClass) {
+    return Rico.getInnerText(this.cell(r),xImg,xForm,xClass);
+  },
+
+  setColWidth: function(wi) {
+    if (typeof wi=='number') {
+      wi=parseInt(wi,10);
+      if (wi < Rico.ColumnConst.MINWIDTH) return;
+      wi=wi+'px';
+    }
+    Rico.log('setColWidth '+this.index+': '+wi);
+    this.colWidth=wi;
+    this.hdrColDiv.style.width=wi;
+    this.dataColDiv.style.width=wi;
+  },
+
+  pluginMouseEvents: function() {
+    if (this.mousePluggedIn==true) return;
+    Rico.eventBind(this.body,"mousemove", this.mouseMoveHandler, false);
+    Rico.eventBind(this.body,"mouseup",   this.mouseUpHandler  , false);
+    Rico.eventBind(this.body,"mouseout",  this.mouseOutHandler , false);
+    this.mousePluggedIn=true;
+  },
+
+  unplugMouseEvents: function() {
+    Rico.eventUnbind(this.body,"mousemove", this.mouseMoveHandler, false);
+    Rico.eventUnbind(this.body,"mouseup",   this.mouseUpHandler  , false);
+    Rico.eventUnbind(this.body,"mouseout",  this.mouseOutHandler , false);
+    this.mousePluggedIn=false;
+  },
+
+  handleMouseDown: function(e) {
+    this.resizeStart=Rico.eventClient(e).x;
+    this.origWidth=parseInt(this.colWidth,10);
+    var p=Rico.positionedOffset(this.hdrCell);
+    if (this.liveGrid.direction=='rtl') {
+      this.edge=p.left+this.liveGrid.options.scrollBarWidth;
+      switch (this.tabIdx) {
+        case 0: this.edge+=this.liveGrid.innerDiv.offsetWidth; break;
+        case 1: this.edge-=this.liveGrid.scrollDiv.scrollLeft; break;
+      }
+    } else {
+      this.edge=p.left+this.hdrCell.offsetWidth;
+      if (this.tabIdx>0) this.edge+=Rico.nan2zero(this.liveGrid.tabs[0].offsetWidth);
+    }
+    this.liveGrid.resizeDiv.style.left=this.edge+"px";
+    this.liveGrid.resizeDiv.style.display="";
+    this.liveGrid.outerDiv.style.cursor='e-resize';
+    this.tmpHighlight=this.liveGrid.highlightEnabled;
+    this.liveGrid.highlightEnabled=false;
+    this.pluginMouseEvents();
+    Rico.eventStop(e);
+  },
+
+  handleMouseMove: function(e) {
+    var delta=Rico.eventClient(e).x-this.resizeStart;
+    var newWidth=(this.liveGrid.direction=='rtl') ? this.origWidth-delta : this.origWidth+delta;
+    if (newWidth < Rico.ColumnConst.MINWIDTH) return;
+    this.liveGrid.resizeDiv.style.left=(this.edge+delta)+"px";
+    this.colWidth=newWidth;
+    Rico.eventStop(e);
+  },
+
+  handleMouseUp: function(e) {
+    this.unplugMouseEvents();
+    Rico.log('handleMouseUp '+this.liveGrid.tableId);
+    this.liveGrid.outerDiv.style.cursor='';
+    this.liveGrid.resizeDiv.style.display="none";
+    this.setColWidth(this.colWidth);
+    this.customWidth=true;
+    this.liveGrid.setCookie();
+    this.liveGrid.highlightEnabled=this.tmpHighlight;
+    this.liveGrid.sizeDivs();
+    Rico.eventStop(e);
+  },
+
+  handleMouseOut: function(e) {
+    var reltg = Rico.eventRelatedTarget(e) || e.toElement;
+    while (reltg != null && reltg.nodeName.toLowerCase() != 'body')
+      reltg=reltg.parentNode;
+    if (reltg!=null && reltg.nodeName.toLowerCase() == 'body') return true;
+    this.handleMouseUp(e);
+    return true;
+  },
+
+  setDisplay: function(d) {
+    this.hdrCell.style.display=d;
+    this.hdrColDiv.style.display=d;
+    this.dataCell.style.display=d;
+    this.dataColDiv.style.display=d;
+  },
+  
+  hideshow: function(visible,noresize) {
+    this.setDisplay(visible ? '' : 'none');
+    this.liveGrid.cancelMenu();
+    this.visible=visible;
+    this.customVisible=true;
+    if (noresize) return;
+    this.liveGrid.setCookie();
+    this.liveGrid.sizeDivs();
+  },
+
+  hideColumn: function() {
+    Rico.log('hideColumn '+this.liveGrid.tableId);
+    this.hideshow(false,false);
+  },
+
+  showColumn: function() {
+    Rico.log('showColumn '+this.liveGrid.tableId);
+    this.hideshow(true,false);
+  },
+
+  chooseColumn: function(e) {
+    var elem=Rico.eventElement(e);
+    this.hideshow(elem.checked,false);
+  },
+
+  setImage: function() {
+    if ( this.currentSort == Rico.ColumnConst.SORT_ASC ) {
+       this.imgSort.style.display='inline-block';
+       this.imgSort.className=Rico.theme.sortAsc || 'rico-icon ricoLG_sortAsc';
+    } else if ( this.currentSort == Rico.ColumnConst.SORT_DESC ) {
+       this.imgSort.style.display='inline-block';
+       this.imgSort.className=Rico.theme.sortDesc || 'rico-icon ricoLG_sortDesc';
+    } else {
+       this.imgSort.style.display='none';
+    }
+    if (this.filterType == Rico.ColumnConst.USERFILTER) {
+       this.imgFilter.style.display='inline-block';
+       this.imgFilter.title=this.getFilterText();
+    } else {
+       this.imgFilter.style.display='none';
+    }
+  },
+
+  canHideShow: function() {
+    return this.hideable;
+  }
+
+};
diff --git a/minsrc/ricoLiveGrid.js b/minsrc/ricoLiveGrid.js
new file mode 100644 (file)
index 0000000..93d1b9d
--- /dev/null
@@ -0,0 +1,2434 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+if(typeof Rico=='undefined') throw("LiveGrid requires the Rico JavaScript framework");
+
+
+/** @namespace */
+if (!Rico.Buffer) Rico.Buffer = {};
+
+Rico.Buffer.Base = function(dataTable, options) {
+  this.initialize(dataTable, options);
+}
+/** @lends Rico.Buffer.Base# */
+Rico.Buffer.Base.prototype = {
+/**
+ * @class Defines the static buffer class (no AJAX).
+ * Loads buffer with data that already exists in the document as an HTML table or passed via javascript.
+ * Also serves as a base class for AJAX-enabled buffers.
+ * @constructs
+ */
+  initialize: function(dataTable, options) {
+    this.clear();
+    this.updateInProgress = false;
+    this.lastOffset = 0;
+    this.rcvdRowCount = false;  // true if an eof element was included in the last xml response
+    this.foundRowCount = false; // true if an xml response is ever received with eof true
+    this.totalRows = 0;
+    this.rowcntContent = "";
+    this.rcvdOffset = -1;
+    this.options = {
+      fixedHdrRows     : 0,
+      canFilter        : true,  // does buffer object support filtering?
+      isEncoded        : true,  // is the data received via ajax html encoded?
+      acceptAttr       : []     // attributes that can be copied from original/ajax data (e.g. className, style, id)
+    };
+    Rico.extend(this.options, options || {});
+    if (dataTable) {
+      this.loadRowsFromTable(dataTable,this.options.fixedHdrRows);
+    } else {
+      this.clear();
+    }
+  },
+
+  registerGrid: function(liveGrid) {
+    this.liveGrid = liveGrid;
+  },
+
+  setTotalRows: function( newTotalRows ) {
+    if (typeof(newTotalRows)!='number') newTotalRows=this.size;
+    if (this.totalRows == newTotalRows) return;
+    this.totalRows = newTotalRows;
+    if (!this.liveGrid) return;
+    Rico.log("setTotalRows, newTotalRows="+newTotalRows);
+    switch (this.liveGrid.sizeTo) {
+      case 'data':
+        this.liveGrid.resizeWindow();
+        break;
+      case 'datamax':
+        this.liveGrid.setPageSize(newTotalRows);
+        break;
+      default:
+        this.liveGrid.updateHeightDiv();
+        break;
+    }
+  },
+
+  loadRowsFromTable: function(tableElement,firstRow) {
+    var newRows = new Array();
+    var trs = tableElement.getElementsByTagName("tr");
+    for ( var i=firstRow || 0; i < trs.length; i++ ) {
+      var row = new Array();
+      var cells = trs[i].getElementsByTagName("td");
+      for ( var j=0; j < cells.length ; j++ )
+        row[j]=cells[j].innerHTML;
+      newRows.push( row );
+    }
+    this.loadRows(newRows);
+  },
+
+  loadRowsFromArray: function(array2D) {
+    for ( var i=0; i < array2D.length; i++ ) {
+      for ( var j=0; j < array2D[i].length ; j++ ) {
+        array2D[i][j]=array2D[i][j].toString();
+      }
+    }
+    this.loadRows(array2D);
+  },
+
+  loadRows: function(jstable) {
+    this.baseRows = jstable;
+    this.startPos = 0;
+    this.size = this.baseRows.length;
+  },
+
+  dom2jstable: function(rowsElement) {
+    Rico.log('dom2jstable: encoded='+this.options.isEncoded);
+    var newRows = new Array();
+    var trs = rowsElement.getElementsByTagName("tr");
+    for ( var i=0; i < trs.length; i++ ) {
+      var row = new Array();
+      var cells = trs[i].getElementsByTagName("td");
+      for ( var j=0; j < cells.length ; j++ )
+        row[j]=Rico.getContentAsString(cells[j],this.options.isEncoded);
+      newRows.push( row );
+    }
+    return newRows;
+  },
+
+  dom2jstableAttr: function(rowsElement,firstRow) {
+    var acceptAttr=this.options.acceptAttr;
+    Rico.log("dom2jstableAttr start, # attr="+acceptAttr.length);
+    var newRows = new Array();
+    var trs = rowsElement.getElementsByTagName("tr");
+    for ( var i=firstRow || 0; i < trs.length; i++ ) {
+      var row = new Array();
+      var cells = trs[i].getElementsByTagName("td");
+      for ( var j=0; j < cells.length ; j++ ) {
+        row[j]={};
+        for (var k=0; k<acceptAttr.length; k++)
+          row[j]['_'+acceptAttr[k]]=cells[j].getAttribute(acceptAttr[k]);
+        if (Rico.isIE) row[j]._class=cells[j].getAttribute('className');
+      }
+      newRows.push( row );
+    }
+    Rico.log("dom2jstableAttr end");
+    return newRows;
+  },
+
+  _blankRow: function() {
+    var newRow=[];
+    for (var i=0; i<this.liveGrid.columns.length; i++) {
+      newRow[i]='';
+    }
+    return newRow;
+  },
+
+  deleteRows: function(rowIndex,cnt) {
+    this.baseRows.splice(rowIndex,typeof(cnt)=='number' ? cnt : 1);
+    this.liveGrid.isPartialBlank=true;
+    this.size=this.baseRows.length;
+  },
+
+  insertRow: function(beforeRowIndex) {
+    var r=this._blankRow();
+    this.baseRows.splice(beforeRowIndex,0,r);
+    this.size=this.baseRows.length;
+    this.liveGrid.isPartialBlank=true;
+    if (this.startPos < 0) this.startPos=0;
+    return r;
+  },
+
+  appendRows: function(cnt) {
+    var newRows=[];
+    for (var i=0; i<cnt; i++) {
+      var r=this._blankRow();
+      this.baseRows.push(r);
+      newRows.push(r);
+    }
+    this.size=this.baseRows.length;
+    this.liveGrid.isPartialBlank=true;
+    if (this.startPos < 0) this.startPos=0;
+    return newRows;
+  },
+  
+  sortFunc: function(coltype) {
+    var self=this;
+    switch (coltype) {
+      case 'number': return function(a,b) { return self._sortNumeric(a,b); };
+      case 'control':return function(a,b) { return self._sortControl(a,b); };
+      default:       return function(a,b) { return self._sortAlpha(a,b); };
+    }
+  },
+
+  sortBuffer: function(colnum) {
+    if (!this.baseRows) {
+      this.delayedSortCol=colnum;
+      return;
+    }
+    this.liveGrid.showMsg(Rico.getPhraseById("sorting"));
+    this.sortColumn=colnum;
+    var col=this.liveGrid.columns[colnum];
+    this.getValFunc=col._sortfunc;
+    this.baseRows.sort(this.sortFunc(col.format.type));
+    if (col.getSortDirection()=='DESC') this.baseRows.reverse();
+  },
+  
+  _sortAlpha: function(a,b) {
+    var aa = this.sortColumn<a.length ? Rico.getInnerText(a[this.sortColumn]) : '';
+    var bb = this.sortColumn<b.length ? Rico.getInnerText(b[this.sortColumn]) : '';
+    if (aa==bb) return 0;
+    if (aa<bb) return -1;
+    return 1;
+  },
+
+  _sortNumeric: function(a,b) {
+    var aa = this.sortColumn<a.length ? this.nan2zero(Rico.getInnerText(a[this.sortColumn])) : 0;
+    var bb = this.sortColumn<b.length ? this.nan2zero(Rico.getInnerText(b[this.sortColumn])) : 0;
+    return aa-bb;
+  },
+
+  nan2zero: function(n) {
+    if (typeof(n)=='string') n=parseFloat(n);
+    return isNaN(n) || typeof(n)=='undefined' ? 0 : n;
+  },
+  
+  _sortControl: function(a,b) {
+    var aa = this.sortColumn<a.length ? Rico.getInnerText(a[this.sortColumn]) : '';
+    var bb = this.sortColumn<b.length ? Rico.getInnerText(b[this.sortColumn]) : '';
+    if (this.getValFunc) {
+      aa=this.getValFunc(aa);
+      bb=this.getValFunc(bb);
+    }
+    if (aa==bb) return 0;
+    if (aa<bb) return -1;
+    return 1;
+  },
+
+  clear: function() {
+    this.baseRows = [];
+    this.rows = [];
+    this.startPos = -1;
+    this.size = 0;
+    this.windowPos = 0;
+  },
+
+  isInRange: function(position) {
+    var lastRow=Math.min(this.totalRows, position + this.liveGrid.pageSize);
+    return (position >= this.startPos) && (lastRow <= this.endPos()); // && (this.size != 0);
+  },
+
+  endPos: function() {
+    return this.startPos + this.rows.length;
+  },
+
+  fetch: function(offset) {
+    Rico.log('fetch '+this.liveGrid.tableId+': offset='+offset);
+    this.applyFilters();
+    this.setTotalRows();
+    this.rcvdRowCount = true;
+    this.foundRowCount = true;
+    if (offset < 0) offset=0;
+    this.liveGrid.refreshContents(offset);
+    return;
+  },
+
+/**
+ * @return a 2D array of buffer data representing the rows that are currently visible on the grid
+ */
+  visibleRows: function() {
+    return this.rows.slice(this.windowStart,this.windowEnd);
+  },
+
+  setWindow: function(startrow, endrow) {
+    this.windowStart = startrow - this.startPos;  // position in the buffer of first visible row
+    Rico.log('setWindow '+this.liveGrid.tableId+': '+startrow+', '+endrow+', newstart='+this.windowStart);
+    this.windowEnd = Math.min(endrow,this.size);  // position in the buffer of last visible row containing data+1
+    this.windowPos = startrow;                    // position in the dataset of first visible row
+  },
+
+/**
+ * @return true if bufRow is currently visible in the grid
+ */
+  isVisible: function(bufRow) {
+    return bufRow < this.rows.length && bufRow >= this.windowStart && bufRow < this.windowEnd;
+  },
+  
+/**
+ * takes a window row index and returns the corresponding buffer row index
+ */
+  bufferRow: function(windowRow) {
+    return this.windowStart+windowRow;
+  },
+
+/**
+ * @return buffer cell at the specified visible row/col index
+ */
+  getWindowCell: function(windowRow,col) {
+    var bufrow=this.bufferRow(windowRow);
+    return this.isVisible(bufrow) && col < this.rows[bufrow].length ? this.rows[bufrow][col] : null;
+  },
+
+  getWindowAttr: function(windowRow,col) {
+    var bufrow=this.bufferRow(windowRow);
+    return this.attr && this.isVisible(bufrow) && col < this.attr[bufrow].length ? this.attr[bufrow][col] : null;
+  },
+
+  getWindowValue: function(windowRow,col) {
+    return this.getWindowCell(windowRow,col);
+  },
+
+  setWindowValue: function(windowRow,col,newval) {
+    var bufrow=this.bufferRow(windowRow);
+    if (bufrow >= this.windowEnd) return false;
+    return this.setValue(bufrow,col,newval);
+  },
+
+  getCell: function(bufRow,col) {
+    return bufRow < this.size ? this.rows[bufRow][col] : null;
+  },
+
+  getValue: function(bufRow,col) {
+    return this.getCell(bufRow,col);
+  },
+
+  setValue: function(bufRow,col,newval,newstyle) {
+    if (bufRow>=this.size) return false;
+    if (!this.rows[bufRow][col]) this.rows[bufRow][col]={};
+    this.rows[bufRow][col]=newval;
+    if (typeof newstyle=='string') this.rows[bufRow][col]._style=newstyle;
+    this.rows[bufRow][col].modified=true;
+    return true;
+  },
+
+  getRows: function(start, count) {
+    var begPos = start - this.startPos;
+    var endPos = Math.min(begPos + count,this.size);
+    var results = new Array();
+    for ( var i=begPos; i < endPos; i++ ) {
+      results.push(this.rows[i]);
+    }
+    return results;
+  },
+
+  applyFilters: function() {
+    var newRows=[],re=[];
+    var r,c,n,i,showRow,filtercnt;
+    var cols=this.liveGrid.columns;
+    for (n=0,filtercnt=0; n<cols.length; n++) {
+      c=cols[n];
+      if (c.filterType == Rico.ColumnConst.UNFILTERED) continue;
+      filtercnt++;
+      if (c.filterOp=='LIKE') re[n]=new RegExp(c.filterValues[0],'i');
+    }
+    Rico.log('applyFilters: # of filters='+filtercnt);
+    if (filtercnt==0) {
+      this.rows = this.baseRows;
+    } else {
+      for (r=0; r<this.baseRows.length; r++) {
+        showRow=true;
+        for (n=0; n<cols.length && showRow; n++) {
+          c=cols[n];
+          if (c.filterType == Rico.ColumnConst.UNFILTERED) continue;
+          switch (c.filterOp) {
+            case 'LIKE':
+              showRow=re[n].test(this.baseRows[r][n]);
+              break;
+            case 'EQ':
+              showRow=this.baseRows[r][n]==c.filterValues[0];
+              break;
+            case 'NE':
+              for (i=0; i<c.filterValues.length && showRow; i++)
+                showRow=this.baseRows[r][n]!=c.filterValues[i];
+              break;
+            case 'LE':
+              if (c.format.type=='number')
+                showRow=this.nan2zero(this.baseRows[r][n])<=this.nan2zero(c.filterValues[0]);
+              else
+                showRow=this.baseRows[r][n]<=c.filterValues[0];
+              break;
+            case 'GE':
+              if (c.format.type=='number')
+                showRow=this.nan2zero(this.baseRows[r][n])>=this.nan2zero(c.filterValues[0]);
+              else
+                showRow=this.baseRows[r][n]>=c.filterValues[0];
+              break;
+            case 'NULL':
+              showRow=this.baseRows[r][n]=='';
+              break;
+            case 'NOTNULL':
+              showRow=this.baseRows[r][n]!='';
+              break;
+          }
+        }
+        if (showRow) newRows.push(this.baseRows[r]);
+      }
+      this.rows = newRows;
+    }
+    this.rowcntContent = this.size = this.rows.length;
+  },
+
+  printAll: function(exportType) {
+    this.liveGrid.showMsg(Rico.getPhraseById('exportInProgress'));
+    Rico.runLater(10,this,'_printAll',exportType);  // allow message to paint
+  },
+
+/**
+ * Support function for printAll()
+ */
+  _printAll: function(exportType) {
+    this.liveGrid.exportStart();
+    this.exportBuffer(this.getRows(0,this.totalRows));
+    this.liveGrid.exportFinish(exportType);
+  },
+
+/**
+ * Copies visible rows to a new window as a simple html table.
+ */
+  printVisible: function(exportType) {
+    this.liveGrid.showMsg(Rico.getPhraseById('exportInProgress'));
+    Rico.runLater(10,this,'_printVisible',exportType);  // allow message to paint
+  },
+
+  _printVisible: function(exportType) {
+    this.liveGrid.exportStart();
+    this.exportBuffer(this.visibleRows());
+    this.liveGrid.exportFinish(exportType);
+  },
+
+/**
+ * Send all rows to print/export window
+ */
+  exportBuffer: function(rows,startPos) {
+    var r,c,v,col,exportText;
+    Rico.log("exportBuffer: "+rows.length+" rows");
+    var exportStyles=this.liveGrid.getExportStyles(this.liveGrid.tbody[0]);
+    var tdstyle=[];
+    var totalcnt=startPos || 0;
+    var cols=this.liveGrid.columns;
+    for (c=0; c<cols.length; c++) {
+      if (cols[c].visible) tdstyle[c]=this.liveGrid.exportStyle(cols[c].cell(0),exportStyles);  // assumes row 0 style applies to all rows
+    }
+    for(r=0; r < rows.length; r++) {
+      exportText='';
+      for (c=0; c<cols.length; c++) {
+        if (!cols[c].visible) continue;
+        col=cols[c];
+        col.expStyle=tdstyle[c];
+        v=col._export(rows[r][c],rows[r]);
+        if (v=='') v='&nbsp;';
+        exportText+="<td style='"+col.expStyle+"'>"+v+"</td>";
+      }
+      this.liveGrid.exportRows.push(exportText);
+      totalcnt++;
+      if (totalcnt % 10 == 0) window.status=Rico.getPhraseById('exportStatus',totalcnt);
+    }
+  }
+
+};
+
+
+// Rico.LiveGrid -----------------------------------------------------
+
+Rico.LiveGrid = function(tableId, buffer, options) {
+  this.initialize(tableId, buffer, options);
+}
+
+/** 
+ * @lends Rico.LiveGrid#
+ * @property tableId id string for this grid
+ * @property options the options object passed to the constructor extended with defaults
+ * @property buffer the buffer object containing the data for this grid
+ * @property columns array of {@link Rico.LiveGridColumn} objects
+ */
+Rico.LiveGrid.prototype = {
+/**
+ * @class Buffered LiveGrid component
+ * @extends Rico.GridCommon
+ * @constructs
+ */
+  initialize: function( tableId, buffer, options ) {
+    Rico.extend(this, Rico.GridCommon);
+    Rico.extend(this, Rico.LiveGridMethods);
+    this.baseInit();
+    this.tableId = tableId;
+    this.buffer = buffer;
+    this.actionId='_action_'+tableId;
+    Rico.setDebugArea(tableId+"_debugmsgs");    // if used, this should be a textarea
+
+    Rico.extend(this.options, {
+      visibleRows      : -3,    // -1 or 'window'=size grid to client window; -2 or 'data'=size grid to min(window,data); -3 or 'body'=size so body does not have a scrollbar; -4 or 'parent'=size to parent element (e.g. if grid is inside a div)
+      frozenColumns    : 0,
+      offset           : 0,     // first row to be displayed
+      prefetchBuffer   : true,  // load table on page load?
+      minPageRows      : 2,
+      maxPageRows      : 50,
+      canSortDefault   : true,  // can be overridden in the column specs
+      canFilterDefault : buffer.options.canFilter, // can be overridden in the column specs
+      canHideDefault   : true,  // can be overridden in the column specs
+
+      // highlight & selection parameters
+      highlightElem    : 'none',// what gets highlighted/selected (cursorRow, cursorCell, menuRow, menuCell, selection, or none)
+      highlightSection : 3,     // which section gets highlighted (frozen=1, scrolling=2, all=3, none=0)
+      highlightMethod  : 'class', // outline, class, both (outline is less CPU intensive on the client)
+      highlightClass   : Rico.theme.gridHighlightClass || 'ricoLG_selection',
+
+      // export/print parameters
+      maxPrint         : 5000,  // max # of rows that can be printed/exported, 0=disable print/export feature
+
+      // heading parameters
+      headingSort      : 'link', // link: make headings a link that will sort column, hover: make headings a hoverset, none: events on headings are disabled
+      hdrIconsFirst    : true    // true: put sort & filter icons before header text, false: after
+    });
+    // other options:
+    //   sortCol: initial sort column
+
+    var self=this;
+    this.options.sortHandler = function() { self.sortHandler(); };
+    this.options.filterHandler = function() { self.filterHandler(); };
+    this.options.onRefreshComplete = function(firstrow,lastrow) { self.bookmarkHandler(firstrow,lastrow); };
+    this.options.rowOverHandler = Rico.eventHandle(this,'rowMouseOver');
+    this.options.mouseDownHandler = Rico.eventHandle(this,'selectMouseDown');
+    this.options.mouseOverHandler = Rico.eventHandle(this,'selectMouseOver');
+    this.options.mouseUpHandler  = Rico.eventHandle(this,'selectMouseUp');
+    Rico.extend(this.options, options || {});
+
+    switch (typeof this.options.visibleRows) {
+      case 'string':
+        this.sizeTo=this.options.visibleRows;
+        switch (this.options.visibleRows) {
+          case 'data':   this.options.visibleRows=-2; break;
+          case 'body':   this.options.visibleRows=-3; break;
+          case 'parent': this.options.visibleRows=-4; break;
+          case 'datamax':this.options.visibleRows=-5; break;
+          default:       this.options.visibleRows=-1; break;
+        }
+        break;
+      case 'number':
+        switch (this.options.visibleRows) {
+          case -1: this.sizeTo='window'; break;
+          case -2: this.sizeTo='data'; break;
+          case -3: this.sizeTo='body'; break;
+          case -4: this.sizeTo='parent'; break;
+          case -5: this.sizeTo='datamax'; break;
+          default: this.sizeTo='fixed'; break;
+        }
+        break;
+      default:
+        this.sizeTo='body';
+        this.options.visibleRows=-3;
+        break;
+    }
+    this.highlightEnabled=this.options.highlightSection>0;
+    this.pageSize=0;
+    this.createTables();
+    if (this.headerColCnt==0) {
+      alert('ERROR: no columns found in "'+this.tableId+'"');
+      return;
+    }
+    this.createColumnArray('LiveGridColumn');
+    if (this.options.headingSort=='hover')
+      this.createHoverSet();
+
+    this.bookmark=document.getElementById(this.tableId+"_bookmark");
+    this.sizeDivs();
+    var filterUIrow=this.buffer.options.canFilter ? this.options.FilterLocation : false;
+    if (typeof(filterUIrow)=='number' && filterUIrow<0)
+      filterUIrow=this.addHeadingRow('ricoLG_FilterRow');
+    this.createDataCells(this.options.visibleRows);
+    if (this.pageSize == 0) return;
+    this.buffer.registerGrid(this);
+    if (this.buffer.setBufferSize) this.buffer.setBufferSize(this.pageSize);
+    this.scrollTimeout = null;
+    this.lastScrollPos = 0;
+    this.attachMenuEvents();
+
+    this.setSortUI( this.options.sortCol, this.options.sortDir );
+    this.setImages();
+    if (this.listInvisible().length==this.columns.length)
+      this.columns[0].showColumn();
+    this.sizeDivs();
+    this.scrollDiv.style.display="";
+    if (this.buffer.totalRows>0)
+      this.updateHeightDiv();
+    if (this.options.prefetchBuffer) {
+      if (this.bookmark) this.bookmark.innerHTML = Rico.getPhraseById('bookmarkLoading');
+      if (this.options.canFilterDefault && this.options.getQueryParms)
+        this.checkForFilterParms();
+      this.scrollToRow(this.options.offset);
+      this.buffer.fetch(this.options.offset);
+    }
+    if (typeof(filterUIrow)=='number')
+      this.createFilters(filterUIrow);
+    this.scrollEventFunc=Rico.eventHandle(this,'handleScroll');
+    this.wheelEventFunc=Rico.eventHandle(this,'handleWheel');
+    this.wheelEvent=(Rico.isIE || Rico.isOpera || Rico.isWebKit) ? 'mousewheel' : 'DOMMouseScroll';
+    if (this.options.offset && this.options.offset < this.buffer.totalRows)
+      Rico.runLater(50,this,'scrollToRow',this.options.offset);  // Safari requires a delay
+    this.pluginScroll();
+    this.setHorizontalScroll();
+    Rico.log("setHorizontalScroll done");
+    if (this.options.windowResize)
+      Rico.runLater(100,this,'pluginWindowResize');
+    Rico.log("initialize complete for "+this.tableId);
+  }
+};
+
+
+Rico.LiveGridMethods = {
+/** @lends Rico.LiveGrid# */
+
+  createHoverSet: function() {
+    var hdrs=[];
+    for( var c=0; c < this.headerColCnt; c++ ) {
+      if (this.columns[c].sortable) {\r
+        hdrs.push(this.columns[c].hdrCellDiv);
+      }
+    }
+    this.hoverSet = new Rico.HoverSet(hdrs);
+  },
+
+  checkForFilterParms: function() {
+    var s=window.location.search;
+    if (s.charAt(0)=='?') s=s.substring(1);
+    var pairs = s.split('&');
+    for (var i=0; i<pairs.length; i++) {
+      if (pairs[i].match(/^f\[\d+\]/)) {
+        this.buffer.options.requestParameters.push(pairs[i]);
+      }
+    }
+  },
+
+/**
+ * Refreshes a detail grid from a master grid
+ * @returns row index on master table on success, -1 otherwise
+ */
+  drillDown: function(e,masterColNum,detailColNum) {
+    var cell=Rico.eventElement(e || window.event);
+    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
+    if (!cell) return -1;
+    var idx=this.winCellIndex(cell);
+    if (idx.row >= this.buffer.totalRows) return -1
+    this.unhighlight();
+    this.menuIdx=idx;  // ensures selection gets cleared when menu is displayed
+    this.highlight(idx);
+    var drillValue=this.buffer.getWindowCell(idx.row,masterColNum);
+    for (var i=3; i<arguments.length; i++)
+      arguments[i].setDetailFilter(detailColNum,drillValue);
+    return idx.row;
+  },
+
+/**
+ * set filter on a detail grid that is in a master-detail relationship
+ */
+  setDetailFilter: function(colNumber,filterValue) {
+    var c=this.columns[colNumber];
+    c.format.ColData=filterValue;
+    c.setSystemFilter('EQ',filterValue);
+  },
+
+/**
+ * Create one table for frozen columns and one for scrolling columns.
+ * Also create div's to contain them.
+ * @returns true on success
+ */
+  createTables: function() {
+    var insertloc,hdrSrc,i;
+    var table = document.getElementById(this.tableId) || document.getElementById(this.tableId+'_outerDiv');
+    if (!table) return false;
+    if (table.tagName.toLowerCase()=='table') {
+      var theads=table.getElementsByTagName("thead");
+      if (theads.length == 1) {
+        Rico.log("createTables: using thead section, id="+this.tableId);
+        if (this.options.PanelNamesOnTabHdr && this.options.panels) {
+          var r=theads[0].insertRow(0);
+          this.insertPanelNames(r, 0, this.options.frozenColumns, 'ricoFrozen');
+          this.insertPanelNames(r, this.options.frozenColumns, this.options.columnSpecs.length);
+        }
+        hdrSrc=theads[0].rows;
+      } else {
+        Rico.log("createTables: using tbody section, id="+this.tableId);
+        hdrSrc=new Array(table.rows[0]);
+      }
+      insertloc=table;
+    } else if (this.options.columnSpecs.length > 0) {
+      if (!table.id.match(/_outerDiv$/)) insertloc=table;
+      Rico.log("createTables: inserting at "+table.tagName+", id="+this.tableId);
+    } else {
+      alert("ERROR!\n\nUnable to initialize '"+this.tableId+"'\n\nLiveGrid terminated");
+      return false;
+    }
+
+    this.createDivs();
+    this.scrollContainer = this.createDiv("scrollContainer",this.structTabLR);
+    this.scrollContainer.appendChild(this.scrollDiv); // move scrollDiv
+    this.scrollTab = this.createDiv("scrollTab",this.scrollContainer);
+    this.shadowDiv  = this.createDiv("shadow",this.scrollDiv);
+    this.shadowDiv.style.direction='ltr';  // avoid FF bug
+    this.scrollDiv.style.display="none";
+    this.scrollDiv.scrollTop=0;
+    if (this.options.highlightMethod!='class') {
+      this.highlightDiv=[];
+      switch (this.options.highlightElem) {
+        case 'menuRow':
+        case 'cursorRow':
+          this.highlightDiv[0] = this.createDiv("highlight",this.outerDiv);
+          this.highlightDiv[0].style.display="none";
+          break;
+        case 'menuCell':
+        case 'cursorCell':
+          for (i=0; i<2; i++) {
+            this.highlightDiv[i] = this.createDiv("highlight",i==0 ? this.frozenTabs : this.scrollTab);
+            this.highlightDiv[i].style.display="none";
+            this.highlightDiv[i].id+=i;
+          }
+          break;
+        case 'selection':
+          // create one div for each side of the rectangle
+          var parentDiv=this.options.highlightSection==1 ? this.frozenTabs : this.scrollTab;
+          for (i=0; i<4; i++) {
+            this.highlightDiv[i] = this.createDiv("highlight",parentDiv);
+            this.highlightDiv[i].style.display="none";
+            this.highlightDiv[i].style.overflow="hidden";
+            this.highlightDiv[i].id+=i;
+            this.highlightDiv[i].style[i % 2==0 ? 'height' : 'width']="0px";
+          }
+          break;
+      }
+    }
+
+    // create new tables
+    for (i=0; i<3; i++) {
+      this.tabs[i] = document.createElement("table");
+      this.tabs[i].className = 'ricoLG_table';
+      this.tabs[i].border=0;
+      this.tabs[i].cellPadding=0;
+      this.tabs[i].cellSpacing=0;
+      this.tabs[i].id = this.tableId+"_tab"+i;
+    }
+    // set headings
+    for (i=0; i<2; i++) {
+      this.thead[i]=this.tabs[i].createTHead();
+      Rico.addClass(this.tabs[i],'ricoLG_top');
+      //this.thead[i].className='ricoLG_top';
+      if (Rico.theme.gridheader) Rico.addClass(this.thead[i],Rico.theme.gridheader);
+    }
+    // set bodies
+    for (i=0; i<2; i++) {
+      this.tbody[i]=Rico.getTBody(this.tabs[i==0?0:2]);
+      this.tbody[i].className='ricoLG_bottom';
+      if (Rico.theme.gridcontent) Rico.addClass(this.tbody[i],Rico.theme.gridcontent);
+      this.tbody[i].insertRow(-1);
+    }
+    this.frozenTabs.appendChild(this.tabs[0]);
+    this.innerDiv.appendChild(this.tabs[1]);
+    this.scrollTab.appendChild(this.tabs[2]);
+    if (insertloc) insertloc.parentNode.insertBefore(this.outerDiv,insertloc);
+    if (hdrSrc) {
+      this.headerColCnt = this.getColumnInfo(hdrSrc);
+      this.loadHdrSrc(hdrSrc);
+    } else {
+      this.createHdr(0,0,this.options.frozenColumns);
+      this.createHdr(1,this.options.frozenColumns,this.options.columnSpecs.length);
+      if (this.options.PanelNamesOnTabHdr && this.options.panels) {
+        this.insertPanelNames(this.thead[0].insertRow(0), 0, this.options.frozenColumns);
+        this.insertPanelNames(this.thead[1].insertRow(0), this.options.frozenColumns, this.options.columnSpecs.length);
+      }
+      for (i=0; i<2; i++)
+        this.headerColCnt = this.getColumnInfo(this.thead[i].rows);
+    }
+    for( var c=0; c < this.headerColCnt; c++ )
+      this.tbody[c<this.options.frozenColumns ? 0 : 1].rows[0].insertCell(-1);
+    if (insertloc) table.parentNode.removeChild(table);
+    Rico.log('createTables end');
+    return true;
+  },
+
+  createDataCells: function(visibleRows) {
+    if (visibleRows < 0) {
+      for (var i=0; i<this.options.minPageRows; i++)
+        this.appendBlankRow();
+      this.sizeDivs();
+      this.autoAppendRows(this.remainingHt());
+    } else {
+      for( var r=0; r < visibleRows; r++ )
+        this.appendBlankRow();
+    }
+    var s=this.options.highlightSection;
+    if (s & 1) this.attachHighlightEvents(this.tbody[0]);
+    if (s & 2) this.attachHighlightEvents(this.tbody[1]);
+  },
+
+/**
+ * @param colnum column number
+ * @return id string for a filter element
+ */
+  filterId: function(colnum) {
+    return 'RicoFilter_'+this.tableId+'_'+colnum;
+  },
+
+/**
+ * Create filter elements in heading
+ * Reads this.columns[].filterUI to determine type of filter element for each column (t=text box, s=select list, c=custom)
+ * @param r heading row where filter elements will be placed
+ */
+  createFilters: function(r) {
+    for( var c=0; c < this.headerColCnt; c++ ) {
+      var col=this.columns[c];
+      var fmt=col.format;
+      if (typeof fmt.filterUI!='string') continue;
+      var cell=this.hdrCells[r][c].cell;
+      var field,name=this.filterId(c);\r
+      var divs=cell.getElementsByTagName('div');
+      // copy text alignment from data cell
+      var align=Rico.getStyle(this.cell(0,c),'textAlign');
+      divs[1].style.textAlign=align;
+      switch (fmt.filterUI.charAt(0)) {
+        case 't':
+          // text field
+          field=Rico.createFormField(divs[1],'input','text',name,name);
+          var size=fmt.filterUI.match(/\d+/);
+          field.maxLength=fmt.Length || 50;\r
+          field.size=size ? parseInt(size,10) : 10;
+          divs[1].appendChild(Rico.clearButton(Rico.eventHandle(col,'filterClear')));
+          if (col.filterType==Rico.ColumnConst.USERFILTER && col.filterOp=='LIKE') {
+            var v=col.filterValues[0];
+            if (v.charAt(0)=='*') v=v.substr(1);
+            if (v.slice(-1)=='*') v=v.slice(0,-1);
+            field.value=v;
+            col.lastKeyFilter=v;
+          }
+          Rico.eventBind(field,'keyup',Rico.eventHandle(col,'filterKeypress'),false);
+          col.filterField=field;\r
+          break;\r
+        case 'm':
+          // multi-select
+        case 's':
+          // drop-down select
+          field=Rico.createFormField(divs[1],'select',null,name);\r
+          Rico.addSelectOption(field,this.options.FilterAllToken,Rico.getPhraseById("filterAll"));\r
+          col.filterField=field;
+          var options={};\r
+          Rico.extend(options, this.buffer.ajaxOptions);
+          var colnum=typeof(fmt.filterCol)=='number' ? fmt.filterCol : c;
+          options.parameters = {id: this.tableId, distinct:colnum};
+          options.parameters[this.actionId]="query";
+          options.onComplete = this.filterValuesUpdateFunc(c);
+          new Rico.ajaxRequest(this.buffer.dataSource, options);
+          break;\r
+        case 'c':
+          // custom
+          if (typeof col._createFilters == 'function')
+            col._createFilters(divs[1], name);
+          break;
+      }
+    }
+    this.initFilterImage(r);
+  },
+  
+  filterValuesUpdateFunc: function(colnum) {
+    var self=this;
+    return function (request) { self.filterValuesUpdate(colnum,request); };
+  },
+
+/**
+ * update select list filter with values in AJAX response
+ * @returns true on success
+ */
+  filterValuesUpdate: function(colnum,request) {
+    var response = request.responseXML.getElementsByTagName("ajax-response");
+    Rico.log("filterValuesUpdate: "+request.status);
+    if (response == null || response.length != 1) return false;
+    response=response[0];
+    var error = response.getElementsByTagName('error');
+    if (error.length > 0) {
+      Rico.log("Data provider returned an error:\n"+Rico.getContentAsString(error[0],this.buffer.isEncoded));
+      alert(Rico.getPhraseById("requestError",Rico.getContentAsString(error[0],this.buffer.isEncoded)));
+      return false;
+    }\r
+    response=response.getElementsByTagName('response')[0];\r
+    var rowsElement = response.getElementsByTagName('rows')[0];\r
+    var col=this.columns[parseInt(colnum,10)];
+    var rows = this.buffer.dom2jstable(rowsElement);\r
+    var c,opt,v;
+    if (col.filterType==Rico.ColumnConst.USERFILTER && col.filterOp=='EQ') v=col.filterValues[0];
+    Rico.log('filterValuesUpdate: col='+colnum+' rows='+rows.length);
+    switch (col.format.filterUI.charAt(0)) {
+      case 'm':
+        // multi-select
+        col.mFilter = document.body.appendChild(document.createElement("div"));
+        col.mFilter.className = 'ricoLG_mFilter'
+        Rico.hide(col.mFilter);
+        var contentDiv = col.mFilter.appendChild(document.createElement("div"));
+        contentDiv.className = 'ricoLG_mFilter_content'
+        var buttonDiv = col.mFilter.appendChild(document.createElement("div"));
+        buttonDiv.className = 'ricoLG_mFilter_button'
+        col.mFilterButton=buttonDiv.appendChild(document.createElement("button"));
+        col.mFilterButton.innerHTML=Rico.getPhraseById("ok");
+        var eventName=Rico.isWebKit ? 'mousedown' : 'click';
+        Rico.eventBind(col.filterField,eventName,Rico.eventHandle(col,'mFilterSelectClick'));
+        Rico.eventBind(col.mFilterButton,'click',Rico.eventHandle(col,'mFilterFinish'));
+        //col.filterField.options[0].text=$('AllLabel').innerHTML;
+        tab = contentDiv.appendChild(document.createElement("table"));
+        tab.border=0;
+        tab.cellPadding=2;
+        tab.cellSpacing=0;
+        //tbody=(tab.tBodies.length==0) ? tab.appendChild(document.createElement("tbody")) : tab.tBodies[0];
+        var baseId=this.filterId(colnum)+'_';
+        this.createMFilterItem(tab,this.options.FilterAllToken,Rico.getPhraseById("filterAll"),baseId+'all',Rico.eventHandle(col,'mFilterAllClick'));
+        var handle=Rico.eventHandle(col,'mFilterOtherClick')
+        for (var i=0; i<rows.length; i++) {
+          if (rows[i].length>0) {
+            c=rows[i][0];
+            this.createMFilterItem(tab,c,c || Rico.getPhraseById("filterBlank"),baseId+i,handle);
+          }
+        }
+        col.mFilterInputs=contentDiv.getElementsByTagName('input');
+        col.mFilterLabels=contentDiv.getElementsByTagName('label');
+        col.mFilterFocus=col.mFilterInputs.length ? col.mFilterInputs[0] : col.mFilterButton;
+        break;
+
+      case 's':
+        // drop-down select
+        for (var i=0; i<rows.length; i++) {
+          if (rows[i].length>0) {
+            c=rows[i][0];
+            opt=Rico.addSelectOption(col.filterField,c,c || Rico.getPhraseById("filterBlank"));
+            if (col.filterType==Rico.ColumnConst.USERFILTER && c==v) opt.selected=true;
+          }
+        }
+        Rico.eventBind(col.filterField,'change',Rico.eventHandle(col,'filterChange'));
+        break;
+    }
+    return true;\r
+  },
+  
+  createMFilterItem: function(table,code,description,id,eventHandle) {
+    var tr=table.insertRow(-1);
+    tr.vAlign='top';
+    if (tr.rowIndex % 2 == 1) tr.className='ricoLG_mFilter_oddrow';
+    var td1=tr.insertCell(-1)
+    var td2=tr.insertCell(-1)
+    var field=Rico.createFormField(td1,'input','checkbox',id);
+    field.value=code;
+    field.checked=true;
+    var label = td2.appendChild(document.createElement("label"));
+    label.htmlFor = id;
+    label.innerHTML=description;
+    Rico.eventBind(field,'click',eventHandle);
+  },
+
+  unplugHighlightEvents: function() {
+    var s=this.options.highlightSection;
+    if (s & 1) this.detachHighlightEvents(this.tbody[0]);
+    if (s & 2) this.detachHighlightEvents(this.tbody[1]);
+  },
+
+/**
+ * place panel names on first row of grid header (used by LiveGridForms)
+ */
+  insertPanelNames: function(r,start,limit,cellClass) {
+    Rico.log('insertPanelNames: start='+start+' limit='+limit);
+    r.className='ricoLG_hdg';
+    var lastIdx=-1, span, newCell=null, spanIdx=0;
+    for( var c=start; c < limit; c++ ) {
+      if (lastIdx == this.options.columnSpecs[c].panelIdx) {
+        span++;
+      } else {
+        if (newCell) newCell.colSpan=span;
+        newCell = r.insertCell(-1);
+        if (cellClass) newCell.className=cellClass;
+        span=1;
+        lastIdx=this.options.columnSpecs[c].panelIdx;
+        newCell.innerHTML=this.options.panels[lastIdx];
+      }
+    }
+    if (newCell) newCell.colSpan=span;
+  },
+
+/**
+ * create grid header for table i (if none was provided)
+ */
+  createHdr: function(i,start,limit) {
+    Rico.log('createHdr: i='+i+' start='+start+' limit='+limit);
+    var mainRow = this.thead[i].insertRow(-1);
+    mainRow.id=this.tableId+'_tab'+i+'h_main';
+    mainRow.className='ricoLG_hdg';
+    for( var c=start; c < limit; c++ ) {
+      var newCell = mainRow.insertCell(-1);
+      newCell.innerHTML=this.options.columnSpecs[c].Hdg;
+    }
+  },
+
+/**
+ * move header cells in original table to grid
+ */
+  loadHdrSrc: function(hdrSrc) {
+    var i,h,c,r,newrow,cells;
+    Rico.log('loadHdrSrc start');
+    for (i=0; i<2; i++) {
+      for (r=0; r<hdrSrc.length; r++) {
+        newrow = this.thead[i].insertRow(-1);
+        newrow.className='ricoLG_hdg '+this.tableId+'_hdg'+r;
+      }
+    }
+    if (hdrSrc.length==1) {
+      cells=hdrSrc[0].cells;
+      for (c=0; cells.length > 0; c++)
+        this.thead[c<this.options.frozenColumns ? 0 : 1].rows[0].appendChild(cells[0]);
+    } else {
+      for (r=0; r<hdrSrc.length; r++) {
+        cells=hdrSrc[r].cells;
+        for (c=0,h=0; cells.length > 0; c++) {
+          if (cells[0].className=='ricoFrozen') {
+            if (r==this.headerRowIdx) this.options.frozenColumns=c+1;
+          } else {
+            h=1;
+          }
+          this.thead[h].rows[r].appendChild(cells[0]);
+        }
+      }
+    }
+    Rico.log('loadHdrSrc end');
+  },
+
+/**
+ * Size div elements
+ */
+  sizeDivs: function() {
+    Rico.log('sizeDivs: '+this.tableId);
+    //this.cancelMenu();
+    this.unhighlight();
+    this.baseSizeDivs();
+    var firstVisible=this.firstVisible();
+    if (this.pageSize == 0 || firstVisible < 0) return;
+    var totRowHt=this.columns[firstVisible].dataColDiv.offsetHeight;
+    this.rowHeight = Math.round(totRowHt/this.pageSize);
+    var scrHt=this.dataHt;
+    if (this.scrTabWi0 == this.scrTabWi) {
+      // no scrolling columns - horizontal scroll bar not needed
+      this.innerDiv.style.height=(this.hdrHt+1)+'px';
+      this.scrollDiv.style.overflowX='hidden';
+    } else {
+      this.scrollDiv.style.overflowX='scroll';
+      scrHt+=this.options.scrollBarWidth;
+    }
+    this.scrollDiv.style.height=scrHt+'px';
+    this.innerDiv.style.width=(this.scrWi)+'px';
+    this.scrollTab.style.width=(this.scrWi-this.options.scrollBarWidth)+'px';
+    //this.resizeDiv.style.height=this.frozenTabs.style.height=this.innerDiv.style.height=(this.hdrHt+this.dataHt+1)+'px';
+    this.resizeDiv.style.height=(this.hdrHt+this.dataHt+1)+'px';
+    Rico.log('sizeDivs scrHt='+scrHt+' innerHt='+this.innerDiv.style.height+' rowHt='+this.rowHeight+' pageSize='+this.pageSize);
+    var pad=(this.scrWi-this.scrTabWi < this.options.scrollBarWidth) ? 2 : 0;
+    this.shadowDiv.style.width=(this.scrTabWi+pad)+'px';
+    this.outerDiv.style.height=(this.hdrHt+scrHt)+'px';
+    this.setHorizontalScroll();
+  },
+
+  setHorizontalScroll: function() {
+    var newLeft=(-this.scrollDiv.scrollLeft)+'px';
+    this.tabs[1].style.marginLeft=newLeft;
+    this.tabs[2].style.marginLeft=newLeft;
+  },
+
+  remainingHt: function() {
+    var tabHt;
+    var winHt=Rico.windowHeight();
+    var margin=Rico.isIE ? 15 : 10;
+    // if there is a horizontal scrollbar take it into account
+    if (!Rico.isIE && window.frameElement && window.frameElement.scrolling=='yes' && this.sizeTo!='parent') margin+=this.options.scrollBarWidth;
+    switch (this.sizeTo) {
+      case 'window':
+        var divTop=Rico.cumulativeOffset(this.outerDiv).top;
+        tabHt=Math.max(this.tabs[0].offsetHeight,this.tabs[1].offsetHeight);
+        Rico.log("remainingHt, winHt="+winHt+' tabHt='+tabHt+' gridY='+divTop);
+        return winHt-divTop-tabHt-this.options.scrollBarWidth-margin;  // allow for scrollbar and some margin
+      case 'parent':
+        var offset=this.offsetFromParent(this.outerDiv);
+        tabHt=Math.max(this.tabs[0].offsetHeight,this.tabs[1].offsetHeight);
+        if (Rico.isIE) Rico.hide(this.outerDiv);
+        var parentHt=this.outerDiv.parentNode.offsetHeight;
+        if (Rico.isIE) Rico.show(this.outerDiv);
+        Rico.log("remainingHt, parentHt="+parentHt+' gridY='+offset+' winHt='+winHt+' tabHt='+tabHt);
+        return parentHt - tabHt - offset - this.options.scrollBarWidth;
+      case 'data':
+      case 'body':
+        var bodyHt=Rico.isIE ? document.body.scrollHeight : document.body.offsetHeight;
+        //alert("remainingHt\n document.height="+document.height+"\n body.offsetHeight="+document.body.offsetHeight+"\n body.scrollHeight="+document.body.scrollHeight+"\n documentElement.scrollHeight="+document.documentElement.scrollHeight);
+        var remHt=winHt-bodyHt-margin;
+        if (!Rico.isWebKit) remHt-=this.options.scrollBarWidth;
+        Rico.log("remainingHt, winHt="+winHt+' pageHt='+bodyHt+' remHt='+remHt);
+        return remHt;
+      default:
+        tabHt=Math.max(this.tabs[0].offsetHeight,this.tabs[1].offsetHeight);
+        Rico.log("remainingHt, winHt="+winHt+' tabHt='+tabHt);
+        if (this.sizeTo.slice(-1)=='%') winHt*=parseFloat(this.sizeTo)/100.0;
+        else if (this.sizeTo.slice(-2)=='px') winHt=parseInt(this.sizeTo,10);
+        return winHt-tabHt-this.options.scrollBarWidth-margin;  // allow for scrollbar and some margin
+    }
+  },
+
+  offsetFromParent: function(element) {
+    var valueT = 0;
+    var elParent=element.parentNode;
+    do {
+      //Rico.log("offsetFromParent: "+element.tagName+' id='+element.id+' otop='+element.offsetTop);
+      valueT += element.offsetTop  || 0;
+      element = element.offsetParent;
+      if (!element || element==null) break;
+      var p = Rico.getStyle(element, 'position');
+      if (element.tagName=='BODY' || element.tagName=='HTML' || p=='absolute') return valueT-elParent.offsetTop;
+    } while (element != elParent);
+    return valueT;
+  },
+
+  adjustPageSize: function() {
+    var remHt=this.remainingHt();
+    Rico.log('adjustPageSize remHt='+remHt+' lastRow='+this.lastRowPos);
+    if (remHt > this.rowHeight)
+      this.autoAppendRows(remHt);
+    else if (remHt < 0 || this.sizeTo=='data')
+      this.autoRemoveRows(-remHt);
+  },
+  
+  setPageSize: function(newRowCount) {
+    newRowCount=Math.min(newRowCount,this.options.maxPageRows);
+    newRowCount=Math.max(newRowCount,this.options.minPageRows);
+    this.sizeTo='fixed';
+    var oldSize=this.pageSize;
+    while (this.pageSize > newRowCount) {
+      this.removeRow();
+    }
+    while (this.pageSize < newRowCount) {
+      this.appendBlankRow();
+    }
+    this.finishResize(oldSize);
+  },
+
+  pluginWindowResize: function() {
+    Rico.log("pluginWindowResize");
+    this.resizeWindowHandler=Rico.eventHandle(this,'resizeWindow');
+    Rico.eventBind(window, "resize", this.resizeWindowHandler, false);
+  },
+
+  unplugWindowResize: function() {
+    if (!this.resizeWindowHandler) return;
+    Rico.eventUnbind(window,"resize", this.resizeWindowHandler, false);
+    this.resizeWindowHandler=null;
+  },
+
+  resizeWindow: function() {
+    Rico.log('resizeWindow '+this.tableId+' lastRow='+this.lastRowPos);
+    if (this.resizeState=='finish') {
+      Rico.log('resizeWindow postponed');
+      this.resizeState='resize';
+      return;
+    }
+    if (!this.sizeTo || this.sizeTo=='fixed') {
+      this.sizeDivs();
+      return;
+    }
+    if (this.sizeTo=='parent' && Rico.getStyle(this.outerDiv.parentNode,'display') == 'none') return;
+    var oldSize=this.pageSize;
+    this.adjustPageSize();
+    this.finishResize(oldSize);
+  },
+
+  finishResize: function(oldSize) {
+    if (this.pageSize > oldSize && this.buffer.totalRows>0) {
+      this.isPartialBlank=true;
+      var adjStart=this.adjustRow(this.lastRowPos);
+      this.buffer.fetch(adjStart);
+    } else if (this.pageSize < oldSize) {
+      if (this.options.onRefreshComplete) this.options.onRefreshComplete(this.contentStartPos,this.contentStartPos+this.pageSize-1);  // update bookmark
+    }
+    this.resizeState='finish';
+    Rico.runLater(20,this,'finishResize2');
+    Rico.log('Resize '+this.tableId+' complete. old size='+oldSize+' new size='+this.pageSize);
+  },
+
+  finishResize2: function() {
+    this.sizeDivs();
+    this.updateHeightDiv();
+    if (this.resizeState=='resize') {
+      this.resizeWindow();
+    } else {
+      this.resizeState='';
+    }
+  },
+
+  topOfLastPage: function() {
+    return Math.max(this.buffer.totalRows-this.pageSize,0);
+  },
+
+  updateHeightDiv: function() {
+    var notdisp=this.topOfLastPage();
+    var ht = notdisp ? this.scrollDiv.clientHeight + Math.floor(this.rowHeight * (notdisp + 0.4)) : 1;
+    Rico.log("updateHeightDiv, ht="+ht+' scrollDiv.clientHeight='+this.scrollDiv.clientHeight+' rowsNotDisplayed='+notdisp);
+    this.shadowDiv.style.height=ht+'px';
+  },
+
+  autoRemoveRows: function(overage) {
+    if (!this.rowHeight) return;
+    var removeCnt=Math.ceil(overage / this.rowHeight);
+    if (this.sizeTo=='data')
+      removeCnt=Math.max(removeCnt,this.pageSize-this.buffer.totalRows);
+    Rico.log("autoRemoveRows overage="+overage+" removeCnt="+removeCnt);
+    for (var i=0; i<removeCnt; i++)
+      this.removeRow();
+  },
+
+  removeRow: function() {
+    if (this.pageSize <= this.options.minPageRows) return;
+    this.pageSize--;
+    for( var c=0; c < this.headerColCnt; c++ ) {
+      var cell=this.columns[c].cell(this.pageSize);
+      this.columns[c].dataColDiv.removeChild(cell);
+    }
+  },
+
+  autoAppendRows: function(overage) {
+    if (!this.rowHeight) return;
+    var addCnt=Math.floor(overage / this.rowHeight);
+    Rico.log("autoAppendRows overage="+overage+" cnt="+addCnt+" rowHt="+this.rowHeight);
+    for (var i=0; i<addCnt; i++) {
+      if (this.sizeTo=='data' && this.pageSize>=this.buffer.totalRows) break;
+      this.appendBlankRow();
+    }
+  },
+
+/**
+ * on older systems, this can be fairly slow
+ */
+  appendBlankRow: function() {
+    if (this.pageSize >= this.options.maxPageRows) return;
+    Rico.log("appendBlankRow #"+this.pageSize);
+    var cls=this.defaultRowClass(this.pageSize);
+    for( var c=0; c < this.headerColCnt; c++ ) {
+      var newdiv = document.createElement("div");
+      newdiv.className = 'ricoLG_cell '+cls;
+      newdiv.id=this.tableId+'_'+this.pageSize+'_'+c;
+      this.columns[c].dataColDiv.appendChild(newdiv);
+      if (this.columns[c].format.canDrag && Rico.registerDraggable)
+        Rico.registerDraggable( new Rico.LiveGridDraggable(this, this.pageSize, c), this.options.dndMgrIdx );
+      newdiv.innerHTML='&nbsp;';   // this seems to be required by IE
+      if (this.columns[c]._create) {
+        this.columns[c]._create(newdiv,this.pageSize);
+      }
+    }
+    this.pageSize++;
+  },
+
+  defaultRowClass: function(rownum) {
+    var cls
+    if (rownum % 2==0) {
+      cls='ricoLG_evenRow';
+      //if (Rico.theme.primary) cls+=' '+Rico.theme.primary;
+    } else {
+      cls='ricoLG_oddRow';
+      //if (Rico.theme.secondary) cls+=' '+Rico.theme.secondary;
+    }
+    return cls;
+  },
+
+  handleMenuClick: function(e) {
+    if (!this.menu) return;
+    this.cancelMenu();
+    this.unhighlight(); // in case highlighting was invoked externally
+    var idx;
+    var cell=Rico.eventElement(e);
+    if (cell.className=='ricoLG_highlightDiv') {
+      idx=this.highlightIdx;
+    } else {
+      cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
+      if (!cell) return;
+      idx=this.winCellIndex(cell);
+      if ((this.options.highlightSection & (idx.tabIdx+1))==0) return;
+    }
+    this.highlight(idx);
+    this.highlightEnabled=false;
+    if (this.hideScroll) this.scrollDiv.style.overflow="hidden";
+    this.menuIdx=idx;
+    if (!this.menu.div) this.menu.createDiv();
+    this.menu.liveGrid=this;
+    if (this.menu.buildGridMenu) {
+      var showMenu=this.menu.buildGridMenu(idx.row, idx.column, idx.tabIdx);
+      if (!showMenu) return;
+    }
+    if (this.options.highlightElem=='selection' && !this.isSelected(idx.cell)) {
+      this.selectCell(idx.cell);
+    }
+    var self=this;
+    this.menu.showmenu(e,function() { self.closeMenu(); });
+  },
+
+  closeMenu: function() {
+    if (!this.menuIdx) return;
+    if (this.hideScroll) this.scrollDiv.style.overflow="";
+    //this.unhighlight();
+    this.highlightEnabled=true;
+    this.menuIdx=null;
+  },
+
+/**
+ * @return index of cell within the window
+ */
+  winCellIndex: function(cell) {
+    var l=cell.id.lastIndexOf('_',cell.id.length);
+    var l2=cell.id.lastIndexOf('_',l-1)+1;
+    var c=parseInt(cell.id.substr(l+1));
+    var r=parseInt(cell.id.substr(l2,l));
+    return {row:r, column:c, tabIdx:this.columns[c].tabIdx, cell:cell};
+  },
+
+/**
+ * @return index of cell within the dataset
+ */
+  datasetIndex: function(cell) {
+    var idx=this.winCellIndex(cell);
+    idx.row+=this.buffer.windowPos;
+    idx.onBlankRow=(idx.row >= this.buffer.endPos());
+    return idx;
+  },
+
+  attachHighlightEvents: function(tBody) {
+    switch (this.options.highlightElem) {
+      case 'selection':
+        Rico.eventBind(tBody,"mousedown", this.options.mouseDownHandler, false);
+        /** @ignore */
+        tBody.ondrag = function () { return false; };
+        /** @ignore */
+        tBody.onselectstart = function () { return false; };
+        break;
+      case 'cursorRow':
+      case 'cursorCell':
+        Rico.eventBind(tBody,"mouseover", this.options.rowOverHandler, false);
+        break;
+    }
+  },
+
+  detachHighlightEvents: function(tBody) {
+    switch (this.options.highlightElem) {
+      case 'selection':
+        Rico.eventUnbind(tBody,"mousedown", this.options.mouseDownHandler, false);
+        tBody.ondrag = null;
+        tBody.onselectstart = null;
+        break;
+      case 'cursorRow':
+      case 'cursorCell':
+        Rico.eventUnbind(tBody,"mouseover", this.options.rowOverHandler, false);
+        break;
+    }
+  },
+
+/**
+ * @return array of objects containing row/col indexes (index values are relative to the start of the window)
+ */
+  getVisibleSelection: function() {
+    var cellList=[];
+    if (this.SelectIdxStart && this.SelectIdxEnd) {
+      var r1=Math.max(Math.min(this.SelectIdxEnd.row,this.SelectIdxStart.row)-this.buffer.startPos,this.buffer.windowStart);
+      var r2=Math.min(Math.max(this.SelectIdxEnd.row,this.SelectIdxStart.row)-this.buffer.startPos,this.buffer.windowEnd-1);
+      var c1=Math.min(this.SelectIdxEnd.column,this.SelectIdxStart.column);
+      var c2=Math.max(this.SelectIdxEnd.column,this.SelectIdxStart.column);
+      //Rico.log("getVisibleSelection "+r1+','+c1+' to '+r2+','+c2+' ('+this.SelectIdxStart.row+',startPos='+this.buffer.startPos+',windowPos='+this.buffer.windowPos+',windowEnd='+this.buffer.windowEnd+')');
+      for (var r=r1; r<=r2; r++) {
+        for (var c=c1; c<=c2; c++)
+          cellList.push({row:r-this.buffer.windowStart,column:c});
+      }
+    }
+    if (this.SelectCtrl) {
+      for (var i=0; i<this.SelectCtrl.length; i++) {
+        if (this.SelectCtrl[i].row>=this.buffer.windowStart && this.SelectCtrl[i].row<this.buffer.windowEnd)
+          cellList.push({row:this.SelectCtrl[i].row-this.buffer.windowStart,column:this.SelectCtrl[i].column});
+      }
+    }
+    return cellList;
+  },
+
+  updateSelectOutline: function() {
+    if (!this.SelectIdxStart || !this.SelectIdxEnd) return;
+    var r1=Math.max(Math.min(this.SelectIdxEnd.row,this.SelectIdxStart.row), this.buffer.windowStart);
+    var r2=Math.min(Math.max(this.SelectIdxEnd.row,this.SelectIdxStart.row), this.buffer.windowEnd-1);
+    if (r1 > r2) {
+      this.HideSelection();
+      return;
+    }
+    var c1=Math.min(this.SelectIdxEnd.column,this.SelectIdxStart.column);
+    var c2=Math.max(this.SelectIdxEnd.column,this.SelectIdxStart.column);
+    var top1=this.columns[c1].cell(r1-this.buffer.windowStart).offsetTop;
+    var cell2=this.columns[c1].cell(r2-this.buffer.windowStart);
+    var bottom2=cell2.offsetTop+cell2.offsetHeight;
+    var left1=this.columns[c1].dataCell.offsetLeft;
+    var left2=this.columns[c2].dataCell.offsetLeft;
+    var right2=left2+this.columns[c2].dataCell.offsetWidth;
+    //window.status='updateSelectOutline: '+r1+' '+r2+' top='+top1+' bot='+bottom2;
+    this.highlightDiv[0].style.top=this.highlightDiv[3].style.top=this.highlightDiv[1].style.top=(this.hdrHt+top1-1) + 'px';
+    this.highlightDiv[2].style.top=(this.hdrHt+bottom2-1)+'px';
+    this.highlightDiv[3].style.left=(left1-2)+'px';
+    this.highlightDiv[0].style.left=this.highlightDiv[2].style.left=(left1-1)+'px';
+    this.highlightDiv[1].style.left=(right2-1)+'px';
+    this.highlightDiv[0].style.width=this.highlightDiv[2].style.width=(right2-left1-1) + 'px';
+    this.highlightDiv[1].style.height=this.highlightDiv[3].style.height=(bottom2-top1) + 'px';
+    //this.highlightDiv[0].style.right=this.highlightDiv[2].style.right=this.highlightDiv[1].style.right=()+'px';
+    //this.highlightDiv[2].style.bottom=this.highlightDiv[3].style.bottom=this.highlightDiv[1].style.bottom=(this.hdrHt+bottom2) + 'px';
+    for (var i=0; i<4; i++)
+      this.highlightDiv[i].style.display='';
+  },
+
+  HideSelection: function() {
+    var i;
+    if (this.options.highlightMethod!='class') {
+      for (i=0; i<this.highlightDiv.length; i++)
+        this.highlightDiv[i].style.display='none';
+    }
+    if (this.options.highlightMethod!='outline') {
+      var cellList=this.getVisibleSelection();
+      Rico.log("HideSelection "+cellList.length);
+      for (i=0; i<cellList.length; i++)
+        this.unhighlightCell(this.columns[cellList[i].column].cell(cellList[i].row));
+    }
+  },
+
+  ShowSelection: function() {
+    if (this.options.highlightMethod!='class')
+      this.updateSelectOutline();
+    if (this.options.highlightMethod!='outline') {
+      var cellList=this.getVisibleSelection();
+      for (var i=0; i<cellList.length; i++)
+        this.highlightCell(this.columns[cellList[i].column].cell(cellList[i].row));
+    }
+  },
+
+  ClearSelection: function() {
+    Rico.log("ClearSelection");
+    this.HideSelection();
+    this.SelectIdxStart=null;
+    this.SelectIdxEnd=null;
+    this.SelectCtrl=[];
+  },
+
+  selectCell: function(cell) {
+    this.ClearSelection();
+    this.SelectIdxStart=this.SelectIdxEnd=this.datasetIndex(cell);
+    this.ShowSelection();
+  },
+
+  AdjustSelection: function(cell) {
+    var newIdx=this.datasetIndex(cell);
+    if (this.SelectIdxStart.tabIdx != newIdx.tabIdx) return;
+    this.HideSelection();
+    this.SelectIdxEnd=newIdx;
+    this.ShowSelection();
+  },
+
+  RefreshSelection: function() {
+    var cellList=this.getVisibleSelection();
+    for (var i=0; i<cellList.length; i++) {
+      this.columns[cellList[i].column].displayValue(cellList[i].row);
+    }
+  },
+
+  FillSelection: function(newVal,newStyle) {
+    if (this.SelectIdxStart && this.SelectIdxEnd) {
+      var r1=Math.min(this.SelectIdxEnd.row,this.SelectIdxStart.row);
+      var r2=Math.max(this.SelectIdxEnd.row,this.SelectIdxStart.row);
+      var c1=Math.min(this.SelectIdxEnd.column,this.SelectIdxStart.column);
+      var c2=Math.max(this.SelectIdxEnd.column,this.SelectIdxStart.column);
+      for (var r=r1; r<=r2; r++) {
+        for (var c=c1; c<=c2; c++) {
+          this.buffer.setValue(r,c,newVal,newStyle);
+        }
+      }
+    }
+    if (this.SelectCtrl) {
+      for (var i=0; i<this.SelectCtrl.length; i++) {
+        this.buffer.setValue(this.SelectCtrl[i].row,this.SelectCtrl[i].column,newVal,newStyle);
+      }
+    }
+    this.RefreshSelection();
+  },
+
+/**
+ * Process mouse down event
+ * @param e event object
+ */
+  selectMouseDown: function(e) {
+    if (this.highlightEnabled==false) return true;
+    this.cancelMenu();
+    var cell=Rico.eventElement(e);
+    if (!Rico.eventLeftClick(e)) return true;
+    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
+    if (!cell) return true;
+    Rico.eventStop(e);
+    var newIdx=this.datasetIndex(cell);
+    if (newIdx.onBlankRow) return true;
+    Rico.log("selectMouseDown @"+newIdx.row+','+newIdx.column);
+    if (e.ctrlKey) {
+      if (!this.SelectIdxStart || this.options.highlightMethod!='class') return true;
+      if (!this.isSelected(cell)) {
+        this.highlightCell(cell);
+        this.SelectCtrl.push(this.datasetIndex(cell));
+      } else {
+        for (var i=0; i<this.SelectCtrl.length; i++) {
+          if (this.SelectCtrl[i].row==newIdx.row && this.SelectCtrl[i].column==newIdx.column) {
+            this.unhighlightCell(cell);
+            this.SelectCtrl.splice(i,1);
+            break;
+          }
+        }
+      }
+    } else if (e.shiftKey) {
+      if (!this.SelectIdxStart) return true;
+      this.AdjustSelection(cell);
+    } else {
+      this.selectCell(cell);
+      this.pluginSelect();
+    }
+    return false;
+  },
+
+  pluginSelect: function() {
+    if (this.selectPluggedIn) return;
+    var tBody=this.tbody[this.SelectIdxStart.tabIdx];
+    Rico.eventBind(tBody,"mouseover", this.options.mouseOverHandler, false);
+    Rico.eventBind(this.outerDiv,"mouseup",  this.options.mouseUpHandler,  false);
+    this.selectPluggedIn=true;
+  },
+
+  unplugSelect: function() {
+    if (!this.selectPluggedIn) return;
+    var tBody=this.tbody[this.SelectIdxStart.tabIdx];
+    Rico.eventUnbind(tBody,"mouseover", this.options.mouseOverHandler , false);
+    Rico.eventUnbind(this.outerDiv,"mouseup", this.options.mouseUpHandler , false);
+    this.selectPluggedIn=false;
+  },
+
+  selectMouseUp: function(e) {
+    this.unplugSelect();
+    var cell=Rico.eventElement(e);
+    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
+    if (!cell) return;
+    if (this.SelectIdxStart && this.SelectIdxEnd)
+      this.AdjustSelection(cell);
+    else
+      this.ClearSelection();
+  },
+
+  selectMouseOver: function(e) {
+    var cell=Rico.eventElement(e);
+    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
+    if (!cell) return;
+    this.AdjustSelection(cell);
+    Rico.eventStop(e);
+  },
+
+  isSelected: function(cell) {
+    if (this.options.highlightMethod!='outline') return Rico.hasClass(cell,this.options.highlightClass);
+    if (!this.SelectIdxStart || !this.SelectIdxEnd) return false;
+    var r1=Math.max(Math.min(this.SelectIdxEnd.row,this.SelectIdxStart.row), this.buffer.windowStart);
+    var r2=Math.min(Math.max(this.SelectIdxEnd.row,this.SelectIdxStart.row), this.buffer.windowEnd-1);
+    if (r1 > r2) return false;
+    var c1=Math.min(this.SelectIdxEnd.column,this.SelectIdxStart.column);
+    var c2=Math.max(this.SelectIdxEnd.column,this.SelectIdxStart.column);
+    var curIdx=this.datasetIndex(cell);
+    return (r1<=curIdx.row && curIdx.row<=r2 && c1<=curIdx.column && curIdx.column<=c2);
+  },
+
+  highlightCell: function(cell) {
+    Rico.addClass(cell,this.options.highlightClass);
+  },
+
+  unhighlightCell: function(cell) {
+    if (cell==null) return;
+    Rico.removeClass(cell,this.options.highlightClass);
+  },
+
+  selectRow: function(r) {
+    for (var c=0; c<this.columns.length; c++)
+      this.highlightCell(this.columns[c].cell(r));
+  },
+
+  unselectRow: function(r) {
+    for (var c=0; c<this.columns.length; c++)
+      this.unhighlightCell(this.columns[c].cell(r));
+  },
+
+  rowMouseOver: function(e) {
+    if (!this.highlightEnabled) return;
+    var cell=Rico.eventElement(e);
+    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
+    if (!cell) return;
+    var newIdx=this.winCellIndex(cell);
+    if ((this.options.highlightSection & (newIdx.tabIdx+1))==0) return;
+    this.highlight(newIdx);
+  },
+
+  highlight: function(newIdx) {
+    if (this.options.highlightMethod!='outline') this.cursorSetClass(newIdx);
+    if (this.options.highlightMethod!='class') this.cursorOutline(newIdx);
+    this.highlightIdx=newIdx;
+  },
+
+  cursorSetClass: function(newIdx) {
+    switch (this.options.highlightElem) {
+      case 'menuCell':
+      case 'cursorCell':
+        if (this.highlightIdx) this.unhighlightCell(this.highlightIdx.cell);
+        this.highlightCell(newIdx.cell);
+        break;
+      case 'menuRow':
+      case 'cursorRow':
+        if (this.highlightIdx) this.unselectRow(this.highlightIdx.row);
+        var s1=this.options.highlightSection & 1;
+        var s2=this.options.highlightSection & 2;
+        var c0=s1 ? 0 : this.options.frozenColumns;
+        var c1=s2 ? this.columns.length : this.options.frozenColumns;
+        for (var c=c0; c<c1; c++)
+          this.highlightCell(this.columns[c].cell(newIdx.row));
+        break;
+      default: return;
+    }
+  },
+
+  cursorOutline: function(newIdx) {
+    var div;
+    switch (this.options.highlightElem) {
+      case 'menuCell':
+      case 'cursorCell':
+        div=this.highlightDiv[newIdx.tabIdx];
+        div.style.left=(this.columns[newIdx.column].dataCell.offsetLeft-1)+'px';
+        div.style.width=this.columns[newIdx.column].colWidth;
+        this.highlightDiv[1-newIdx.tabIdx].style.display='none';
+        break;
+      case 'menuRow':
+      case 'cursorRow':
+        div=this.highlightDiv[0];
+        var s1=this.options.highlightSection & 1;
+        var s2=this.options.highlightSection & 2;
+        div.style.left=s1 ? '0px' : this.frozenTabs.style.width;
+        div.style.width=((s1 ? this.frozenTabs.offsetWidth : 0) + (s2 ? this.innerDiv.offsetWidth : 0) - 4)+'px';
+        break;
+      default: return;
+    }
+    div.style.top=(this.hdrHt+newIdx.row*this.rowHeight-1)+'px';
+    div.style.height=(this.rowHeight-1)+'px';
+    div.style.display='';
+  },
+
+  unhighlight: function() {
+    switch (this.options.highlightElem) {
+      case 'menuCell':
+        //this.highlightIdx=this.menuIdx;
+        /*jsl:fallthru*/
+      case 'cursorCell':
+        if (this.highlightIdx) this.unhighlightCell(this.highlightIdx.cell);
+        if (!this.highlightDiv) return;
+        for (var i=0; i<2; i++)
+          this.highlightDiv[i].style.display='none';
+        break;
+      case 'menuRow':
+        //this.highlightIdx=this.menuIdx;
+        /*jsl:fallthru*/
+      case 'cursorRow':
+        if (this.highlightIdx) this.unselectRow(this.highlightIdx.row);
+        if (this.highlightDiv) this.highlightDiv[0].style.display='none';
+        break;
+    }
+  },
+
+  resetContents: function() {
+    Rico.log("resetContents");
+    this.ClearSelection();
+    this.buffer.clear();
+    this.clearRows();
+    this.clearBookmark();
+  },
+
+  setImages: function() {
+    for (var n=0; n<this.columns.length; n++)
+      this.columns[n].setImage();
+  },
+
+/**
+ * @return column index, or -1 if there are no sorted columns
+ */
+  findSortedColumn: function() {
+    for (var n=0; n<this.columns.length; n++) {
+      if (this.columns[n].isSorted()) return n;
+    }
+    return -1;
+  },
+
+  findColumnName: function(name) {
+    for (var n=0; n<this.columns.length; n++) {
+      if (this.columns[n].fieldName == name) return n;
+    }
+    return -1;
+  },
+  
+/**
+ * Searches options.columnSpecs colAttr for matching colValue
+ * @return array of matching column indexes
+ */
+  findColumnsBySpec: function(colAttr, colValue) {
+    var result=[]
+    for (var n=0; n<this.options.columnSpecs.length; n++) {
+      if (this.options.columnSpecs[n][colAttr] == colValue) result.push(n);
+    }
+    return result;
+  },
+
+/**
+ * Set initial sort
+ */
+  setSortUI: function( columnNameOrNum, sortDirection ) {
+    Rico.log("setSortUI: "+columnNameOrNum+' '+sortDirection);
+    var colnum=this.findSortedColumn();
+    if (colnum >= 0) {
+      sortDirection=this.columns[colnum].getSortDirection();
+    } else {
+      if (typeof sortDirection!='string') {
+        sortDirection=Rico.ColumnConst.SORT_ASC;
+      } else {
+        sortDirection=sortDirection.toUpperCase();
+        if (sortDirection != Rico.ColumnConst.SORT_DESC) sortDirection=Rico.ColumnConst.SORT_ASC;
+      }
+      switch (typeof columnNameOrNum) {
+        case 'string':
+          colnum=this.findColumnName(columnNameOrNum);
+          break;
+        case 'number':
+          colnum=columnNameOrNum;
+          break;
+      }
+    }
+    if (typeof(colnum)!='number' || colnum < 0) return;
+    this.clearSort();
+    this.columns[colnum].setSorted(sortDirection);
+    this.buffer.sortBuffer(colnum);
+  },
+
+/**
+ * clear sort flag on all columns
+ */
+  clearSort: function() {
+    for (var x=0;x<this.columns.length;x++)
+      this.columns[x].setUnsorted();
+  },
+
+/**
+ * clear filters on all columns
+ */
+  clearFilters: function() {
+    for (var x=0;x<this.columns.length;x++) {
+      this.columns[x].setUnfiltered(true);
+    }
+    if (this.options.filterHandler) {
+      this.options.filterHandler();
+    }
+  },
+
+/**
+ * returns number of columns with a user filter set
+ */
+  filterCount: function() {
+    for (var x=0,cnt=0;x<this.columns.length;x++) {
+      if (this.columns[x].isFiltered()) cnt++;
+    }
+    return cnt;
+  },
+
+  sortHandler: function() {
+    this.cancelMenu();
+    this.ClearSelection();
+    this.setImages();
+    var n=this.findSortedColumn();
+    if (n < 0) return;
+    Rico.log("sortHandler: sorting column "+n);
+    this.buffer.sortBuffer(n);
+    this.clearRows();
+    this.scrollDiv.scrollTop = 0;
+    this.buffer.fetch(0);
+  },
+
+  filterHandler: function() {
+    Rico.log("filterHandler");
+    this.cancelMenu();
+    if (this.buffer.processingRequest) {
+      this.queueFilter=true;
+      return;
+    }
+    this.unplugScroll();
+    this.ClearSelection();
+    this.setImages();
+    this.clearBookmark();
+    this.clearRows();
+    this.buffer.fetch(-1);
+    Rico.runLater(10,this,'pluginScroll'); // resetting ht div can cause a scroll event, triggering an extra fetch
+  },
+
+  clearBookmark: function() {
+    if (this.bookmark) this.bookmark.innerHTML="&nbsp;";
+  },
+
+  bookmarkHandler: function(firstrow,lastrow) {
+    var newhtml;
+    if (isNaN(firstrow) || !this.bookmark) return;
+    var totrows=this.buffer.totalRows;
+    if (totrows < lastrow) lastrow=totrows;
+    if (totrows<=0) {
+      newhtml = Rico.getPhraseById('bookmarkNoMatch');
+    } else if (lastrow<0) {
+      newhtml = Rico.getPhraseById('bookmarkNoRec');
+    } else if (this.buffer.foundRowCount) {
+      newhtml = Rico.getPhraseById('bookmarkExact',firstrow,lastrow,totrows);
+    } else {
+      newhtml = Rico.getPhraseById('bookmarkAbout',firstrow,lastrow,totrows);
+    }
+    this.bookmark.innerHTML = newhtml;
+  },
+
+  clearRows: function() {
+    if (this.isBlank==true) return;
+    for (var c=0; c < this.columns.length; c++)
+      this.columns[c].clearColumn();
+    this.isBlank = true;
+  },
+
+  refreshContents: function(startPos) {
+    Rico.log("refreshContents1 "+this.tableId+": startPos="+startPos+" lastRow="+this.lastRowPos+" PartBlank="+this.isPartialBlank+" pageSize="+this.pageSize);
+    this.hideMsg();
+    this.cancelMenu();
+    this.unhighlight(); // in case highlighting was manually invoked
+    if (this.queueFilter) {
+      Rico.log("refreshContents: cancelling refresh because filter has changed");
+      this.queueFilter=false;
+      this.filterHandler();
+      return;
+    }
+    this.highlightEnabled=this.options.highlightSection!='none';
+    var viewPrecedesBuffer = this.buffer.startPos > startPos;
+    var contentStartPos = viewPrecedesBuffer ? this.buffer.startPos: startPos;
+    this.contentStartPos = contentStartPos+1;
+    var contentEndPos = Math.min(this.buffer.startPos + this.buffer.size, startPos + this.pageSize);
+    this.buffer.setWindow(contentStartPos, contentEndPos);
+    Rico.log('refreshContents2 '+this.tableId+': cStartPos='+contentStartPos+' cEndPos='+contentEndPos+' vPrecedesBuf='+viewPrecedesBuffer+' b.startPos='+this.buffer.startPos);
+    if (startPos == this.lastRowPos && !this.isPartialBlank && !this.isBlank) return;
+    this.isBlank = false;
+    var onRefreshComplete = this.options.onRefreshComplete;
+
+    if ((startPos + this.pageSize < this.buffer.startPos) ||
+        (this.buffer.startPos + this.buffer.size < startPos) ||
+        (this.buffer.size == 0)) {
+      this.clearRows();
+      if (onRefreshComplete) onRefreshComplete(this.contentStartPos,contentEndPos);  // update bookmark
+      return;
+    }
+
+    Rico.log('refreshContents: contentStartPos='+contentStartPos+' contentEndPos='+contentEndPos+' viewPrecedesBuffer='+viewPrecedesBuffer);
+    var rowSize = contentEndPos - contentStartPos;
+    var blankSize = this.pageSize - rowSize;
+    var blankOffset = viewPrecedesBuffer ? 0: rowSize;
+    var contentOffset = viewPrecedesBuffer ? blankSize: 0;
+
+    for (var r=0; r < rowSize; r++) { //initialize what we have
+      for (var c=0; c < this.columns.length; c++)
+        this.columns[c].displayValue(r + contentOffset);
+    }
+    for (var i=0; i < blankSize; i++)     // blank out the rest
+      this.blankRow(i + blankOffset);
+    if (this.options.highlightElem=='selection') this.ShowSelection();
+    this.isPartialBlank = blankSize > 0;
+    this.lastRowPos = startPos;
+    Rico.log("refreshContents complete, startPos="+startPos);
+    if (onRefreshComplete) onRefreshComplete(this.contentStartPos,contentEndPos);  // update bookmark
+  },
+
+  scrollToRow: function(rowOffset) {
+     var p=this.rowToPixel(rowOffset);
+     Rico.log("scrollToRow, rowOffset="+rowOffset+" pixel="+p);
+     this.scrollDiv.scrollTop = p; // this causes a scroll event
+     if ( this.options.onscroll )
+        this.options.onscroll( this, rowOffset );
+  },
+
+  scrollUp: function() {
+     this.moveRelative(-1);
+  },
+
+  scrollDown: function() {
+     this.moveRelative(1);
+  },
+
+  pageUp: function() {
+     this.moveRelative(-this.pageSize);
+  },
+
+  pageDown: function() {
+     this.moveRelative(this.pageSize);
+  },
+
+  adjustRow: function(rowOffset) {
+     var notdisp=this.topOfLastPage();
+     if (notdisp == 0 || !rowOffset) return 0;
+     return Math.min(notdisp,rowOffset);
+  },
+
+  rowToPixel: function(rowOffset) {
+     return this.adjustRow(rowOffset) * this.rowHeight;
+  },
+
+/**
+ * @returns row to display at top of scroll div
+ */
+  pixeltorow: function(p) {
+     var notdisp=this.topOfLastPage();
+     if (notdisp == 0) return 0;
+     var prow=parseInt(p/this.rowHeight,10);
+     return Math.min(notdisp,prow);
+  },
+
+  moveRelative: function(relOffset) {
+     var newoffset=Math.max(this.scrollDiv.scrollTop+relOffset*this.rowHeight,0);
+     newoffset=Math.min(newoffset,this.scrollDiv.scrollHeight);
+     //Rico.log("moveRelative, newoffset="+newoffset);
+     this.scrollDiv.scrollTop=newoffset;
+  },
+
+  pluginScroll: function() {
+     if (this.scrollPluggedIn) return;
+     Rico.log("pluginScroll: wheelEvent="+this.wheelEvent);
+     Rico.eventBind(this.scrollDiv,"scroll",this.scrollEventFunc, false);
+     for (var t=0; t<2; t++)
+       Rico.eventBind(this.tabs[t],this.wheelEvent,this.wheelEventFunc, false);
+     this.scrollPluggedIn=true;
+  },
+
+  unplugScroll: function() {
+     if (!this.scrollPluggedIn) return;
+     Rico.log("unplugScroll");
+     Rico.eventUnbind(this.scrollDiv,"scroll", this.scrollEventFunc , false);
+     for (var t=0; t<2; t++)
+       Rico.eventUnbind(this.tabs[t],this.wheelEvent,this.wheelEventFunc, false);
+     this.scrollPluggedIn=false;
+  },
+
+  handleWheel: function(e) {
+    var delta = 0;
+    if (e.wheelDelta) {
+      if (Rico.isOpera)
+        delta = e.wheelDelta/120;
+      else if (Rico.isWebKit)
+        delta = -e.wheelDelta/12;
+      else
+        delta = -e.wheelDelta/120;
+    } else if (e.detail) {
+      delta = e.detail/3; /* Mozilla/Gecko */
+    }
+    if (delta) this.moveRelative(delta);
+    Rico.eventStop(e);
+    return false;
+  },
+
+  handleScroll: function(e) {
+     if ( this.scrollTimeout )
+       clearTimeout( this.scrollTimeout );
+     this.setHorizontalScroll();
+     var scrtop=this.scrollDiv.scrollTop;
+     var vscrollDiff = this.lastScrollPos-scrtop;
+     if (vscrollDiff == 0.00) return;
+     var newrow=this.pixeltorow(scrtop);
+     if (newrow == this.lastRowPos && !this.isPartialBlank && !this.isBlank) return;
+     var stamp1 = new Date();
+     Rico.log("handleScroll, newrow="+newrow+" scrtop="+scrtop);
+     if (this.options.highlightElem=='selection') this.HideSelection();
+     this.buffer.fetch(newrow);
+     if (this.options.onscroll) this.options.onscroll(this, newrow);
+     this.scrollTimeout = Rico.runLater(1200,this,'scrollIdle');
+     this.lastScrollPos = this.scrollDiv.scrollTop;
+     var stamp2 = new Date();
+     //Rico.log("handleScroll, time="+(stamp2.getTime()-stamp1.getTime()));
+  },
+
+  scrollIdle: function() {
+     if ( this.options.onscrollidle )
+        this.options.onscrollidle();
+  }
+
+};
+
+
+Rico.LiveGridColumn = function(grid,colIdx,hdrInfo,tabIdx) {
+  this.initialize(grid,colIdx,hdrInfo,tabIdx);
+};
+
+Rico.LiveGridColumn.prototype = 
+/** @lends Rico.LiveGridColumn# */
+{
+/**
+ * Implements a LiveGrid column. Also contains static properties used by SimpleGrid columns.
+ * @extends Rico.TableColumnBase
+ * @constructs
+ */
+initialize: function(liveGrid,colIdx,hdrInfo,tabIdx) {
+  Rico.extend(this, new Rico.TableColumnBase());
+  this.baseInit(liveGrid,colIdx,hdrInfo,tabIdx);
+  this.buffer=liveGrid.buffer;
+  if (typeof(this.format.type)!='string' || this.format.EntryType=='tinyMCE') this.format.type='raw';
+  if (typeof this.isNullable!='boolean') this.isNullable = /number|date/.test(this.format.type);
+  this.isText = /raw|text|showTags/.test(this.format.type);
+  Rico.log(" sortable="+this.sortable+" filterable="+this.filterable+" hideable="+this.hideable+" isNullable="+this.isNullable+' isText='+this.isText);
+  this.fixHeaders(this.liveGrid.tableId, this.options.hdrIconsFirst);
+  if (this['format_'+this.format.type]) {
+    this._format=this['format_'+this.format.type];
+  }
+  if (this.format.control) {
+    // copy all properties/methods that start with '_'
+    if (typeof this.format.control=='string') {
+      this.format.control=eval(this.format.control);
+    }
+    for (var property in this.format.control) {
+      if (property.charAt(0)=='_') {
+        Rico.log("Copying control property "+property+ ' to ' + this);
+        this[property] = this.format.control[property];
+      }
+    }
+  }
+},
+
+/**
+ * Sorts the column in ascending order
+ */
+sortAsc: function() {
+  this.setColumnSort(Rico.ColumnConst.SORT_ASC);
+},
+
+/**
+ * Sorts the column in descending order
+ */
+sortDesc: function() {
+  this.setColumnSort(Rico.ColumnConst.SORT_DESC);
+},
+
+/**
+ * Sorts the column in the specified direction
+ * @param direction must be one of Rico.ColumnConst.UNSORTED, .SORT_ASC, or .SORT_DESC
+ */
+setColumnSort: function(direction) {
+  this.liveGrid.clearSort();
+  this.setSorted(direction);
+  if (this.liveGrid.options.saveColumnInfo.sort)
+    this.liveGrid.setCookie();
+  if (this.options.sortHandler)
+    this.options.sortHandler();
+},
+
+/**
+ * @returns true if this column is allowed to be sorted
+ */
+isSortable: function() {
+  return this.sortable;
+},
+
+/**
+ * @returns true if this column is currently sorted
+ */
+isSorted: function() {
+  return this.currentSort != Rico.ColumnConst.UNSORTED;
+},
+
+/**
+ * @returns Rico.ColumnConst.UNSORTED, .SORT_ASC, or .SORT_DESC
+ */
+getSortDirection: function() {
+  return this.currentSort;
+},
+
+/**
+ * toggle the sort sequence for this column
+ */
+toggleSort: function() {
+  if (this.buffer && this.buffer.totalRows==0) return;
+  if (this.currentSort == Rico.ColumnConst.SORT_ASC)
+    this.sortDesc();
+  else
+    this.sortAsc();
+},
+
+/**
+ * Flags that this column is not sorted
+ */
+setUnsorted: function() {
+  this.setSorted(Rico.ColumnConst.UNSORTED);
+},
+
+/**
+ * Flags that this column is sorted, but doesn't actually carry out the sort
+ * @param direction must be one of Rico.ColumnConst.UNSORTED, .SORT_ASC, or .SORT_DESC
+ */
+setSorted: function(direction) {
+  this.currentSort = direction;
+},
+
+/**
+ * @returns true if this column is allowed to be filtered
+ */
+canFilter: function() {
+  return this.filterable;
+},
+
+/**
+ * @returns a textual representation of how this column is filtered
+ */
+getFilterText: function() {
+  var vals=[];
+  for (var i=0; i<this.filterValues.length; i++) {
+    var v=this.filterValues[i];
+    vals.push(v=='' ? Rico.getPhraseById('filterBlank') : v);
+  }
+  switch (this.filterOp) {
+    case 'EQ':   return '= '+vals.join(', ');
+    case 'NE':   return Rico.getPhraseById('filterNot',vals.join(', '));
+    case 'LE':   return '<= '+vals[0];
+    case 'GE':   return '>= '+vals[0];
+    case 'LIKE': return Rico.getPhraseById('filterLike',vals[0]);
+    case 'NULL': return Rico.getPhraseById('filterEmpty');
+    case 'NOTNULL': return Rico.getPhraseById('filterNotEmpty');
+  }
+  return '?';
+},
+
+/**
+ * @returns returns the query string representation of the filter
+ */
+getFilterQueryParm: function() {
+  if (this.filterType == Rico.ColumnConst.UNFILTERED) return '';
+  var retval='&f['+this.index+'][op]='+this.filterOp;
+  retval+='&f['+this.index+'][len]='+this.filterValues.length;
+  for (var i=0; i<this.filterValues.length; i++) {
+    retval+='&f['+this.index+']['+i+']='+escape(this.filterValues[i]);
+  }
+  return retval;
+},
+
+/**
+ * removes the filter from this column
+ */
+setUnfiltered: function(skipHandler) {
+  this.filterType = Rico.ColumnConst.UNFILTERED;
+  if (this.liveGrid.options.saveColumnInfo.filter)
+    this.liveGrid.setCookie();
+  if (this.removeFilterFunc)
+    this.removeFilterFunc();
+  if (this.options.filterHandler && !skipHandler)
+    this.options.filterHandler();
+},
+
+setFilterEQ: function() {
+  this.setUserFilter('EQ');
+},
+setFilterNE: function() {
+  this.setUserFilter('NE');
+},
+addFilterNE: function() {
+  this.filterValues.push(this.userFilter);
+  if (this.liveGrid.options.saveColumnInfo.filter)
+    this.liveGrid.setCookie();
+  if (this.options.filterHandler)
+    this.options.filterHandler();
+},
+setFilterGE: function() { this.setUserFilter('GE'); },
+setFilterLE: function() { this.setUserFilter('LE'); },
+setFilterKW: function(keyword) {
+  if (keyword!='' && keyword!=null) {
+    this.setFilter('LIKE',keyword,Rico.ColumnConst.USERFILTER);
+  } else {
+    this.setUnfiltered(false);
+  }
+},
+
+setUserFilter: function(relop) {
+  this.setFilter(relop,this.userFilter,Rico.ColumnConst.USERFILTER);
+},
+
+setSystemFilter: function(relop,filter) {
+  this.setFilter(relop,filter,Rico.ColumnConst.SYSTEMFILTER);
+},
+
+setFilter: function(relop,filter,type,removeFilterFunc) {
+  this.filterValues = typeof(filter)=='object' ? filter : [filter];
+  this.filterType = type;
+  this.filterOp = relop;
+  if (type == Rico.ColumnConst.USERFILTER && this.liveGrid.options.saveColumnInfo.filter)
+    this.liveGrid.setCookie();
+  this.removeFilterFunc=removeFilterFunc;
+  if (this.options.filterHandler)
+    this.options.filterHandler();
+},
+
+isFiltered: function() {
+  return this.filterType == Rico.ColumnConst.USERFILTER;
+},
+
+filterChange: function(e) {\r
+  var selbox=Rico.eventElement(e);
+  if (selbox.value==this.liveGrid.options.FilterAllToken)\r
+    this.setUnfiltered();\r
+  else
+    this.setFilter('EQ',selbox.value,Rico.ColumnConst.USERFILTER,function() {selbox.selectedIndex=0;});\r
+},
+
+filterClear: function(e) {\r
+  this.filterField.value='';
+  this.setUnfiltered();\r
+},
+
+filterKeypress: function(e) {\r
+  var txtbox=Rico.eventElement(e);
+  if (typeof this.lastKeyFilter != 'string') this.lastKeyFilter='';\r
+  if (this.lastKeyFilter==txtbox.value) return;\r
+  var v=txtbox.value;\r
+  Rico.log("filterKeypress: "+this.index+' '+v);\r
+  this.lastKeyFilter=v;
+  if (v=='' || v=='*')\r
+    this.setUnfiltered();\r
+  else {
+    this.setFilter('LIKE', v, Rico.ColumnConst.USERFILTER, function() {txtbox.value='';});
+  }\r
+},\r
+
+mFilterSelectClick: function(e) {
+  Rico.eventStop(e);
+  if (this.mFilter.style.display!='none') {
+    this.mFilterFinish(e);
+    if (Rico.isIE && Rico.ieVersion <= 6) {
+      this.filterField.focus();
+    } else {
+      this.filterField.blur();
+    }
+  } else {
+    var offset=Rico.cumulativeOffset(this.filterField);
+    this.mFilter.style.top=(offset.top+this.filterField.offsetHeight)+'px';
+    this.mFilter.style.left=offset.left+'px';
+    this.mFilter.style.width=Math.min(this.filterField.offsetWidth,parseInt(this.colWidth,10))+'px';
+    Rico.show(this.mFilter);
+    this.mFilterFocus.focus();
+  }
+},
+
+mFilterFinish: function(e) {
+  if (!this.mFilterChange) {
+    Rico.hide(this.mFilter);
+    return;
+  }
+  if (this.mFilterInputs[0].checked) {
+    this.mFilterReset();
+    Rico.hide(this.mFilter);
+    this.setUnfiltered();
+    return;
+  }
+  var newValues=[];
+  var newLabels=[];
+  for (var i=1; i<this.mFilterInputs.length; i++) {
+    if (this.mFilterInputs[i].checked) {
+      newValues.push(this.mFilterInputs[i].value)
+      newLabels.push(this.mFilterLabels[i].innerHTML)
+    }
+  }
+  if (newValues.length > 0) {
+    var newText=newLabels.join(', ');
+    this.filterField.options[0].text=newText;
+    this.filterField.title=newText;
+    Rico.hide(this.mFilter);
+    this.mFilterChange=false;
+    var self=this;
+    this.setFilter('EQ',newValues,Rico.ColumnConst.USERFILTER,function() { self.mFilterReset(); });
+  } else {
+    alert('Please select at least one value');
+  }
+},
+
+mFilterReset: function() {
+  var newText=this.mFilterLabels[0].innerHTML;  // all
+  this.filterField.options[0].text=newText;
+  this.filterField.title=newText;
+},
+
+mFilterAllClick: function(e) {
+  var allChecked=this.mFilterInputs[0].checked;
+  for (var i=1; i<this.mFilterInputs.length; i++) {
+    this.mFilterInputs[i].checked=allChecked;
+  }
+  this.mFilterChange=true;
+},
+
+mFilterOtherClick: function(e) {
+  this.mFilterInputs[0].checked=false;
+  this.mFilterChange=true;
+},
+
+format_text: function(v) {
+  if (typeof v!='string')
+    return '&nbsp;';
+  else
+    return Rico.stripTags(v);
+},
+
+format_showTags: function(v) {
+  if (typeof v!='string')
+    return '&nbsp;';
+  else
+    return v.replace(/&/g, '&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
+},
+
+format_number: function(v) {
+  if (typeof v=='undefined' || v=='' || v==null)
+    return '&nbsp;';
+  else
+    return Rico.formatNumber(v,this.format);
+},
+
+format_datetime: function(v) {
+  if (typeof v=='undefined' || v=='' || v==null)
+    return '&nbsp;';
+  else {
+    var d=Rico.setISO8601(v);
+    if (!d) return v;
+    return (this.format.prefix || '')+Rico.formatDate(d,this.format.dateFmt || 'translateDateTime')+(this.format.suffix || '');
+  }
+},
+
+// converts GMT/UTC to local time
+format_UTCasLocalTime: function(v) {
+  if (typeof v=='undefined' || v=='' || v==null)
+    return '&nbsp;';
+  else {
+    var tz=new Date();
+    var d=Rico.setISO8601(v,-tz.getTimezoneOffset());
+    if (!d) return v;
+    return (this.format.prefix || '')+Rico.formatDate(d,this.format.dateFmt || 'translateDateTime')+(this.format.suffix || '');
+  }
+},
+
+format_date: function(v) {
+  if (typeof v=='undefined' || v==null || v=='')
+    return '&nbsp;';
+  else {
+    var d=Rico.setISO8601(v);
+    if (!d) return v;
+    return (this.format.prefix || '')+Rico.formatDate(d,this.format.dateFmt || 'translateDate')+(this.format.suffix || '');
+  }
+},
+
+fixHeaders: function(prefix, iconsfirst) {
+  if (this.sortable) {
+    var handler=Rico.eventHandle(this,'toggleSort');
+    switch (this.options.headingSort) {
+      case 'link':
+        var a=Rico.wrapChildren(this.hdrCellDiv,'ricoSort',undefined,'a');
+        a.href = "javascript:void(0)";
+        Rico.eventBind(a,"click", handler);
+        break;
+      case 'hover':
+        Rico.eventBind(this.hdrCellDiv,"click", handler);
+        break;
+    }
+  }
+  this.imgFilter = document.createElement('span');
+  this.imgFilter.style.display='none';
+  this.imgFilter.className='rico-icon ricoLG_filterCol';
+  this.imgSort = document.createElement('span');
+  this.imgSort.style.display='none';
+  this.imgSort.style.verticalAlign='top';
+  if (iconsfirst) {
+    this.hdrCellDiv.insertBefore(this.imgSort,this.hdrCellDiv.firstChild);
+    this.hdrCellDiv.insertBefore(this.imgFilter,this.hdrCellDiv.firstChild);
+  } else {
+    this.hdrCellDiv.appendChild(this.imgFilter);
+    this.hdrCellDiv.appendChild(this.imgSort);
+  }
+  if (!this.format.filterUI) {
+    Rico.eventBind(this.imgFilter, 'click', Rico.eventHandle(this,'filterClick'), false);
+  }
+},
+
+filterClick: function(e) {
+  if (this.filterType==Rico.ColumnConst.USERFILTER && this.filterOp=='LIKE') {
+    this.liveGrid.openKeyword(this.index);
+  }
+},
+
+getValue: function(windowRow) {
+  return this.buffer.getWindowCell(windowRow,this.index);
+},
+
+getBufferAttr: function(windowRow) {
+  return this.buffer.getWindowAttr(windowRow,this.index);
+},
+
+setValue: function(windowRow,newval) {
+  this.buffer.setWindowValue(windowRow,this.index,newval);
+},
+
+_format: function(v) {
+  return v;
+},
+
+_display: function(v,gridCell) {
+  gridCell.innerHTML=this._format(v);
+},
+
+_export: function(v) {
+  return this._format(v);
+},
+
+exportBuffer: function(bufRow) {
+  return this._export(this.buffer.getValue(bufRow,this.index));
+},
+
+displayValue: function(windowRow) {
+  var bufval=this.getValue(windowRow);
+  if (bufval==null) {
+    this.clearCell(windowRow);
+    return;
+  }
+  var gridCell=this.cell(windowRow);
+  this._display(bufval,gridCell,windowRow);
+  var acceptAttr=this.buffer.options.acceptAttr;
+  if (acceptAttr.length==0) return;
+  var bufAttr=this.getBufferAttr(windowRow);
+  if (bufAttr==null) return;
+  for (var k=0; k<acceptAttr.length; k++) {
+    bufAttr=bufAttr['_'+acceptAttr[k]] || '';
+    switch (acceptAttr[k]) {
+      case 'style': gridCell.style.cssText=bufAttr; break;
+      case 'class': gridCell.className=bufAttr; break;
+      default:      gridCell['_'+acceptAttr[k]]=bufAttr; break;
+    }
+  }
+}
+
+};
diff --git a/minsrc/ricoLiveGridAjax.js b/minsrc/ricoLiveGridAjax.js
new file mode 100644 (file)
index 0000000..cab820f
--- /dev/null
@@ -0,0 +1,577 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+if(typeof Rico=='undefined') throw("LiveGridAjax requires the Rico JavaScript framework");
+
+if (!Rico.Buffer) Rico.Buffer = {};
+
+Rico.Buffer.AjaxXML = function(url,options,ajaxOptions) {
+  this.initialize(url,options,ajaxOptions);
+}
+
+Rico.Buffer.AjaxXML.prototype = {
+/**
+ * @class Implements buffer for LiveGrid. Loads data from server via a single AJAX call.
+ * @extends Rico.Buffer.Base
+ * @constructs
+ */
+  initialize: function(url,options,ajaxOptions) {
+    Rico.extend(this, new Rico.Buffer.Base());
+    Rico.extend(this, Rico.Buffer.AjaxXMLMethods);
+    this.dataSource=url;
+    this.options.bufferTimeout=20000;            // time to wait for ajax response (milliseconds)
+    this.options.requestParameters=[];
+    this.options.waitMsg=Rico.getPhraseById("waitForData");  // replace this with an image tag if you prefer
+    this.options.canFilter=true;
+    this.options.fmt='xml';
+    Rico.extend(this.options, options || {});
+    this.ajaxOptions = { parameters: null, method : 'get' };
+    Rico.extend(this.ajaxOptions, ajaxOptions || {});
+    this.requestCount=0;
+    this.processingRequest=false;
+    this.pendingRequest=-2;
+    this.fetchData=true;
+    this.sortParm={};
+  }
+}
+
+Rico.Buffer.AjaxXMLMethods = {
+
+/** @lends Rico.Buffer.AjaxXML# */
+  fetch: function(offset) {
+    if (this.fetchData) {
+      this.foundRowCount=true;
+      this.fetchData=false;
+      this.processingRequest=true;
+      this.liveGrid.showMsg(this.options.waitMsg);
+      this.timeoutHandler = Rico.runLater(this.options.bufferTimeout,this,'handleTimedOut');
+      this.ajaxOptions.parameters = this.formQueryHashXML(0,-1);
+      Rico.log('sending request');
+      var self=this;
+      if (typeof this.dataSource=='string') {
+        this.ajaxOptions.onComplete = function(xhr) { self.ajaxUpdate(offset,xhr); };
+        new Rico.ajaxRequest(this.dataSource, this.ajaxOptions);
+      } else {
+        this.ajaxOptions.onComplete = function(newRows, newAttr, totalRows, errMsg) { self.jsUpdate(offset, newRows, newAttr, totalRows, errMsg); };
+        this.dataSource(this.ajaxOptions);
+      }
+    } else {
+      if (offset < 0) {
+        this.applyFilters();
+        this.setTotalRows(this.size);
+        offset=0;
+      }
+      this.liveGrid.refreshContents(offset);
+    }
+  },
+
+/**
+ * Server did not respond in time... assume that there could have been
+ * an error, and allow requests to be processed again.
+ */
+  handleTimedOut: function() {
+    Rico.log("Request Timed Out");
+    this.liveGrid.showMsg(Rico.getPhraseById("requestTimedOut"));
+  },
+
+  formQueryHashXML: function(startPos,fetchSize) {
+    var queryHash= {
+      id: this.liveGrid.tableId,
+      page_size: (typeof fetchSize=='number') ? fetchSize : this.totalRows,
+      offset: startPos.toString()
+    };
+    if (!this.foundRowCount) queryHash['get_total']='true';
+    if (this.options.requestParameters) {
+      for ( var i=0; i < this.options.requestParameters.length; i++ ) {
+        var anArg = this.options.requestParameters[i];
+        if ( anArg.name != undefined && anArg.value != undefined ) {
+          queryHash[anArg.name]=anArg.value;
+        } else {
+          var ePos  = anArg.indexOf('=');
+          var argName  = anArg.substring( 0, ePos );
+          var argValue = anArg.substring( ePos + 1 );
+          queryHash[argName]=argValue;
+        }
+      }
+    }
+    return queryHash;
+  },
+
+  clearTimer: function() {
+    if(typeof this.timeoutHandler != "number") return;
+    window.clearTimeout(this.timeoutHandler);
+    delete this.timeoutHandler;
+  },
+
+  // used by both XML and SQL buffers
+  jsUpdate: function(startPos, newRows, newAttr, totalRows, errMsg) {
+    this.clearTimer();
+    this.processingRequest=false;
+    Rico.log("jsUpdate: "+arguments.length);
+    if (errMsg) {
+      Rico.log("jsUpdate: received error="+errMsg);
+      this.liveGrid.showMsg(Rico.getPhraseById("requestError",errMsg));
+      return;
+    }
+    this.rcvdRows = newRows.length;
+    if (typeof totalRows=='number') {
+      this.rowcntContent = totalRows.toString();
+      this.rcvdRowCount = true;
+      this.foundRowCount = true;
+      Rico.log("jsUpdate: found RowCount="+this.rowcntContent);
+    }
+    this.updateBuffer(startPos, newRows, newAttr);
+    if (this.options.onAjaxUpdate)
+      this.options.onAjaxUpdate();
+    this.updateGrid(startPos);
+    if (this.options.TimeOut && this.timerMsg)
+      this.restartSessionTimer();
+    if (this.pendingRequest>=-1) {
+      var offset=this.pendingRequest;
+      Rico.log("jsUpdate: found pending request for offset="+offset);
+      this.pendingRequest=-2;
+      this.fetch(offset);
+    }
+  },
+
+  // used by both XML and SQL buffers
+  ajaxUpdate: function(startPos,xhr) {
+    this.clearTimer();
+    this.processingRequest=false;
+    if (xhr.status != 200) {
+      Rico.log("ajaxUpdate: received http error="+xhr.status);
+      this.liveGrid.showMsg(Rico.getPhraseById("httpError",xhr.status));
+      return;
+    }
+    this._responseHandler=this['processResponse'+this.options.fmt.toUpperCase()];
+    if (!this._responseHandler(startPos,xhr)) return;
+    if (this.options.onAjaxUpdate)
+      this.options.onAjaxUpdate();
+    this.updateGrid(startPos);
+    if (this.options.TimeOut && this.timerMsg)
+      this.restartSessionTimer();
+    if (this.pendingRequest>=-1) {
+      var offset=this.pendingRequest;
+      Rico.log("ajaxUpdate: found pending request for offset="+offset);
+      this.pendingRequest=-2;
+      this.fetch(offset);
+    }
+  },
+  
+  // used by both XML and SQL buffers
+  processResponseXML: function(startPos,request) {
+    // The response text may contain META DATA for debugging if client side debugging is enabled in VS\r
+    var xmlDoc = request.responseXML;\r
+    if (request.responseText.substring(0, 4) == "<!--") {\r
+      var nEnd = request.responseText.indexOf("-->");\r
+      if (nEnd == -1) {\r
+        this.liveGrid.showMsg('Web server error - client side debugging may be enabled');\r
+        return false;\r
+      }\r
+      xmlDoc = Rico.createXmlDocument();\r
+      xmlDoc.loadXML(request.responseText.substring(nEnd+3));\r
+    }
+    
+    if (!xmlDoc) {
+      alert("Data provider returned an invalid XML response");
+      Rico.log("Data provider returned an invalid XML response");
+      return false;
+    }
+
+    // process children of <ajax-response>
+    var response = xmlDoc.getElementsByTagName("ajax-response");
+    if (response == null || response.length != 1) return false;
+    this.rcvdRows = 0;
+    this.rcvdRowCount = false;
+    var ajaxResponse=response[0];
+    var debugtags = ajaxResponse.getElementsByTagName('debug');
+    for (var i=0; i<debugtags.length; i++)
+      Rico.log("ajaxUpdate: debug msg "+i+": "+Rico.getContentAsString(debugtags[i],this.options.isEncoded));
+    var error = ajaxResponse.getElementsByTagName('error');
+    if (error.length > 0) {
+      var msg=Rico.getContentAsString(error[0],this.options.isEncoded);
+      alert("Data provider returned an error:\n"+msg);
+      Rico.log("Data provider returned an error:\n"+msg);
+      return false;
+    }
+    var rowsElement = ajaxResponse.getElementsByTagName('rows')[0];
+    if (!rowsElement) {
+      Rico.log("ajaxUpdate: invalid response");
+      this.liveGrid.showMsg(Rico.getPhraseById("invalidResponse"));
+      return false;
+    }
+    var rowcnttags = ajaxResponse.getElementsByTagName('rowcount');
+    if (rowcnttags && rowcnttags.length==1) {
+      this.rowcntContent = Rico.getContentAsString(rowcnttags[0],this.options.isEncoded);
+      this.rcvdRowCount = true;
+      this.foundRowCount = true;
+      Rico.log("ajaxUpdate: found RowCount="+this.rowcntContent);
+    }
+
+    // process <rows>
+    this.updateUI = rowsElement.getAttribute("update_ui") == "true";
+    this.rcvdOffset = rowsElement.getAttribute("offset");
+    Rico.log("ajaxUpdate: rcvdOffset="+this.rcvdOffset);
+    var newRows = this.dom2jstable(rowsElement);
+    var newAttr = (this.options.acceptAttr.length > 0) ? this.dom2jstableAttr(rowsElement) : false;
+    this.rcvdRows = newRows.length;
+    this.updateBuffer(startPos, newRows, newAttr);
+    return true;
+  },
+
+  processResponseJSON: function(startPos,request) {
+    var json = Rico.getJSON(request);
+    if (!json || json == null) {
+      alert("Data provider returned an invalid JSON response");
+      Rico.log("Data provider returned an invalid JSON response");
+      return false;
+    }
+
+    if (json.debug) {
+      for (var i=0; i<json.debug.length; i++)
+        Rico.writeDebugMsg("debug msg "+i+": "+json.debug[i]);
+    }
+    if (json.error) {
+      alert("Data provider returned an error:\n"+json.error);
+      Rico.writeDebugMsg("Data provider returned an error:\n"+json.error);
+      return false;
+    }
+
+    if (json.rowcount) {
+      this.rowcntContent = json.rowcount;
+      this.rcvdRowCount = true;
+      this.foundRowCount = true;
+      Rico.writeDebugMsg("loadRows, found RowCount="+json.rowcount);
+    }
+
+    this.rcvdRows = json.rows.length;
+    this.updateBuffer(startPos, json.rows);
+    return true;
+  },
+
+  // specific to XML buffer
+  updateBuffer: function(start, newRows, newAttr) {
+    this.baseRows = newRows;
+    this.attr = newAttr;
+    Rico.log("updateBuffer: # of rows="+this.rcvdRows);
+    this.rcvdRowCount=true;
+    this.rowcntContent=this.rcvdRows;
+    if (typeof this.delayedSortCol=='number')
+      this.sortBuffer(this.delayedSortCol);
+    this.applyFilters();
+    this.startPos = 0;
+  },
+
+  // used by both XML and SQL buffers
+  updateGrid: function(offset) {
+    Rico.log("updateGrid, size="+this.size+' rcv cnt type='+typeof(this.rowcntContent));
+    var newpos;
+    if (this.rcvdRowCount==true) {
+      Rico.log("found row cnt: "+this.rowcntContent);
+      var eofrow=parseInt(this.rowcntContent,10);
+      var lastTotalRows=this.totalRows;
+      if (!isNaN(eofrow) && eofrow!=lastTotalRows) {
+        this.setTotalRows(eofrow);
+        newpos=Math.min(this.liveGrid.topOfLastPage(),offset);
+        Rico.log("updateGrid: new rowcnt="+eofrow+" newpos="+newpos);
+        if (lastTotalRows==0 && this.liveGrid.sizeTo=='data')
+          Rico.runLater(100,this.liveGrid,'adjustPageSize');  // FF takes a long time to calc initial size
+        this.liveGrid.scrollToRow(newpos);
+        if ( this.isInRange(newpos) ) {
+          this.liveGrid.refreshContents(newpos);
+        } else {
+          this.fetch(newpos);
+        }
+        return;
+      }
+    } else {
+      var lastbufrow=offset+this.rcvdRows;
+      if (lastbufrow>this.totalRows) {
+        var newcnt=lastbufrow;
+        Rico.log("extending totrows to "+newcnt);
+        this.setTotalRows(newcnt);
+      }
+    }
+    newpos=this.liveGrid.pixeltorow(this.liveGrid.scrollDiv.scrollTop);
+    Rico.log("updateGrid: newpos="+newpos);
+    this.liveGrid.refreshContents(newpos);
+  }
+
+};
+
+
+
+Rico.Buffer.AjaxSQL = function(url,options,ajaxOptions) {
+  this.initialize(url,options,ajaxOptions);
+}
+
+Rico.Buffer.AjaxSQL.prototype = {
+/**
+ * @class Implements buffer for LiveGrid. Loads data from server in chunks as user scrolls through the grid.
+ * @extends Rico.Buffer.AjaxXML
+ * @constructs
+ */
+  initialize: function(url,options,ajaxOptions) {
+    Rico.extend(this, new Rico.Buffer.AjaxXML());
+    Rico.extend(this, Rico.Buffer.AjaxSQLMethods);
+    this.dataSource=url;
+    this.options.canFilter=true;
+    this.options.largeBufferSize  = 7.0;   // 7 pages
+    this.options.nearLimitFactor  = 1.0;   // 1 page
+    Rico.extend(this.options, options || {});
+    Rico.extend(this.ajaxOptions, ajaxOptions || {});
+  }
+}
+
+Rico.Buffer.AjaxSQLMethods = {
+/** @lends Rico.Buffer.AjaxSQL# */
+
+  registerGrid: function(liveGrid) {
+    this.liveGrid = liveGrid;
+    this.sessionExpired=false;
+    this.timerMsg=document.getElementById(liveGrid.tableId+'_timer');
+    if (this.options.TimeOut && this.timerMsg) {
+      if (!this.timerMsg.title) this.timerMsg.title=Rico.getPhraseById("sessionExpireMinutes");
+      this.restartSessionTimer();
+    }
+  },
+
+  setBufferSize: function(pageSize) {
+    this.maxFetchSize = Math.max(50,parseInt(this.options.largeBufferSize * pageSize,10));
+    this.nearLimit = parseInt(this.options.nearLimitFactor * pageSize,10);
+    this.maxBufferSize = this.maxFetchSize * 3;
+  },
+
+  restartSessionTimer: function() {
+    if (this.sessionExpired==true) return;
+    this.sessionEndTime = (new Date()).getTime() + this.options.TimeOut*60000;
+    if (this.sessionTimer) clearTimeout(this.sessionTimer);
+    this.updateSessionTimer();
+  },
+
+  updateSessionTimer: function() {
+    var now=(new Date()).getTime();
+    if (now > this.sessionEndTime) {
+      this.displaySessionTimer(Rico.getPhraseById("sessionExpired"));
+      this.timerMsg.style.backgroundColor="red";
+      this.sessionExpired=true;
+    } else {
+      var timeRemaining=Math.ceil((this.sessionEndTime - now) / 60000);
+      this.displaySessionTimer(timeRemaining);
+      this.sessionTimer=Rico.runLater(10000,this,'updateSessionTimer');
+    }
+  },
+
+  displaySessionTimer: function(msg) {
+    this.timerMsg.innerHTML='&nbsp;'+msg+'&nbsp;';
+  },
+
+  /**
+   * Update the grid with fresh data from the database, maintaining scroll position.
+   * @param resetRowCount indicates whether the total row count should be refreshed as well
+   */
+  refresh: function(resetRowCount) {
+    var lastGridPos=this.liveGrid.lastRowPos;\r
+    this.clear();
+    if (resetRowCount) {
+      this.setTotalRows(0);
+      this.foundRowCount = false;
+    }
+    this.liveGrid.clearBookmark();
+    this.liveGrid.clearRows();
+    this.fetch(lastGridPos);
+  },
+
+  /**
+   * Fetch data from database.
+   * @param offset position (row) within the dataset (-1=clear existing buffer before issuing request)
+   */
+  fetch: function(offset) {
+    Rico.log("AjaxSQL fetch: offset="+offset+', lastOffset='+this.lastOffset);
+    if (this.processingRequest) {
+      Rico.log("AjaxSQL fetch: queue request");
+      this.pendingRequest=offset;
+      return;
+    }
+    if (offset < 0) {
+      this.clear();
+      this.setTotalRows(0);
+      this.foundRowCount = false;
+      offset=0;
+    }
+    var lastOffset = this.lastOffset;
+    this.lastOffset = offset;
+    if (this.isInRange(offset)) {
+      Rico.log("AjaxSQL fetch: in buffer");
+      this.liveGrid.refreshContents(offset);
+      if (offset > lastOffset) {
+        if (offset+this.liveGrid.pageSize < this.endPos()-this.nearLimit) return;
+        if (this.endPos()==this.totalRows && this.foundRowCount) return;
+      } else if (offset < lastOffset) {
+        if (offset > this.startPos+this.nearLimit) return;
+        if (this.startPos==0) return;
+      } else return;
+    }
+    if (offset >= this.totalRows && this.foundRowCount) return;
+
+    this.processingRequest=true;
+    Rico.log("AjaxSQL fetch: processing offset="+offset);
+    var bufferStartPos = this.getFetchOffset(offset);
+    var fetchSize = this.getFetchSize(bufferStartPos);
+    var partialLoaded = false;
+
+    this.liveGrid.showMsg(this.options.waitMsg);
+    this.timeoutHandler = Rico.runLater(this.options.bufferTimeout, this, 'handleTimedOut');
+    this.ajaxOptions.parameters = this.formQueryHashSQL(bufferStartPos,fetchSize,this.options.fmt);
+    this.requestCount++;
+    Rico.log('sending req #'+this.requestCount);
+    var self=this;
+    if (typeof this.dataSource=='string') {
+      this.ajaxOptions.onComplete = function(xhr) { self.ajaxUpdate(bufferStartPos, xhr); };
+      new Rico.ajaxRequest(this.dataSource, this.ajaxOptions);
+    } else {
+      this.ajaxOptions.onComplete = function(newRows, newAttr, totalRows, errMsg) { self.jsUpdate(bufferStartPos, newRows, newAttr, totalRows, errMsg); };
+      this.dataSource(this.ajaxOptions);
+    }
+  },
+
+  formQueryHashSQL: function(startPos,fetchSize,fmt) {
+    var queryHash=this.formQueryHashXML(startPos,fetchSize);
+    queryHash[this.liveGrid.actionId]="query";
+    if (fmt) queryHash._fmt=fmt;
+
+    // sort
+    Rico.extend(queryHash,this.sortParm);
+
+    // filters
+    for (var n=0; n<this.liveGrid.columns.length; n++) {
+      var c=this.liveGrid.columns[n];
+      if (c.filterType == Rico.ColumnConst.UNFILTERED) continue;
+      var colnum=typeof(c.format.filterCol)=='number' ? c.format.filterCol : c.index;
+      queryHash['f['+colnum+'][op]']=c.filterOp;
+      queryHash['f['+colnum+'][len]']=c.filterValues.length;
+      for (var i=0; i<c.filterValues.length; i++) {
+        var fval=c.filterValues[i];
+        if (c.filterOp=='LIKE' && fval.indexOf('*')==-1) fval='*'+fval+'*';
+        queryHash['f['+colnum+']['+i+']']=fval;
+      }
+    }
+    return queryHash;
+  },
+
+  getFetchSize: function(adjustedOffset) {
+    var adjustedSize = 0;
+    if (adjustedOffset >= this.startPos) { //appending
+      var endFetchOffset = this.maxFetchSize + adjustedOffset;
+      adjustedSize = endFetchOffset - adjustedOffset;
+      if(adjustedOffset == 0 && adjustedSize < this.maxFetchSize)
+        adjustedSize = this.maxFetchSize;
+      Rico.log("getFetchSize/append, adjustedSize="+adjustedSize+" adjustedOffset="+adjustedOffset+' endFetchOffset='+endFetchOffset);
+    } else { //prepending
+      adjustedSize = Math.min(this.startPos - adjustedOffset,this.maxFetchSize);
+    }
+    return adjustedSize;
+  },
+
+  getFetchOffset: function(offset) {
+    var adjustedOffset = offset;
+    if (offset > this.startPos)
+      adjustedOffset = Math.max(offset, this.endPos());  //appending
+    else if (offset + this.maxFetchSize >= this.startPos)
+      adjustedOffset = Math.max(this.startPos - this.maxFetchSize, 0);  //prepending
+    return adjustedOffset;
+  },
+
+  updateBuffer: function(start, newRows, newAttr) {
+    Rico.log("updateBuffer: start="+start+", # of rows="+this.rcvdRows);
+    if (this.rows.length == 0) { // initial load
+      this.rows = newRows;
+      this.attr = newAttr;
+      this.startPos = start;
+    } else if (start > this.startPos) { //appending
+      if (this.startPos + this.rows.length < start) {
+        this.rows =  newRows;
+        this.attr = newAttr;
+        this.startPos = start;
+      } else {
+        this.rows = this.rows.concat( newRows.slice(0, newRows.length));
+        if (this.attr) this.attr = this.attr.concat( newAttr.slice(0, newAttr.length));
+        if (this.rows.length > this.maxBufferSize) {
+          var fullSize = this.rows.length;
+          this.rows = this.rows.slice(this.rows.length - this.maxBufferSize, this.rows.length);
+          if (this.attr) this.attr = this.attr.slice(this.attr.length - this.maxBufferSize, this.attr.length);
+          this.startPos = this.startPos +  (fullSize - this.rows.length);
+        }
+      }
+    } else { //prepending
+      if (start + newRows.length < this.startPos) {
+        this.rows =  newRows;
+      } else {
+        this.rows = newRows.slice(0, this.startPos).concat(this.rows);
+        if (this.maxBufferSize && this.rows.length > this.maxBufferSize)
+          this.rows = this.rows.slice(0, this.maxBufferSize);
+      }
+      this.startPos =  start;
+    }
+    this.size = this.rows.length;
+  },
+
+  sortBuffer: function(colnum) {
+    this.sortParm={};
+    var col=this.liveGrid.columns[colnum];
+    if (this.options.sortParmFmt) {
+      this.sortParm['sort_col']=col[this.options.sortParmFmt];
+      this.sortParm['sort_dir']=col.getSortDirection();
+    } else {
+      this.sortParm['s'+colnum]=col.getSortDirection();
+    }
+    this.clear();
+  },
+
+  printAllSQL: function(exportType) {
+    var parms=this.formQueryHashSQL(0,this.liveGrid.options.maxPrint,exportType);
+    parms.hidden=this.liveGrid.listInvisible('index').join(',');
+    var url=this.dataSource+'?'+Rico.toQueryString(parms);
+    window.open(url,'',this.liveGrid.options.exportWindow);
+  },
+
+  printVisibleSQL: function(exportType) {
+    var parms=this.formQueryHashSQL(this.liveGrid.contentStartPos-1, this.liveGrid.pageSize, exportType);
+    parms.hidden=this.liveGrid.listInvisible('index').join(',');
+    var url=this.dataSource+'?'+Rico.toQueryString(parms);
+    window.open(url,'',this.liveGrid.options.exportWindow);
+  },
+
+  // for datasource that is a javascript function
+  _printAll: function(exportType) {
+    this.liveGrid.exportStart();
+    this.ajaxOptions.parameters = this.formQueryHashSQL(0,this.liveGrid.options.maxPrint);
+    var self=this;
+    this.ajaxOptions.onComplete = function() { self._jsExport(exportType); };
+    this.dataSource(this.ajaxOptions);
+  },
+
+  _jsExport: function(exportType, newRows, newAttr, totalRows, errMsg) {
+    Rico.log("_jsExport: "+arguments.length);
+    if (errMsg) {
+      Rico.log("_jsExport: received error="+errMsg);
+      this.liveGrid.showMsg(Rico.getPhraseById("requestError",errMsg));
+      return;
+    }
+    this.exportBuffer(newRows,0);
+    this.liveGrid.exportFinish(exportType);
+  }
+
+};
diff --git a/minsrc/ricoLiveGridControls.js b/minsrc/ricoLiveGridControls.js
new file mode 100644 (file)
index 0000000..cd2989c
--- /dev/null
@@ -0,0 +1,423 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+\r
+// -----------------------------------------------------\r
+//\r
+// Custom formatting for LiveGrid columns\r
+//\r
+// columnSpecs Usage: { type:'control', control:new Rico.TableColumn.CONTROLNAME() }\r
+//\r
+// -----------------------------------------------------\r
+
+Rico.TableColumn = {};
+\r
+Rico.TableColumn.checkboxKey = function(showKey) {
+  this.initialize(showKey);
+}
+
+Rico.TableColumn.checkboxKey.prototype = {
+/**
+ * @class Custom formatting for a LiveGrid column.
+ * Display unique key column as: &lt;checkbox&gt; &lt;key value&gt;
+ * and keep track of which keys the user selects
+ * Key values should not contain &lt;, &gt;, or &amp;
+ * @constructs
+ */
+  initialize: function(showKey) {
+    this._checkboxes=[];
+    this._spans=[];
+    this._KeyHash={};
+    this._showKey=showKey;\r
+  },
+
+  _create: function(gridCell,windowRow) {
+    this._checkboxes[windowRow]=Rico.createFormField(gridCell,'input','checkbox',this.liveGrid.tableId+'_chkbox_'+this.index+'_'+windowRow);
+    this._spans[windowRow]=Rico.createFormField(gridCell,'span',null,this.liveGrid.tableId+'_desc_'+this.index+'_'+windowRow);
+    this._clear(gridCell,windowRow);
+    Rico.eventBind(this._checkboxes[windowRow], 'click', Rico.eventHandle(this,'_onclick'));
+  },
+
+  _onclick: function(e) {
+    var elem=Event.element(e);
+    var windowRow=parseInt(elem.id.substr((elem.id.lastIndexOf('_',elem.id.length)+1)));  //faster than split
+    var v=this.getValue(windowRow);
+    if (elem.checked)
+      this._addChecked(v);
+    else
+      this._remChecked(v);
+  },
+
+  _clear: function(gridCell,windowRow) {
+    var box=this._checkboxes[windowRow];
+    box.checked=false;
+    box.style.display='none';
+    this._spans[windowRow].innerHTML='';
+  },
+
+  _display: function(v,gridCell,windowRow) {
+    var box=this._checkboxes[windowRow];
+    box.style.display='';
+    box.checked=this._KeyHash[v];
+    if (this._showKey) this._spans[windowRow].innerHTML=v;
+  },
+
+  _SelectedKeys: function() {
+    return Rico.keys(this._KeyHash);
+  },
+
+  _addChecked: function(k){\r
+    this._KeyHash[k]=1;\r
+  },\r
+\r
+  _remChecked: function(k){\r
+    delete this._KeyHash[k];\r
+  }\r
+}
+
+
+Rico.TableColumn.checkbox = function(checkedValue, uncheckedValue, defaultValue, readOnly)
+{
+  this.initialize(checkedValue, uncheckedValue, defaultValue, readOnly);
+}
+
+Rico.TableColumn.checkbox.prototype = {
+/**
+ * @class display checkboxes for two-valued column (e.g. yes/no)
+ * @constructs
+ */
+  initialize: function(checkedValue, uncheckedValue, defaultValue, readOnly) {
+    this._checkedValue=checkedValue;
+    this._uncheckedValue=uncheckedValue;
+    this._defaultValue=defaultValue || false;
+    this._readOnly=readOnly || false;
+    this._checkboxes=[];
+  },
+
+  _create: function(gridCell,windowRow) {
+    this._checkboxes[windowRow]=Rico.createFormField(gridCell,'input','checkbox',this.liveGrid.tableId+'_chkbox_'+this.index+'_'+windowRow);
+    this._clear(gridCell,windowRow);
+    if (this._readOnly)
+      this._checkboxes[windowRow].disabled=true;
+    else
+      Rico.eventBind(this._checkboxes[windowRow], 'click', Rico.eventHandle(this,'_onclick'));
+  },
+
+  _onclick: function(e) {
+    var elem=Event.element(e);
+    var windowRow=parseInt(elem.id.substr((elem.id.lastIndexOf('_',elem.id.length)+1)));  //faster than split
+    var newval=elem.checked ? this._checkedValue : this._uncheckedValue;
+    this.setValue(windowRow,newval);
+  },
+
+  _clear: function(gridCell,windowRow) {
+    var box=this._checkboxes[windowRow];
+    box.checked=this._defaultValue;
+    box.style.display='none';
+  },
+
+  _display: function(v,gridCell,windowRow) {
+    var box=this._checkboxes[windowRow];
+    box.style.display='';
+    box.checked=(v==this._checkedValue);
+  }
+
+}
+
+
+Rico.TableColumn.textbox = function(boxSize, boxMaxLen, readOnly) {
+  this.initialize(boxSize, boxMaxLen, readOnly);
+}
+
+Rico.TableColumn.textbox.prototype = {
+/**
+ * @class display value in a text box
+ * @constructs
+ */
+  initialize: function(boxSize, boxMaxLen, readOnly) {
+    this._boxSize=boxSize;
+    this._boxMaxLen=boxMaxLen;
+    this._readOnly=readOnly || false;
+    this._textboxes=[];
+  },
+
+  _create: function(gridCell,windowRow) {
+    var box=Rico.createFormField(gridCell,'input','text',this.liveGrid.tableId+'_txtbox_'+this.index+'_'+windowRow);
+    box.size=this._boxSize;
+    box.maxLength=this._boxMaxLen;
+    this._textboxes[windowRow]=box;
+    this._clear(gridCell,windowRow);
+    if (this._readOnly)
+      box.disabled=true;
+    else
+      Rico.eventBind(box, 'change', Rico.eventHandle(this,'_onchange'));
+  },
+
+  _onchange: function(e) {
+    var elem=Event.element(e);
+    var windowRow=parseInt(elem.id.substr((elem.id.lastIndexOf('_',elem.id.length)+1)));  //faster than split
+    this.setValue(windowRow,elem.value);
+  },
+
+  _clear: function(gridCell,windowRow) {
+    var box=this._textboxes[windowRow];
+    box.value='';
+    box.style.display='none';
+  },
+
+  _display: function(v,gridCell,windowRow) {
+    var box=this._textboxes[windowRow];
+    box.style.display='';
+    box.value=v;
+  }
+
+}
+
+
+Rico.TableColumn.HighlightCell = function(chkcol,chkval,highlightColor,highlightBackground,chkop) {
+  this.initialize(chkcol,chkval,highlightColor,highlightBackground,chkop);
+}
+
+Rico.TableColumn.HighlightCell.prototype = {
+/**
+ * @class highlight a grid cell when a particular value is present in the specified column
+ * @constructs
+ */
+  initialize: function(chkcol,chkval,highlightColor,highlightBackground,chkop) {\r
+    this._chkcol=chkcol;\r
+    this._chkval=chkval;
+    this._chkop=chkop;\r
+    this._highlightColor=highlightColor;\r
+    this._highlightBackground=highlightBackground;\r
+  },\r
+\r
+  _clear: function(gridCell,windowRow) {\r
+    gridCell.style.color='';\r
+    gridCell.style.backgroundColor='';\r
+    gridCell.innerHTML='&nbsp;';\r
+  },\r
+\r
+  _display: function(v,gridCell,windowRow) {\r
+    var gridval=this.liveGrid.buffer.getWindowValue(windowRow,this._chkcol);\r
+    var match;\r
+    switch(this._chkop){
+        case '!=':
+          match=(gridval!=this._chkval);
+          break;
+        case '>':
+          match=(gridval>this._chkval);
+          break;
+        case '<':
+          match=(gridval<this._chkval);
+          break;
+        case '>=':
+          match=(gridval>=this._chkval);
+          break;
+        case '<=':
+          match=(gridval<=this._chkval);
+          break;
+        case 'abs>':
+          match=(Math.abs(gridval)>this._chkval);
+          break;
+        case 'abs<':
+          match=(Math.abs(gridval)<this._chkval);
+          break;
+        case 'abs>=':
+          match=(Math.abs(gridval)>=this._chkval);
+          break;
+        case 'abs<=':
+          match=(Math.abs(gridval)<=this._chkval);
+          break;
+        default:
+          match=(gridval==this._chkval);
+          break;
+    }
+    gridCell.style.color=match ? this._highlightColor : '';\r
+    gridCell.style.backgroundColor=match ? this._highlightBackground : '';\r
+    gridCell.innerHTML=this._format(v);\r
+  }\r
+}
+
+
+Rico.TableColumn.bgColor = function() {
+}
+
+Rico.TableColumn.bgColor.prototype = {
+/**
+ * @class database value contains a css color name/value
+ */
+ _clear: function(gridCell,windowRow) {
+    gridCell.style.backgroundColor='';
+  },
+
+  _display: function(v,gridCell,windowRow) {
+    gridCell.style.backgroundColor=v;
+  }
+
+}
+
+
+Rico.TableColumn.link = function(href,target,linktext) {
+  this.initialize(href,target,linktext);
+}
+
+Rico.TableColumn.link.prototype = {
+/**
+ * @class database value contains a url to another page
+ * @constructs
+ */
+  initialize: function(href,target,linktext) {
+    this._href=href;
+    this._target=target;
+    this._linktext=linktext;
+    this._anchors=[];
+  },
+
+  _create: function(gridCell,windowRow) {
+    this._anchors[windowRow]=Rico.createFormField(gridCell,'a',null,this.liveGrid.tableId+'_a_'+this.index+'_'+windowRow);
+    if (this._target) this._anchors[windowRow].target=this._target;
+    this._clear(gridCell,windowRow);
+  },
+
+  _clear: function(gridCell,windowRow) {
+    this._anchors[windowRow].href='';
+    this._anchors[windowRow].innerHTML='';
+  },
+
+  _display: function(v,gridCell,windowRow) {
+    var buf=this.liveGrid.buffer;
+    var href=this._href=='self' ? v : this._href.replace(/\{\d+\}/g,
+      function ($1) {
+        var colIdx=parseInt($1.substr(1),10);
+        return encodeURIComponent(buf.getWindowValue(windowRow,colIdx));
+      }
+    );
+    var desc=this._linktext || v;
+    if (href && desc) {
+      this._anchors[windowRow].href=href;
+      this._anchors[windowRow].innerHTML=desc;
+    } else {
+      this._clear(gridCell,windowRow);
+    }
+  }
+
+};
+
+
+Rico.TableColumn.image = function(prefix,suffix) {
+  this.initialize(prefix,suffix);
+};
+
+Rico.TableColumn.image.prototype = {
+/**
+ * @class database value contains a url to an image
+ * @constructs
+ */
+  initialize: function(prefix,suffix) {
+    this._img=[];
+    this._prefix=prefix || '';
+    this._suffix=suffix || '';
+  },
+
+  _create: function(gridCell,windowRow) {
+    this._img[windowRow]=Rico.createFormField(gridCell,'img',null,this.liveGrid.tableId+'_img_'+this.index+'_'+windowRow);
+    this._clear(gridCell,windowRow);
+  },
+
+  _clear: function(gridCell,windowRow) {
+    var img=this._img[windowRow];
+    img.style.display='none';
+    img.src='';
+  },
+
+  _display: function(v,gridCell,windowRow) {
+    var img=this._img[windowRow];
+    this._img[windowRow].src=this._prefix+v+this._suffix;
+    img.style.display='';
+  }
+
+};
+
+
+Rico.TableColumn.lookup = function(map, defaultCode, defaultDesc) {
+  this.initialize(map, defaultCode, defaultDesc);
+};
+
+Rico.TableColumn.lookup.prototype = {
+/**
+ * @class map a database value to a display value
+ * @constructs
+ */
+  initialize: function(map, defaultCode, defaultDesc) {
+    this._map=map;
+    this._defaultCode=defaultCode || '';
+    this._defaultDesc=defaultDesc || '&nbsp;';
+    var self=this;
+    this._sortfunc=function(v) { return self._sortvalue(v); };
+    this._codes=[];
+    this._descriptions=[];
+  },
+
+  _create: function(gridCell,windowRow) {
+    this._descriptions[windowRow]=Rico.createFormField(gridCell,'span',null,this.liveGrid.tableId+'_desc_'+this.index+'_'+windowRow);
+    this._codes[windowRow]=Rico.createFormField(gridCell,'input','hidden',this.liveGrid.tableId+'_code_'+this.index+'_'+windowRow);
+    this._clear(gridCell,windowRow);
+  },
+
+  _clear: function(gridCell,windowRow) {
+    this._codes[windowRow].value=this._defaultCode;
+    this._descriptions[windowRow].innerHTML=this._defaultDesc;
+  },
+
+  _sortvalue: function(v) {
+    return this._getdesc(v).replace(/&amp;/g, '&').replace(/&lt;/g,'<').replace(/&gt;/g,'>').replace(/&nbsp;/g,' ');
+  },
+
+  _getdesc: function(v) {
+    var desc=this._map[v];
+    return (typeof desc=='string') ? desc : this._defaultDesc;
+  },
+
+  _export: function(v) {
+    return this._getdesc(v);
+  },
+
+  _display: function(v,gridCell,windowRow) {
+    this._codes[windowRow].value=v;
+    this._descriptions[windowRow].innerHTML=this._getdesc(v);
+  }
+
+};
+
+
+
+Rico.TableColumn.MultiLine = function() {
+};
+
+Rico.TableColumn.MultiLine.prototype = {
+/**
+ * @class Fix issues with multiline content in IE
+ */
+  _display: function(v,gridCell,windowRow) {\r
+    var newdiv = document.createElement("div");\r
+    newdiv.innerHTML = this._format(v);\r
+    newdiv.style.height='100%';\r
+    if (gridCell.firstChild)\r
+      gridCell.replaceChild(newdiv, gridCell.firstChild);\r
+    else\r
+      gridCell.appendChild(newdiv);\r
+  }\r
+
+};
diff --git a/minsrc/ricoLiveGridForms.js b/minsrc/ricoLiveGridForms.js
new file mode 100644 (file)
index 0000000..dfbf887
--- /dev/null
@@ -0,0 +1,1136 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+if(typeof Rico=='undefined') throw("LiveGridForms requires the Rico JavaScript framework");
+
+
+Rico.TableEdit = function(liveGrid) {
+  this.initialize(liveGrid);
+}
+
+Rico.TableEdit.prototype = {
+/**
+ * @class Supports editing LiveGrid data.
+ * @constructs
+ */
+  initialize: function(liveGrid) {
+    Rico.log('Rico.TableEdit initialize: '+liveGrid.tableId);
+    this.grid=liveGrid;
+    this.options = {
+      maxDisplayLen    : 20,    // max displayed text field length
+      panelHeight      : 200,   // size of tabbed panels
+      panelWidth       : 500,
+      compact          : false,    // compact corners
+      RecordName       : Rico.getPhraseById("record"),
+      updateURL        : window.location.href, // default is that updates post back to the generating page
+      showSaveMsg      : 'errors'  // disposition of database update responses (full - show full response, errors - show full response for errors and short response otherwise)
+    };
+    Rico.extend(this.options, liveGrid.options);
+    var self=this;
+    this.hasWF2=(document.implementation && document.implementation.hasFeature && document.implementation.hasFeature('WebForms', '2.0'));
+    this.menu=liveGrid.menu;
+    this.menu.options.dataMenuHandler=function(grid,r,c,onBlankRow) { return self.editMenu(grid,r,c,onBlankRow); };
+    this.menu.ignoreClicks();
+    this.editText=Rico.getPhraseById("editRecord",this.options.RecordName);
+    this.cloneText=Rico.getPhraseById("cloneRecord",this.options.RecordName);
+    this.delText=Rico.getPhraseById("deleteRecord",this.options.RecordName);
+    this.addText=Rico.getPhraseById("addRecord",this.options.RecordName);
+    this.buttonHover=new Rico.HoverSet();
+    this.dateRegExp=/^\s*(\w+)(\W)(\w+)(\W)(\w+)/i;
+    Rico.EditControls.atLoad();
+    this.createKeyArray();
+    this.createEditDiv();
+    this.saveMsg=Rico.$(liveGrid.tableId+'_savemsg');
+    Rico.eventBind(document,"click", Rico.eventHandle(this,'clearSaveMsg'));
+    this.extraMenuItems=[];
+    this.responseHandler=function() { self.processResponse(); };
+    Rico.log("Rico.TableEdit.initialize complete, hasWF2="+this.hasWF2);
+  },
+
+  createKeyArray: function() {
+    this.keys=[];
+    for (var i=0; i<this.grid.columns.length; i++) {
+      if (this.grid.columns[i].format && this.grid.columns[i].format.isKey)
+        this.keys.push({colidx:i});
+    }
+  },
+
+  createEditDiv: function() {
+
+    // create popup form
+
+    this.requestCount=1;
+    this.formPopup=new Rico.Window('', {overflow: this.options.panels ? 'hidden' : 'auto'});
+    Rico.addClass(this.formPopup.content,'ricoLG_editDiv');
+    if (this.options.canEdit || this.options.canAdd) {
+      this.startForm();
+      this.createForm(this.form);
+    } else {
+      var buttonClose=this.createButton(Rico.getPhraseById("close"));
+      Rico.eventBind(buttonClose,"click", Rico.eventHandle(this,'cancelEdit'), false);
+      this.createForm(this.formPopup.contentDiv);
+    }
+    this.editDivCreated=true;
+
+    // create responseDialog
+
+    this.responseDialog = this.grid.createDiv('editResponse',document.body);
+    this.responseDialog.style.display='none';
+
+    var buttonOK = document.createElement('button');
+    buttonOK.appendChild(document.createTextNode('OK'));
+    Rico.eventBind(buttonOK,"click", Rico.eventHandle(this,'ackResponse'));
+    this.responseDialog.appendChild(buttonOK);
+
+    this.responseDiv = this.grid.createDiv('editResponseText',this.responseDialog);
+
+    if (this.panelGroup) {
+      Rico.log("createEditDiv complete, requestCount="+this.requestCount);
+      Rico.runLater(50,this,'initPanelGroup');
+    }
+  },
+
+  initPanelGroup: function() {
+    this.requestCount--;
+    Rico.log("initPanelGroup: "+this.requestCount);
+    if (this.requestCount>0) return;
+    var wi=parseInt(this.options.panelWidth,10);
+    if (this.form) {
+      //this.form.style.width=(wi+10)+'px';
+      if (Rico.isWebKit) this.formPopup.container.style.display='block';  // this causes display to flash briefly
+      this.options.bgColor = Rico.Color.createColorFromBackground(this.form).toString();
+    }
+    this.formPopup.container.style.display='none';
+    //this.options.panelHdrWidth=(Math.floor(wi / this.options.panels.length)-20)+'px';
+    this.formPanels=new Rico.TabbedPanel(this.panelGroup, this.options);
+  },
+
+  notEmpty: function(v) {
+    return typeof(v)!='undefined';
+  },
+
+  startForm: function() {
+    this.form = document.createElement('form');
+    /** @ignore */
+    this.form.onsubmit=function() {return false;};
+    this.form.autocomplete="off"; // seems to fix "Permission denied..." errors in FF
+    this.formPopup.contentDiv.appendChild(this.form);
+
+    var tab = document.createElement('div');
+    tab.className='ButtonBar';
+    var button=tab.appendChild(this.createButton(Rico.getPhraseById("saveRecord",this.options.RecordName)));
+    Rico.eventBind(button,"click", Rico.eventHandle(this,'TESubmit'), false);
+    button=tab.appendChild(this.createButton(Rico.getPhraseById("cancel")));
+    Rico.eventBind(button,"click", Rico.eventHandle(this,'cancelEdit'), false);
+    this.form.appendChild(tab);
+
+    // hidden fields
+    this.hiddenFields = document.createElement('div');
+    this.hiddenFields.style.display='none';
+    this.action = this.appendHiddenField(this.grid.actionId,'');
+    var i,fldSpec;
+    for (i=0; i<this.grid.columns.length; i++) {
+      fldSpec=this.grid.columns[i].format;
+      if (fldSpec && fldSpec.FormView && fldSpec.FormView=="hidden")
+        this.appendHiddenField(fldSpec.FieldName,fldSpec.ColData);
+    }
+    for (var k=0; k<this.keys.length; k++) {
+      this.keys[k].keyField = this.appendHiddenField('_k'+this.keys[k].colidx,'');
+    }
+    this.form.appendChild(this.hiddenFields);
+  },
+
+  createButton: function(buttonLabel) {
+    var button = document.createElement('a');
+    button.href='javascript:void(0)';
+    button.innerHTML=buttonLabel;
+    button.className='RicoButton';
+    if (Rico.theme.button) Rico.addClass(button,Rico.theme.button);
+    this.buttonHover.add(button);
+    return button;
+  },
+
+  createPanel: function(i) {
+    var hasFields=false;
+    for (var j=0; j<this.grid.columns.length; j++) {
+      var fldSpec=this.grid.columns[j].format;
+      if (!fldSpec) continue;
+      if (!fldSpec.EntryType) continue;
+      if (fldSpec.EntryType=='H') continue;
+      if (fldSpec.FormView && fldSpec.FormView=="hidden") continue;
+      var panelIdx=fldSpec.panelIdx || 0;
+      if (panelIdx==i) {
+        hasFields=true;
+        break;
+      }
+    }
+    if (!hasFields) return null;
+    this.panelHdr[i] = document.createElement('li');
+    this.panelHdr[i].innerHTML=this.options.panels[i];
+    this.panelHdrs.appendChild(this.panelHdr[i]);
+    this.panelContent[i] = document.createElement('div');
+    this.panelContents.appendChild(this.panelContent[i]);
+    this.panelActualIdx[i]=this.panelCnt++;
+    return this.createFormTable(this.panelContent[i],'tabContent');
+  },
+
+  createForm: function(parentDiv) {
+    var i,div,fldSpec,panelIdx,tables=[];
+    this.panelCnt=0;
+    this.panelHdr=[];
+    this.panelContent=[];
+    if (this.options.panels) {
+      this.panelGroup = document.createElement('div');
+      this.panelGroup.className='tabPanelGroup';
+      this.panelHdrs = document.createElement('ul');
+      this.panelGroup.appendChild(this.panelHdrs);
+      this.panelContents = document.createElement('div');
+      this.panelContents.className='tabContentContainer';
+      this.panelGroup.appendChild(this.panelContents);
+      this.panelActualIdx=[];
+      parentDiv.appendChild(this.panelGroup);
+      if (this.grid.direction=='rtl') {
+        for (i=this.options.panels.length-1; i>=0; i--) {
+          tables[i]=this.createPanel(i);
+        }
+      } else {
+        for (i=0; i<this.options.panels.length; i++) {
+          tables[i]=this.createPanel(i);
+        }
+      }
+      parentDiv.appendChild(this.panelGroup);
+    } else {
+      div=document.createElement('div');
+      div.className='noTabContent';
+      tables[0]=this.createFormTable(div);
+      parentDiv.appendChild(div);
+    }
+    for (i=0; i<this.grid.columns.length; i++) {
+      fldSpec=this.grid.columns[i].format;
+      if (!fldSpec) continue;
+      panelIdx=fldSpec.panelIdx || 0;
+      if (tables[panelIdx]) this.appendFormField(this.grid.columns[i],tables[panelIdx]);
+      if (typeof fldSpec.pattern=='string') {
+        switch (fldSpec.pattern) {
+          case 'email':
+            fldSpec.regexp=/^[_a-zA-Z0-9-]+(\.[_a-zA-Z0-9-]+)*@[a-zA-Z0-9-]+(\.[a-zA-Z0-9-]+)*\.(([0-9]{1,3})|([a-zA-Z]{2,3})|(aero|coop|info|museum|name))$/;
+            break;
+          case 'float-unsigned':
+            fldSpec.regexp=/^\d+(\.\d+)?$/;
+            break;
+          case 'float-signed':
+            fldSpec.regexp=/^[-+]?\d+(\.\d+)?$/;
+            break;
+          case 'int-unsigned':
+            fldSpec.regexp=/^\d+$/;
+            break;
+          case 'int-signed':
+            fldSpec.regexp=/^[-+]?\d+$/;
+            break;
+          default:
+            fldSpec.regexp=new RegExp(fldSpec.pattern);
+            break;
+        }
+      }
+    }
+  },
+
+  createFormTable: function(div) {
+    var tab=document.createElement('table');
+    tab.border=0;
+    div.appendChild(tab);
+    return tab;
+  },
+
+  appendHiddenField: function(name,value) {
+    var field=Rico.createFormField(this.hiddenFields,'input','hidden',name,name);
+    field.value=value;
+    return field;
+  },
+
+  appendFormField: function(column, table) {
+    var fmt=column.format;
+    if (!fmt.EntryType) return;
+    if (fmt.EntryType=="H") return;
+    if (fmt.FormView) return;
+    Rico.log('appendFormField: '+column.displayName+' - '+fmt.EntryType);
+    var row = fmt.noFormBreak && table.rows.length > 0 ? table.rows[table.rows.length-1] : table.insertRow(-1);
+    var hdr = row.insertCell(-1);
+    column.formLabel=hdr;
+    if (hdr.noWrap) hdr.noWrap=true;
+    var entry = row.insertCell(-1);
+    if (entry.noWrap) entry.noWrap=true;
+    hdr.id='lbl_'+fmt.FieldName;
+    var field, name=fmt.FieldName;
+    switch (fmt.EntryType) {
+      case 'TA':
+      case 'tinyMCE':
+        field=Rico.createFormField(entry,'textarea',null,name);
+        field.cols=fmt.TxtAreaCols;
+        field.rows=fmt.TxtAreaRows;
+        field.innerHTML=fmt.ColData;
+        hdr.style.verticalAlign='top';
+        break;
+      case 'R':
+      case 'RL':
+        field=Rico.createFormField(entry,'div',null,name);
+        if (fmt.DescriptionField) field.RicoUpdate=fmt.DescriptionField;
+        if (fmt.MultiSelect) Rico.addClass(field, 'MultiSelect');
+        if (fmt.isNullable && !fmt.MultiSelect) this.addSelectNone(field);
+        this.selectValuesRequest(field,column);
+        break;
+      case 'N':
+        field=Rico.createFormField(entry,'select',null,name);
+        if (fmt.isNullable) this.addSelectNone(field);
+        Rico.eventBind(field,"change", Rico.eventHandle(this,'checkSelectNew'));
+        this.selectValuesRequest(field,column);
+        field=document.createElement('span');
+        field.className='ricoEditLabel';
+        field.id='labelnew__'+fmt.FieldName;
+        field.innerHTML='&nbsp;&nbsp;&nbsp;'+Rico.getPhraseById('formNewValue').replace(' ','&nbsp;');
+        entry.appendChild(field);
+        name='textnew__'+fmt.FieldName;
+        field=Rico.createFormField(entry,'input','text',name,name);
+        break;
+      case 'S':
+      case 'SL':
+        if (fmt.ReadOnly) {
+          field=Rico.createFormField(entry,'input','text',name,name);
+          this.initField(field,fmt);
+        } else {
+          field=Rico.createFormField(entry,'select',null,name);
+          if (fmt.MultiSelect) field.multiple=true;
+          if (fmt.SelectRows) field.size=parseInt(fmt.SelectRows,10);
+          if (fmt.isNullable && !fmt.MultiSelect) this.addSelectNone(field);
+          if (fmt.DescriptionField) {
+            field.RicoUpdate=fmt.DescriptionField;
+            Rico.eventBind(field,"change", Rico.eventHandle(this,'selectClick'), false);
+          }
+          this.selectValuesRequest(field,column);
+        }
+        break;
+      case 'D':
+        if (!fmt.isNullable) fmt.required=true;
+        if (!fmt.dateFmt) fmt.dateFmt=Rico.dateFmt;
+        if (!fmt.Help) fmt.Help=fmt.dateFmt;
+        if (typeof fmt.min=='string') fmt.min=Rico.setISO8601(fmt.min) || new Date(fmt.min);
+        if (typeof fmt.max=='string') fmt.max=Rico.setISO8601(fmt.max) || new Date(fmt.max);
+        if (this.hasWF2) {
+          field=Rico.createFormField(entry,'input','date',name,name);
+          field.required=fmt.required;
+          if (fmt.min) field.min=Rico.toISO8601String(fmt.min,3);
+          if (fmt.max) field.max=Rico.toISO8601String(fmt.max,3);
+          field.required=fmt.required;
+          fmt.SelectCtl=null;  // use the WebForms calendar instead of the Rico calendar
+        } else {
+          field=Rico.createFormField(entry,'input','text',name,name);
+        }
+        this.initField(field,fmt);
+        break;
+      case 'I':
+        if (!fmt.isNullable) fmt.required=true;
+        if (!fmt.pattern) fmt.pattern='int-signed';
+        if (this.hasWF2) {
+          field=Rico.createFormField(entry,'input','number',name,name);
+          field.required=fmt.required;
+          field.min=fmt.min;
+          field.max=fmt.max;
+          field.step=1;
+        } else {
+          field=Rico.createFormField(entry,'input','text',name,name);
+        }
+        if (typeof fmt.min=='string') fmt.min=parseInt(fmt.min,10);
+        if (typeof fmt.max=='string') fmt.max=parseInt(fmt.max,10);
+        this.initField(field,fmt);
+        break;
+      case 'F':
+        if (!fmt.isNullable) fmt.required=true;
+        if (!fmt.pattern) fmt.pattern='float-signed';
+        field=Rico.createFormField(entry,'input','text',name,name);
+        this.initField(field,fmt);
+        if (typeof fmt.min=='string') fmt.min=parseFloat(fmt.min);
+        if (typeof fmt.max=='string') fmt.max=parseFloat(fmt.max);
+        break;
+      default:
+        field=Rico.createFormField(entry,'input','text',name,name);
+        if (!fmt.isNullable && fmt.EntryType!='T') fmt.required=true;
+        this.initField(field,fmt);
+        break;
+    }
+    if (field) {
+      if (fmt.SelectCtl)
+        Rico.EditControls.applyTo(column,field);
+    }
+    var hdrSuffix='';
+    hdr.className='ricoEditLabel';
+    if (fmt.Help) {
+      hdr.title=fmt.Help;
+      hdrSuffix="&nbsp;<span class='rico-icon rico-info'></span>";
+    }
+    var hdrText=fmt.EntryType.length>1 && fmt.EntryType.charAt(1)=='L' ? column.next.displayName : column.displayName;
+    hdr.innerHTML=hdrText+hdrSuffix;
+  },
+
+  addSelectNone: function(field) {
+    this.addSelectOption(field,this.options.TableSelectNone,Rico.getPhraseById("selectNone"));
+  },
+
+  initField: function(field,fmt) {
+    if (fmt.Length) {
+      field.maxLength=fmt.Length;
+      field.size=Math.min(fmt.Length, this.options.maxDisplayLen);
+    }
+    field.value=fmt.ColData;
+  },
+  
+  selectClick: function(e) {
+    var SelObj=Rico.eventElement(e);
+    if (SelObj.readOnly) {
+      Rico.eventStop(e);
+      return false;
+    }
+    if (SelObj.RicoUpdate) {
+      var opt=SelObj.options[SelObj.selectedIndex];
+      Rico.$(SelObj.RicoUpdate).value=opt.innerHTML;
+    }
+  },
+  
+  radioClick: function(e) {
+    var ChkBoxObj=Rico.eventElement(e);
+    if (ChkBoxObj.readOnly) {
+      Rico.eventStop(e);
+      return false;
+    }
+    var container=Rico.getParentByTagName(ChkBoxObj,'div');
+    if (container.RicoUpdate) {
+      Rico.$(container.RicoUpdate).value=ChkBoxObj.nextSibling.innerHTML;
+    }
+  },
+
+  checkSelectNew: function(e) {
+    this.updateSelectNew(Rico.eventElement(e));
+  },
+
+  updateSelectNew: function(SelObj) {
+    var vis=(SelObj.value==this.options.TableSelectNew) ? "" : "hidden";
+    Rico.$("labelnew__" + SelObj.id).style.visibility=vis;
+    Rico.$("textnew__" + SelObj.id).style.visibility=vis;
+  },
+
+  selectValuesRequest: function(elem,column) {
+    var fldSpec=column.format;
+    if (fldSpec.SelectValues) {
+      var valueList=fldSpec.SelectValues.split(',');
+      for (var i=0; i<valueList.length; i++)
+        this.addSelectOption(elem,valueList[i],valueList[i],i);
+    } else {
+      this.requestCount++;
+      var options={}, self=this;
+      Rico.extend(options, this.grid.buffer.ajaxOptions);
+      options.parameters = {id: this.grid.tableId, offset: '0', page_size: '-1', edit: column.index};
+      options.parameters[this.grid.actionId]="query";
+      options.onComplete = function(request) { self.selectValuesUpdate(elem,request); };
+      new Rico.ajaxRequest(this.grid.buffer.dataSource, options);
+      Rico.log("selectValuesRequest: "+fldSpec.FieldName);
+    }
+  },
+
+  selectValuesUpdate: function(elem,request) {
+    var response = request.responseXML.getElementsByTagName("ajax-response");
+    Rico.log("selectValuesUpdate: "+request.status);
+    if (response == null || response.length != 1) return;
+    response=response[0];
+    var error = response.getElementsByTagName('error');
+    if (error.length > 0) {
+      var errmsg=Rico.getContentAsString(error[0],this.grid.buffer.isEncoded);
+      Rico.log("Data provider returned an error:\n"+errmsg);
+      alert(Rico.getPhraseById("requestError",errmsg));
+      return;
+    }
+    response=response.getElementsByTagName('response')[0];
+    var rowsElement = response.getElementsByTagName('rows')[0];
+    var rows = this.grid.buffer.dom2jstable(rowsElement);
+    Rico.log("selectValuesUpdate: id="+elem.id+' rows='+rows.length);
+    for (var i=0; i<rows.length; i++) {
+      if (rows[i].length>0) {
+        var c0=rows[i][0];
+        var c1=(rows[i].length>1) ? rows[i][1] : c0;
+        this.addSelectOption(elem,c0,c1,i);
+      }
+    }
+    if (Rico.$('textnew__'+elem.id))
+      this.addSelectOption(elem,this.options.TableSelectNew,Rico.getPhraseById("selectNewVal"));
+    if (this.panelGroup)
+      Rico.runLater(50,this,'initPanelGroup');
+  },
+
+  addSelectOption: function(elem,value,text,idx) {
+    switch (elem.tagName.toLowerCase()) {
+      case 'div':
+        var opt=Rico.createFormField(elem,'input', Rico.hasClass(elem, 'MultiSelect') ? 'checkbox' : 'radio', elem.id+'_'+idx, elem.id);
+        opt.value=value;
+        var lbl=document.createElement('label');
+        lbl.innerHTML=text;
+        lbl.htmlFor=opt.id;
+        elem.appendChild(lbl);
+        Rico.eventBind(opt,"click", Rico.eventHandle(this,'radioClick'), false);
+        break;
+      case 'select':
+        Rico.addSelectOption(elem,value,text);
+        break;
+    }
+  },
+
+  clearSaveMsg: function() {
+    if (this.saveMsg) this.saveMsg.innerHTML="";
+  },
+
+  addMenuItem: function(menuText,menuAction,enabled) {
+    this.extraMenuItems.push({menuText:menuText,menuAction:menuAction,enabled:enabled});
+  },
+
+  editMenu: function(grid,r,c,onBlankRow) {
+    this.clearSaveMsg();
+    if (this.grid.buffer.sessionExpired==true || this.grid.buffer.startPos<0) return false;
+    this.rowIdx=r;
+    var elemTitle=Rico.$('pageTitle');
+    var pageTitle=elemTitle ? elemTitle.innerHTML : document.title;
+    this.menu.addMenuHeading(pageTitle);
+    var self=this;
+    if (onBlankRow==false) {
+      for (var i=0; i<this.extraMenuItems.length; i++) {
+        this.menu.addMenuItem(this.extraMenuItems[i].menuText,this.extraMenuItems[i].menuAction,this.extraMenuItems[i].enabled);
+      }
+      this.menu.addMenuItem(this.editText, function() { self.editRecord(); },this.canEdit(r));
+      this.menu.addMenuItem(this.delText, function() { self.deleteRecord(); },this.canDelete(r));
+      if (this.options.canClone) {
+        this.menu.addMenuItem(this.cloneText, function() { self.cloneRecord(); },this.canAdd(r) && this.canEdit(r));
+      }
+    }
+    this.menu.addMenuItem(this.addText, function() { self.addRecord(); },this.canAdd(r));
+    return true;
+  },
+  
+  canAdd: function(r) {
+    return (typeof this.options.canAdd=='function') ? this.options.canAdd(r) : this.options.canAdd;
+  },
+
+  canEdit: function(r) {
+    return (typeof this.options.canEdit=='function') ? this.options.canEdit(r) : this.options.canEdit;
+  },
+
+  canDelete: function(r) {
+    return (typeof this.options.canDelete=='function') ? this.options.canDelete(r) : this.options.canDelete;
+  },
+
+  cancelEdit: function(e) {
+    Rico.eventStop(e);
+    for (var i=0; i<this.grid.columns.length; i++) {
+      if (this.grid.columns[i].format && this.grid.columns[i].format.SelectCtl)
+        Rico.EditControls.close(this.grid.columns[i].format.SelectCtl);
+    }
+    this.makeFormInvisible();
+    this.grid.highlightEnabled=true;
+    this.menu.cancelmenu();
+    return false;
+  },
+
+  setField: function(fldnum,fldvalue) {
+    var fldSpec=this.grid.columns[fldnum].format;
+    var e=Rico.$(fldSpec.FieldName);
+    var a,i,o,elems,opts,txt;
+    if (!e) return;
+    Rico.log('setField: '+fldSpec.FieldName+'='+fldvalue);
+    switch (e.tagName.toUpperCase()) {
+      case 'DIV':
+        elems=e.getElementsByTagName('INPUT');
+        o={}
+        if (fldSpec.MultiSelect && fldvalue) {
+          a=fldvalue.split(',');
+          for (var i=0; i<a.length; i++) o[a[i]]=1;
+        } else {
+          o[fldvalue]=1;
+        }
+        for (i=0; i<elems.length; i++)
+          elems[i].checked=o[elems[i].value]==1;
+        break;
+      case 'INPUT':
+        if (fldSpec.EntryType=='D' && fldvalue!=fldSpec.ColData) {
+          // remove time data if it exists
+          a=fldvalue.split(/\s|T/);
+          fldvalue=a[0];
+          if (this.isTextInput(e)) {
+            var d=Rico.setISO8601(fldvalue);
+            if (d) fldvalue=Rico.formatDate(d,fldSpec.dateFmt);
+          }
+        }
+        e.value=fldvalue;
+        break;
+      case 'SELECT':
+        opts=e.options;
+        //alert('setField SELECT: id='+e.id+'\nvalue='+fldvalue+'\nopt cnt='+opts.length)
+        o={}
+        if (fldSpec.MultiSelect && fldvalue) {
+          a=fldvalue.split(',');
+          for (var i=0; i<a.length; i++) o[a[i]]=1;
+          for (i=0; i<opts.length; i++)
+            opts[i].selected=o[opts[i].value]==1;
+        } else {
+          for (i=0; i<opts.length; i++) {
+            if (opts[i].value==fldvalue) {
+              e.selectedIndex=i;
+              break;
+            }
+          }
+        }
+        if (fldSpec.EntryType=='N') {
+          txt=Rico.$('textnew__'+e.id);
+          if (!txt) alert('Warning: unable to find id "textnew__'+e.id+'"');
+          txt.value=fldvalue;
+          if (e.selectedIndex!=i) e.selectedIndex=opts.length-1;
+          this.updateSelectNew(e);
+        }
+        return;
+      case 'TEXTAREA':
+        e.value=fldvalue;
+        if (fldSpec.EntryType=='tinyMCE' && typeof(tinyMCE)!='undefined' && this.initialized) {
+          if (tinyMCE.updateContent) {
+            tinyMCE.updateContent(e.id);  // version 2.x
+          } else {
+            tinyMCE.execInstanceCommand(e.id, 'mceSetContent', false, fldvalue);  // version 3.x
+          }
+        }
+        return;
+    }
+  },
+
+  setReadOnly: function(action) {
+    for (var ro,i=0; i<this.grid.columns.length; i++) {
+      var fldSpec=this.grid.columns[i].format;
+      if (!fldSpec) continue;
+      var e=Rico.$(fldSpec.FieldName);
+      if (!e) continue;
+      switch (action) {
+        case 'ins': ro=!fldSpec.Writeable || fldSpec.ReadOnly || fldSpec.UpdateOnly; break;
+        case 'upd': ro=!fldSpec.Writeable || fldSpec.ReadOnly || fldSpec.InsertOnly; break;
+        default:    ro=false; break;
+      }
+      switch (e.tagName.toUpperCase()) {
+        case 'DIV':
+          var elems=e.getElementsByTagName('INPUT');
+          for (var j=0; j<elems.length; j++) {
+            elems[j].disabled=ro;
+          }
+          break;
+        case 'SELECT':
+          if (fldSpec.EntryType=='N') {
+            var txt=Rico.$('textnew__'+e.id);
+            txt.disabled=ro;
+          }
+          e.disabled=ro;
+          break;
+        case 'TEXTAREA':
+        case 'INPUT':
+          e.disabled=ro;
+          if (fldSpec.selectIcon) fldSpec.selectIcon.style.display=ro ? 'none' : '';
+          break;
+      }
+    }
+  },
+
+  hideResponse: function(msg) {
+    this.responseDiv.innerHTML=msg;
+    this.responseDialog.style.display='none';
+  },
+
+  showResponse: function() {
+    var offset=Rico.cumulativeOffset(this.grid.outerDiv);
+    offset.top+=Rico.docScrollTop();
+    this.responseDialog.style.top=offset.top+"px";
+    this.responseDialog.style.left=offset.left+"px";
+    this.responseDialog.style.display='';
+  },
+
+  processResponse: function(xhr) {
+    var responseText,success=true;
+    Rico.log('Processing response from form submittal');
+    this.responseDiv.innerHTML=xhr.responseText;
+    var respNodes=Rico.select('.ricoFormResponse',this.responseDiv);
+    if (respNodes) {
+      // generate a translated response
+      var phraseId=Rico.trim(respNodes[0].className).split(/\s+/)[1];
+      responseText=Rico.getPhraseById(phraseId,this.options.RecordName);
+    } else {
+      // present the response as sent from the server (untranslated)
+      var ch=this.responseDiv.childNodes;
+      for (var i=ch.length-1; i>=0; i--) {
+        if (ch[i].nodeType==1 && ch[i].nodeName!='P' && ch[i].nodeName!='DIV' && ch[i].nodeName!='BR')
+          this.responseDiv.removeChild(ch[i]);
+      }
+      responseText=Rico.stripTags(this.responseDiv.innerHTML);
+      success=(responseText.toLowerCase().indexOf('error')==-1);
+    }
+    if (success && this.options.showSaveMsg!='full') {
+      this.hideResponse('');
+      this.grid.resetContents();
+      this.grid.buffer.foundRowCount = false;
+      this.grid.buffer.fetch(this.grid.lastRowPos || 0);
+      if (this.saveMsg) this.saveMsg.innerHTML='&nbsp;'+responseText+'&nbsp;';
+    }
+    this.processCallback(this.options.onSubmitResponse);
+    Rico.log('Processing response completed');
+  },
+
+  processCallback: function(callback) {
+    switch (typeof callback) {
+      case 'string': return eval(callback);
+      case 'function': return callback();
+    }
+  },
+
+  // called when ok pressed on error response message
+  ackResponse: function(e) {
+    this.hideResponse('');
+    this.grid.highlightEnabled=true;
+  },
+
+  cloneRecord: function() {
+    this.formPopup.setTitle(this.cloneText);
+    this.displayEditForm("ins");
+  },
+
+  editRecord: function() {
+    this.formPopup.setTitle(this.editText);
+    this.displayEditForm("upd");
+  },
+
+  displayEditForm: function(action) {
+    this.grid.highlightEnabled=false;
+    this.menu.cancelmenu();
+    this.hideResponse(Rico.getPhraseById('saving'));
+    this.grid.outerDiv.style.cursor = 'auto';
+    this.action.value=action;
+    for (var i=0; i<this.grid.columns.length; i++) {
+      var c=this.grid.columns[i];
+      if (c.format) {
+        var v=c.getValue(this.rowIdx);
+        this.setField(i,v);
+        if (c.format.selectDesc) {
+          if (c.format.EntryType.length>1 && c.format.EntryType.charAt(1)=='L')
+            v=this.grid.columns[i+1].getValue(this.rowIdx);
+          v=c._format(v);
+          if (v==='') v='&nbsp;';
+          c.format.selectDesc.innerHTML=v;
+        }
+        if (c.format.SelectCtl)
+          Rico.EditControls.displayClrImg(c, !c.format.InsertOnly);
+      }
+    }
+    this.setReadOnly(action);
+    for (var k=0; k<this.keys.length; k++) {
+      this.keys[k].keyField.value = this.grid.buffer.getWindowValue(this.rowIdx,this.keys[k].colidx);
+    }
+    this.makeFormVisible(this.rowIdx);
+  },
+  
+  addPrepare: function() {
+    this.hideResponse(Rico.getPhraseById('saving'));
+    this.form.reset();
+    this.setReadOnly("ins");
+    this.action.value="ins";
+    for (var i=0; i<this.grid.columns.length; i++) {
+      var c=this.grid.columns[i];
+      if (c.format) {
+        this.setField(i,c.format.ColData);
+        if (c.format.SelectCtl) {
+          Rico.EditControls.resetValue(c);
+          Rico.EditControls.displayClrImg(c, !c.format.UpdateOnly);
+        }
+      }
+    }
+  },
+
+  addRecord: function() {
+    this.menu.cancelmenu();
+    this.formPopup.setTitle(this.addText);
+    this.addPrepare();
+    this.makeFormVisible(-1);
+    if (this.formPanels) this.formPanels.select(0);
+  },
+
+  drillDown: function(e,masterColNum,detailColNum) {
+    return this.grid.drillDown.apply(this.grid, arguments);
+  },
+
+  // set filter on a detail grid that is in a master-detail relationship
+  setDetailFilter: function(colNumber,filterValue) {
+    this.grid.setDetailFilter(colNumber,filterValue);
+  },
+
+  makeFormVisible: function(row) {
+    this.formPopup.container.style.display='block';
+
+    // set left position
+    var editWi=this.formPopup.container.offsetWidth;
+    var odOffset=Rico.cumulativeOffset(this.grid.outerDiv);
+    var winWi=Rico.windowWidth();
+    this.formPopup.container.style.left=editWi+odOffset.left > winWi ? (winWi-editWi)+'px' : (odOffset.left+1)+'px';
+
+    // set top position
+    var scrTop=Rico.docScrollTop();
+    var editHt=this.formPopup.container.offsetHeight;
+    var newTop=odOffset.top+this.grid.hdrHt+scrTop;
+    var bottom=Rico.windowHeight()+scrTop;
+    if (row >= 0) {
+      newTop+=(row+1)*this.grid.rowHeight;
+      if (newTop+editHt>bottom) newTop-=(editHt+this.grid.rowHeight);
+    } else {
+      if (newTop+editHt>bottom) newTop=bottom-editHt-2;
+    }
+
+    if (this.processCallback(this.options.formOpen) === false) return;
+    this.formPopup.openPopup(null,Math.max(newTop,scrTop));
+    this.formPopup.container.style.visibility='visible';
+    Rico.EditControls.setZ(Rico.getStyle(this.formPopup.container,'zIndex'));
+    if (this.initialized) return;
+
+    var i, spec;
+    for (i = 0; i < this.grid.columns.length; i++) {
+      spec=this.grid.columns[i].format;
+      if (!spec || !spec.EntryType || !spec.FieldName) continue;
+      switch (spec.EntryType) {
+        case 'tinyMCE':
+          if (typeof tinyMCE!='undefined') tinyMCE.execCommand('mceAddControl', true, spec.FieldName);
+          break;
+      }
+    }
+    this.initialized=true;
+  },
+
+  makeFormInvisible: function() {
+    this.formPopup.container.style.visibility='hidden';
+    this.formPopup.closePopup();
+    this.processCallback(this.options.formClose);
+  },
+
+  getConfirmDesc: function(rowIdx) {
+    return Rico.stripTags(this.grid.cell(rowIdx,this.options.ConfirmDeleteCol).innerHTML).replace('&nbsp;',' '); //.unescapeHTML();
+  },
+
+  deleteRecord: function() {
+    this.menu.cancelmenu();
+    var desc;
+    switch(this.options.ConfirmDeleteCol){
+      case -1 :
+        desc=Rico.getPhraseById("thisRecord",this.options.RecordName);
+        break;
+      case -2 : // Use key/column header to identify the row
+        for (var k=0; k<this.keys.length; k++) {
+          var i=this.keys[k].colidx;
+          var fmt=this.grid.columns[i].format;
+          if (fmt.EntryType.length>1 && fmt.EntryType.charAt(1)=='L') i++;
+          var value=Rico.stripTags(this.grid.cell(rowIdx,i).innerHTML).replace(/&nbsp;/g,' ');
+          if (desc) desc+=', ';
+          desc+=this.grid.columns[i].displayName+" "+value;
+        }
+        break;
+      default   :
+        desc='\"' + Rico.truncate(this.getConfirmDesc(this.rowIdx),50) + '\"';
+        break;
+    }
+    if (!this.options.ConfirmDelete.valueOf || confirm(Rico.getPhraseById("confirmDelete",desc))) {
+      this.hideResponse(Rico.getPhraseById('deleting'));
+      this.showResponse();
+      var parms={};
+      parms[this.grid.actionId]="del";
+      for (var k=0; k<this.keys.length; k++) {
+        var i=this.keys[k].colidx;
+        var value=this.grid.columns[i].getValue(this.rowIdx);
+        parms['_k'+i]=value;  // prototype does the encoding automatically
+        //parms['_k'+i]=encodeURIComponent(value);
+      }
+      new Rico.ajaxRequest(this.options.updateURL, {parameters:parms,method:'post',onComplete:this.responseHandler});
+    }
+    this.menu.cancelmenu();
+  },
+
+  validationMsg: function(elem,colnum,phraseId) {
+    var col=this.grid.columns[colnum];
+    if (this.formPanels) this.formPanels.select(this.panelActualIdx[col.format.panelIdx]);
+    var label=Rico.stripTags(col.formLabel.innerHTML).replace(/&nbsp;/g,' ');
+    var msg=Rico.getPhraseById(phraseId," \"" + label + "\"");
+    Rico.log(' Validation error: '+msg);
+    if (col.format.Help) msg+="\n\n"+col.format.Help;
+    alert(msg);
+    setTimeout(function() { try { elem.focus(); elem.select(); } catch(e) {}; }, 10);
+    return false;
+  },
+  
+  isTextInput: function(elem) {
+    if (!elem) return false;
+    if (elem.tagName.toLowerCase()!='input') return false;
+    if (elem.type.toLowerCase()!='text') return false;
+    if (elem.readOnly) return false;
+    if (!Rico.visible(elem)) return false;
+    return true;
+  },
+  
+  parseDate: function(v, dateFmt) {
+    dateParts={};
+    if (!this.dateRegExp.exec(dateFmt)) return NaN;
+    dateParts[RegExp.$1]=0;
+    dateParts[RegExp.$3]=1;
+    dateParts[RegExp.$5]=2;
+    var aDate = v.split(/\D/);
+    var d=new Date();
+    var curyr=d.getFullYear();
+    if (aDate.length==2 && dateParts.yyyy==2) aDate.push(curyr);
+    if (aDate.length!=3) return NaN;
+    var dd=parseInt(aDate[dateParts.dd], 10);
+    if (dd==0 || dd>31) return NaN;
+    var mm=parseInt(aDate[dateParts.mm], 10) - 1;
+    if (mm > 11) return NaN;
+    var yy=parseInt(aDate[dateParts.yyyy], 10);
+    if (yy < 100) {
+      // apply a century to 2-digit years
+      yy+=curyr - (curyr % 100);
+    }
+    d.setFullYear(yy,mm,dd);
+    return d;
+  },
+
+  TESubmit: function(e) {
+    var i,ro,lbl,spec,elem,n,dateValues=[];
+
+    Rico.eventStop(e);
+    Rico.log('Event: TESubmit called to validate input');
+
+    // check fields that are supposed to be non-blank
+
+    for (i = 0; i < this.grid.columns.length; i++) {
+      spec=this.grid.columns[i].format;
+      if (!spec || !spec.EntryType || !spec.FieldName) continue;
+      elem=Rico.$(spec.FieldName);
+      if (!this.isTextInput(elem)) continue;
+      switch (this.action.value) {
+        case 'ins': ro=!spec.Writeable || spec.ReadOnly || spec.UpdateOnly; break;
+        case 'upd': ro=!spec.Writeable || spec.ReadOnly || spec.InsertOnly; break;
+        default:    ro=false; break;
+      }
+      if (ro) continue;  // readonly, so don't validate
+      Rico.log(' Validating field #'+i+' EntryType='+spec.EntryType+' ('+spec.FieldName+')');
+
+      // check for blanks
+      if (elem.value.length == 0) {
+        if (spec.required)
+          return this.validationMsg(elem,i,"formPleaseEnter");
+        else
+          continue;
+      }
+
+      // check pattern
+      if (elem.value.length > 0 && spec.regexp && !spec.regexp.test(elem.value))
+        return this.validationMsg(elem,i,"formInvalidFmt");
+
+      // check min/max and date values
+      switch (spec.EntryType.charAt(0)) {
+        case 'I': n=parseInt(elem.value,10); break;
+        case 'F': n=parseFloat(elem.value); break;
+        case 'D': 
+          n=this.parseDate(elem.value,spec.dateFmt);
+          if (isNaN(n)) return this.validationMsg(elem,i,"formInvalidFmt");
+          dateValues.push({e:elem,v:n});
+          break;
+        default:  n=NaN; break;
+      }
+      if (typeof spec.min!='undefined' && !isNaN(n) && n < spec.min)
+        return this.validationMsg(elem,i,"formOutOfRange");
+      if (typeof spec.max!='undefined' && !isNaN(n) && n > spec.max)
+        return this.validationMsg(elem,i,"formOutOfRange");
+    }
+    if (this.processCallback(this.options.formSubmit) === false) return false;
+
+    // update drop-down for any columns with entry type of N
+
+    for (i = 0; i < this.grid.columns.length; i++) {
+      spec=this.grid.columns[i].format;
+      if (!spec || !spec.EntryType || !spec.FieldName) continue;
+      if (spec.EntryType.charAt(0) != 'N') continue;
+      var SelObj=Rico.$(spec.FieldName);
+      if (!SelObj || SelObj.value!=this.options.TableSelectNew) continue;
+      var newtext=Rico.$("textnew__" + SelObj.id).value;
+      this.addSelectOption(SelObj,newtext,newtext);
+    }
+    
+    // set date values to ISO format
+    for (i = 0; i < dateValues.length; i++) {
+      dateValues[i].e.value = Rico.formatDate(dateValues[i].v,'yyyy-mm-dd');
+    }
+
+    if (typeof tinyMCE!='undefined') tinyMCE.triggerSave();
+    this.makeFormInvisible();
+    this.sendForm();
+    this.menu.cancelmenu();
+    return false;
+  },
+  
+  sendForm: function() {
+    this.setReadOnly("reset");  // reset disabled flag so that all fields are sent to server
+    this.showResponse();
+    Rico.log("sendForm: "+this.grid.tableId);
+    Rico.ajaxSubmit(this.form, this.options.updateURL, {method:'post',onComplete:this.responseHandler});
+  }
+};
+
+
+/**
+ * @namespace Registers custom popup widgets to fill in a text box (e.g. ricoCalendar and ricoTree)
+ * <pre>
+ * Custom widget must implement:
+ *   open() method (make control visible)
+ *   close() method (hide control)
+ *   container property (div element that contains the control)
+ *   id property (uniquely identifies the widget class)
+ *
+ * widget calls returnValue method to return a value to the caller
+ *
+ * this object handles clicks on the control's icon and positions the control appropriately.
+ * </pre>
+ */
+Rico.EditControls = {
+  widgetList : {},
+  elemList   : {},
+  zIndex     : 0,
+
+  register: function(widget, imgsrc) {
+    this.widgetList[widget.id] = {imgsrc:imgsrc, widget:widget, currentEl:''};
+    var self=this;
+    widget.returnValue=function(newVal,newDesc) { self.setValue(widget,newVal,newDesc); };
+    Rico.log("Rico.EditControls.register:"+widget.id);
+  },
+  
+  setZ: function(z) {
+    this.zIndex=Math.max(this.zIndex,z+10);
+  },
+
+  atLoad: function() {
+    for (var id in this.widgetList) {
+      var widget=this.widgetList[id].widget;
+      if (widget.atLoad && !widget.atLoadComplete) {
+        Rico.log("Rico.EditControls.atLoad: "+id);
+        widget.atLoad();
+        widget.atLoadComplete=true;
+      }
+    }
+  },
+
+  applyTo: function(column,inputCtl) {
+    var wInfo=this.widgetList[column.format.SelectCtl];
+    if (!wInfo) return;
+    Rico.log('Rico.EditControls.applyTo: '+column.displayName+' : '+column.format.SelectCtl);
+    var newimg, descSpan = document.createElement('span');
+    if (wInfo.imgsrc.indexOf('.')==-1 && wInfo.imgsrc.indexOf('/')==-1) {
+      // treat imgsrc as a class name
+      newimg = document.createElement('span');
+      newimg.className=wInfo.imgsrc;
+    } else {
+      // treat imgsrc as an image uri
+      newimg = document.createElement('img');
+      newimg.src=wInfo.imgsrc;
+    }
+    newimg.style.verticalAlign='top';
+    newimg.style.marginLeft='4px';
+    newimg.style.cursor='pointer';
+    newimg.id=this.imgId(column.format.FieldName);
+    Rico.eventBind(newimg,"click", Rico.eventHandle(this,'processClick'));
+    inputCtl.parentNode.appendChild(descSpan);
+    inputCtl.parentNode.appendChild(newimg);
+    inputCtl.style.display='none';    // comment out this line for debugging
+    var clr;
+    if (column.format.isNullable) {
+      clr=Rico.clearButton(Rico.eventHandle(this,'processClear'));
+      clr.id=newimg.id+'_clear';
+      inputCtl.parentNode.appendChild(clr);
+    }
+    this.elemList[newimg.id] = {descSpan:descSpan, inputCtl:inputCtl, widget:wInfo.widget, listObj:wInfo, column:column, clrimg:clr};
+    column.format.selectIcon=newimg;
+    column.format.selectDesc=descSpan;
+  },
+
+  displayClrImg: function(column,bShow) {
+    var el=this.elemList[this.imgId(column.format.FieldName)];
+    //alert(column.format.FieldName+': '+bShow+' '+el.clrimg.id);
+    if (el && el.clrimg) el.clrimg.style.display=bShow ? 'inline-block' : 'none';
+  },
+
+  processClear: function(e) {
+    var elem=Rico.eventElement(e);
+    var el=this.elemList[elem.id.slice(0,-6)];
+    if (!el) return;
+    el.inputCtl.value='';
+    el.descSpan.innerHTML=el.column._format('');
+  },
+
+  processClick: function(e) {
+    var elem=Rico.eventElement(e);
+    var el=this.elemList[elem.id];
+    if (!el) return;
+    if (el.listObj.currentEl==elem.id && el.widget.container.style.display!='none') {
+      el.widget.close();
+      el.listObj.currentEl='';
+    } else {
+      el.listObj.currentEl=elem.id;
+      Rico.log('Rico.EditControls.processClick: '+el.widget.id+' : '+el.inputCtl.value);
+      el.widget.container.style.zIndex=this.zIndex;
+      el.widget.open(el.inputCtl.value,el.column);     // this may change the size of the widget
+      Rico.positionCtlOverIcon(el.widget.container,elem);
+    }
+  },
+
+  imgId: function(fieldname) {
+    return 'icon_'+fieldname;
+  },
+
+  resetValue: function(column) {
+    var el=this.elemList[this.imgId(column.format.FieldName)];
+    if (!el) return;
+    el.inputCtl.value=column.format.ColData;
+    var v=column._format(column.format.ColData);
+    if (v==='') v='&nbsp;';
+    el.descSpan.innerHTML=v;
+  },
+
+  setValue: function(widget,newVal,newDesc) {
+    var wInfo=this.widgetList[widget.id];
+    if (!wInfo) return null;
+    var id=wInfo.currentEl;
+    if (!id) return null;
+    var el=this.elemList[id];
+    if (!el) return null;
+    el.inputCtl.value=newVal;
+    if (!newDesc) newDesc=el.column._format(newVal);
+    el.descSpan.innerHTML=newDesc;
+    if (el.column.format.DescriptionField)
+      Rico.$(el.column.format.DescriptionField).value = newDesc;
+    //alert(widget.id+':'+id+':'+el.inputCtl.id+':'+el.inputCtl.value+':'+newDesc);
+  },
+
+  close: function(id) {
+    var wInfo=this.widgetList[id];
+    if (!wInfo) return;
+    if (wInfo.widget.container.style.display!='none')
+      wInfo.widget.close();
+  }
+};
diff --git a/minsrc/ricoLiveGridMenu.js b/minsrc/ricoLiveGridMenu.js
new file mode 100644 (file)
index 0000000..bb85181
--- /dev/null
@@ -0,0 +1,136 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+ if(typeof Rico=='undefined')
+  throw("GridMenu requires the Rico JavaScript framework");
+
+/**
+ * Standard menu for LiveGrid
+ */
+Rico.GridMenu = function(options) {
+  this.initialize(options);
+};
+
+Rico.GridMenu.prototype = {
+
+initialize: function(options) {
+  this.options = {
+    width           : '18em',
+    dataMenuHandler : null          // put custom items on the menu
+  };
+  Rico.extend(this.options, options || {});
+  Rico.extend(this, new Rico.Menu(this.options));
+  this.sortmenu = new Rico.Menu({ width: '15em' });
+  this.filtermenu = new Rico.Menu({ width: '22em' });
+  this.exportmenu = new Rico.Menu({ width: '24em' });
+  this.hideshowmenu = new Rico.Menu({ width: '22em' });
+  this.createDiv();
+  this.sortmenu.createDiv();
+  this.filtermenu.createDiv();
+  this.exportmenu.createDiv();
+  this.hideshowmenu.createDiv();
+},
+
+// Build context menu for grid
+buildGridMenu: function(r,c) {
+  this.clearMenu();
+  var livegrid=this.liveGrid;
+  var buffer=livegrid.buffer;
+  var totrows=buffer.totalRows;
+  var maxprint=livegrid.options.maxPrint;
+  var onBlankRow=(r >= totrows);
+  var column=livegrid.columns[c];
+  if (this.options.dataMenuHandler) {
+     var showDefaultMenu=this.options.dataMenuHandler(livegrid,r,c,onBlankRow);
+     if (!showDefaultMenu) return (this.itemCount > 0);
+  }
+
+  // menu items for sorting
+  if (column.sortable && totrows>0) {
+    this.sortmenu.clearMenu();
+    this.addSubMenuItem(Rico.getPhraseById("gridmenuSortBy",column.displayName), this.sortmenu, false);
+    this.sortmenu.addMenuItemId("gridmenuSortAsc", function() { column.sortAsc(); }, true);
+    this.sortmenu.addMenuItemId("gridmenuSortDesc", function() { column.sortDesc(); }, true);
+  }
+
+  // menu items for filtering
+  this.filtermenu.clearMenu();
+  if (column.canFilter() && !column.format.filterUI && (!onBlankRow || column.filterType == Rico.ColumnConst.USERFILTER)) {
+    this.addSubMenuItem(Rico.getPhraseById("gridmenuFilterBy",column.displayName), this.filtermenu, false);
+    column.userFilter=column.getValue(r);
+    if (column.filterType == Rico.ColumnConst.USERFILTER) {
+      this.filtermenu.addMenuItemId("gridmenuRemoveFilter", function() { column.setUnfiltered(false); }, true);
+      if (column.filterOp=='LIKE')
+        this.filtermenu.addMenuItemId("gridmenuChgKeyword", function() { livegrid.openKeyword(c); }, true);
+      if (column.filterOp=='NE' && !onBlankRow)
+        this.filtermenu.addMenuItemId("gridmenuExcludeAlso", function() { column.addFilterNE(); }, true);
+    } else if (!onBlankRow) {
+      this.filtermenu.addMenuItemId("gridmenuInclude", function() { column.setFilterEQ(); }, true);
+      this.filtermenu.addMenuItemId("gridmenuGreaterThan", function() { column.setFilterGE(); }, column.userFilter!='');
+      this.filtermenu.addMenuItemId("gridmenuLessThan", function() { column.setFilterLE(); }, column.userFilter!='');
+      if (column.isText)
+        this.filtermenu.addMenuItemId("gridmenuContains", function() { livegrid.openKeyword(c); }, true);
+      this.filtermenu.addMenuItemId("gridmenuExclude", function() { column.setFilterNE(); }, true);
+    }
+    if (livegrid.filterCount() > 0) {
+      this.filtermenu.addMenuItemId("gridmenuRefresh", function() { livegrid.filterHandler(); }, true);
+      this.filtermenu.addMenuItemId("gridmenuRemoveAll", function() { livegrid.clearFilters(); }, true);
+    }
+  } else if (livegrid.filterCount() > 0) {
+    this.addSubMenuItem(Rico.getPhraseById("gridmenuFilterBy",column.displayName), this.filtermenu, false);
+    this.filtermenu.addMenuItemId("gridmenuRemoveAll", function() { livegrid.clearFilters(); }, true);
+  }
+
+  // menu items for Print/Export
+  this.exportmenu.clearMenu();
+  if (buffer.printVisibleSQL && typeof(buffer.dataSource)=='string') {
+    // SQL buffer
+    this.addSubMenuItem(Rico.getPhraseById('gridmenuExport'),this.exportmenu,false);
+    this.exportmenu.addMenuItemId("gridmenuExportVis2Web", function() { buffer.printVisibleSQL('html'); });
+    this.exportmenu.addMenuItemId("gridmenuExportAll2Web", function() { buffer.printAllSQL('html'); }, buffer.totalRows <= maxprint);
+    this.exportmenu.addMenuBreak();
+    this.exportmenu.addMenuItemId("gridmenuExportVis2SS", function() { buffer.printVisibleSQL('xl'); });
+    this.exportmenu.addMenuItemId("gridmenuExportAll2SS", function() { buffer.printAllSQL('xl'); }, buffer.totalRows <= maxprint);
+  } else if (maxprint > 0 && totrows>0) {
+    // any other buffer
+    this.addSubMenuItem(Rico.getPhraseById('gridmenuExport'),this.exportmenu,false);
+    this.exportmenu.addMenuItemId("gridmenuExportVis2Web", function() { buffer.printVisible('plain'); });
+    this.exportmenu.addMenuItemId("gridmenuExportAll2Web", function() { buffer.printAll('plain'); }, buffer.totalRows <= maxprint);
+    if (Rico.isIE) {
+      this.exportmenu.addMenuBreak();
+      this.exportmenu.addMenuItemId("gridmenuExportVis2SS", function() { buffer.printVisible('owc'); });
+      this.exportmenu.addMenuItemId("gridmenuExportAll2SS", function() { buffer.printAll('owc'); }, buffer.totalRows <= maxprint);
+    }
+  }
+
+  // menu items for hide/unhide
+  var hiddenCols=livegrid.listInvisible();
+  for (var showableCnt=0,x=0; x<hiddenCols.length; x++) {
+    if (hiddenCols[x].canHideShow()) showableCnt++;
+  }
+  if (showableCnt > 0 || column.canHideShow()) {
+    this.hideshowmenu.clearMenu();
+    this.addSubMenuItem(Rico.getPhraseById('gridmenuHideShow'),this.hideshowmenu,false);
+    this.hideshowmenu.addMenuItemId('gridmenuChooseCols', function() { livegrid.chooseColumns(); },true,false);
+    var visibleCnt=livegrid.columns.length-hiddenCols.length;
+    var enabled=(visibleCnt>1 && column.visible && column.canHideShow());
+    this.hideshowmenu.addMenuItem(Rico.getPhraseById('gridmenuHide',column.displayName), function() { column.hideColumn(); }, enabled);
+    if (hiddenCols.length > 1)
+      this.hideshowmenu.addMenuItemId('gridmenuShowAll', function() { livegrid.showAll(); });
+  }
+  return true;
+}
+
+}
diff --git a/minsrc/ricoSearch.js b/minsrc/ricoSearch.js
new file mode 100644 (file)
index 0000000..0f9e440
--- /dev/null
@@ -0,0 +1,134 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+Rico.KeywordSearch = function(id,options) {
+  this.initialize(id,options);
+};
+
+Rico.KeywordSearch.prototype = {
+/**
+ * @class Implements a pop-up keyword search control.
+ * @extends Rico.Popup
+ * @constructs
+ * @param id unique identifier
+ * @param options object may contain any of the following:<dl>
+ *   <dt>showColorCode</dt><dd> show hex color code as user hovers over color grid? default=false</dd>
+ *   <dt>cellsPerRow  </dt><dd> number of colors per row in the grid? default=18</dd>
+ *   <dt>palette      </dt><dd> array of 6 digit hex values, default=216 "web safe" colors</dd>
+ *</dl>
+ */
+  initialize: function(id,options) {
+    this.id=id;
+    Rico.extend(this, new Rico.Window(Rico.getPhraseById("keywordTitle"),options));
+    Rico.addClass(this.content,'ricoKeywordSearch');
+    Rico.extend(this.options, {
+      listLength : 10,
+      maxSuggest : 20,
+      width: '12em'
+    });
+  },
+
+  atLoad : function() {
+    this.searchField=Rico.createFormField(this.contentDiv,'input','text',this.id+'_search');
+    this.searchField.style.display="block";
+    this.searchField.style.width=this.options.width;
+    Rico.eventBind(this.searchField,'keyup',Rico.eventHandle(this,'filterKeypress'),false);
+    this.selectList=Rico.createFormField(this.contentDiv,'select',null,this.id+'_list');
+    this.selectList.size=this.options.listLength;
+    this.selectList.style.display="block";
+    this.selectList.style.width=this.options.width;
+    Rico.eventBind(this.selectList,'change',Rico.eventHandle(this,'listClick'),false);
+    /**
+     * alias for closePopup
+     * @function
+     */
+    this.close=this.closePopup;
+    this.close();
+  },
+  
+  open: function(currentVal,column) {
+    this.column=column;
+    this.grid=this.column.liveGrid;
+    this.searchField.value='';
+    this.selectList.options.length=0;
+    this.openPopup();
+    this.searchField.focus();
+    this.selectValuesRequest('');
+  },
+
+  selectValuesRequest: function(filter) {
+    var colnum=this.column.index;
+    var options={};
+    Rico.extend(options, this.grid.buffer.ajaxOptions);
+    options.parameters = {id: this.grid.tableId, offset: '0', page_size: this.options.maxSuggest, edit: colnum};
+    options.parameters[this.grid.actionId]="query";
+    if (filter!='' && filter!='*') {
+      if (filter.indexOf('*')==-1) filter='*'+filter+'*';
+      options.parameters['f[1][op]']="LIKE";
+      options.parameters['f[1][len]']=1;
+      options.parameters['f[1][0]']=filter;
+    }
+    var self=this;
+    options.onComplete = function(request) { self.selectValuesUpdate(request); };
+    new Rico.ajaxRequest(this.grid.buffer.dataSource, options);
+  },
+
+  selectValuesUpdate: function(request) {
+    var response = request.responseXML.getElementsByTagName("ajax-response");
+    Rico.log("selectValuesUpdate: "+request.status);
+    if (response == null || response.length != 1) return;
+    response=response[0];
+    var error = response.getElementsByTagName('error');
+    if (error.length > 0) {
+      var errmsg=Rico.getContentAsString(error[0],this.grid.buffer.isEncoded);
+      Rico.log("Data provider returned an error:\n"+errmsg);
+      alert(Rico.getPhraseById("requestError",errmsg));
+      return null;
+    }
+    this.selectList.options.length=0;
+    response=response.getElementsByTagName('response')[0];
+    var rowsElement = response.getElementsByTagName('rows')[0];
+    var rows = this.grid.buffer.dom2jstable(rowsElement);
+    Rico.log("selectValuesUpdate: id="+this.selectList.id+' rows='+rows.length);
+    for (var i=0; i<rows.length; i++) {
+      if (rows[i].length>0) {
+        var c0=rows[i][0];
+        var c1=(rows[i].length>1) ? rows[i][1] : c0;
+        Rico.addSelectOption(this.selectList,c0,c1);
+      }
+    }
+  },
+
+  filterKeypress: function(e) {
+    var txtbox=Rico.eventElement(e);
+    if (typeof this.lastKeyFilter != 'string') this.lastKeyFilter='';
+    if (this.lastKeyFilter==txtbox.value) return;
+    var v=txtbox.value;
+    Rico.log("filterKeypress: "+this.index+' '+v);
+    this.lastKeyFilter=v;
+    this.selectValuesRequest(v);
+  },
+  
+  listClick: function(e) {
+    var elem=Rico.eventElement(e);
+    if (elem.tagName.toLowerCase() != 'select') return;
+    if (this.returnValue) {
+      var opt=elem.options[elem.selectedIndex];
+      this.returnValue(opt.value,opt.innerHTML);
+    }
+    this.close();
+  }
+
+};
diff --git a/minsrc/ricoSimpleGrid.js b/minsrc/ricoSimpleGrid.js
new file mode 100644 (file)
index 0000000..170c942
--- /dev/null
@@ -0,0 +1,401 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+if(typeof Rico=='undefined') throw("SimpleGrid requires the Rico JavaScript framework");
+
+Rico.SimpleGrid = function(tableId, options) {
+  this.initialize(tableId, options);
+}
+
+Rico.SimpleGrid.prototype = {
+/**
+ * @class Create & manage an unbuffered grid.
+ * Supports: frozen columns & headings, resizable columns.
+ * @extends Rico.GridCommon
+ * @constructs
+ */
+  initialize: function( tableId, options ) {
+    Rico.extend(this, Rico.GridCommon);
+    this.baseInit();
+    Rico.setDebugArea(tableId+"_debugmsgs");    // if used, this should be a textarea
+    Rico.extend(this.options, options || {});
+    this.tableId = tableId;
+    Rico.log("SimpleGrid initialize start: "+tableId);\r
+    this.createDivs();
+    this.hdrTabs=new Array(2);
+    this.simpleGridInit();
+    Rico.log("SimpleGrid initialize end: "+tableId);\r
+  },
+
+  simpleGridInit: function() {
+    var i;
+    for (i=0; i<2; i++) {
+      Rico.log("simpleGridInit "+i);\r
+      this.tabs[i]=document.getElementById(this.tableId+'_tab'+i);
+      if (!this.tabs[i]) return;
+      this.hdrTabs[i]=document.getElementById(this.tableId+'_tab'+i+'h');
+      if (!this.hdrTabs[i]) return;
+      //if (i==0) this.tabs[i].style.position='absolute';
+      //if (i==0) this.tabs[i].style.left='0px';
+      //this.hdrTabs[i].style.position='absolute';
+      //this.hdrTabs[i].style.top='0px';
+      //this.hdrTabs[i].style.zIndex=1;
+      this.thead[i]=this.hdrTabs[i];
+      this.tbody[i]=this.tabs[i];
+      this.headerColCnt = this.getColumnInfo(this.hdrTabs[i].rows);
+      if (i==0) this.options.frozenColumns=this.headerColCnt;
+      if (Rico.theme.gridheader) Rico.addClass(this.thead[i],Rico.theme.gridheader);
+      if (Rico.theme.gridcontent) Rico.addClass(this.tbody[i],Rico.theme.gridcontent);
+    }
+    if (this.headerColCnt==0) {
+      alert('ERROR: no columns found in "'+this.tableId+'"');
+      return;
+    }
+    //this.hdrHt=Math.max(Rico.nan2zero(this.hdrTabs[0].offsetHeight),this.hdrTabs[1].offsetHeight);
+    //for (i=0; i<2; i++) {
+    //  if (i==0) this.tabs[i].style.top=this.hdrHt+'px';
+    //}
+    this.createColumnArray('SimpleGridColumn');
+    this.pageSize=this.columns[0].dataColDiv.childNodes.length;
+    this.sizeDivs();
+    if (typeof(this.options.FilterLocation)=='number')
+      this.createFilters(this.options.FilterLocation);
+    this.attachMenuEvents();
+    this.scrollEventFunc=Rico.eventHandle(this,'handleScroll');
+    this.pluginScroll();
+    if (this.options.windowResize)
+      Rico.eventBind(window,"resize", Rico.eventHandle(this,'sizeDivs'), false);
+  },
+
+  // return id string for a filter element
+  filterId: function(colnum) {
+    return 'RicoFilter_'+this.tableId+'_'+colnum;
+  },
+  
+  // create filter elements on heading row r
+  createFilters: function(r) {
+    if (r < 0) {
+      r=this.addHeadingRow();
+      this.sizeDivs();
+    }
+    for( var c=0; c < this.headerColCnt; c++ ) {
+      var col=this.columns[c];
+      var fmt=col.format;
+      if (typeof fmt.filterUI!='string') continue;
+      var cell=this.hdrCells[r][c].cell;
+      var field,name=this.filterId(c);\r
+      var divs=cell.getElementsByTagName('div');
+      switch (fmt.filterUI.charAt(0)) {
+        case 't':
+          field=Rico.createFormField(divs[1],'input','text',name,name);
+          var size=fmt.filterUI.match(/\d+/);
+          field.maxLength=fmt.Length || 50;\r
+          field.size=size ? parseInt(size,10) : 10;
+          Rico.eventBind(field,'keyup',Rico.eventHandle(col,'filterKeypress'),false);\r
+          break;\r
+        case 's':
+          field=Rico.createFormField(divs[1],'select',null,name);\r
+          Rico.addSelectOption(field,this.options.FilterAllToken,Rico.getPhraseById("filterAll"));
+          this.getFilterValues(col);
+          var keys=Rico.keys(col.filterHash);
+          keys.sort();
+          for (var i=0; i<keys.length; i++)
+            Rico.addSelectOption(field,keys[i],keys[i] || Rico.getPhraseById("filterBlank"));\r
+          Rico.eventBind(field,'change',Rico.eventHandle(col,'filterChange'),false);\r
+          break;\r
+      }
+    }
+    this.initFilterImage(r);
+  },
+  
+  getFilterValues: function(col) {
+    var h={};
+    var n=col.numRows();
+    for (var i=0; i<n; i++) {
+      var v=Rico.getInnerText(col.cell(i));
+      var hval=h[v];
+      if (hval)
+        hval.push(i);
+      else
+        h[v]=[i];
+    }
+    col.filterHash=h;\r
+  },
+  
+  // hide filtered rows
+  applyFilters: function() {
+    // rows to display will be the intersection of all filterRows arrays
+    var fcols=[];
+    for (var c=0; c<this.columns.length; c++) {
+      if (this.columns[c].filterRows)
+        fcols.push(this.columns[c].filterRows);
+    }
+    if (fcols.length==0) {
+      // no filters are set
+      this.showAllRows();
+      return;
+    }
+    for (var r=0; r<this.pageSize; r++) {
+      var showflag=true;
+      for (var j=0; j<fcols.length; j++) {
+        if (fcols[j].indexOf(r)==-1) {
+          showflag=false;
+          break;
+        }
+      }
+      if (showflag)
+        this.showRow(r);
+      else
+        this.hideRow(r);
+    }
+    this.sizeDivs();\r
+  },
+
+  handleScroll: function(e) {
+    var newTop=(-this.scrollDiv.scrollTop)+'px';
+    this.tabs[0].style.marginTop=newTop;
+    this.setHorizontalScroll();
+  },
+
+  /**
+   * Register a menu that will only be used in the scrolling part of the grid.
+   * If submenus are used, they must be registered after the main menu.
+   */
+  registerScrollMenu: function(menu) {
+    if (!this.menu) this.menu=menu;
+    menu.grid=this;
+    menu.showmenu=menu.showSimpleMenu;
+    menu.showSubMenu=menu.showSimpleSubMenu;
+    menu.createDiv(this.outerDiv);
+  },
+
+  handleMenuClick: function(e) {
+    if (!this.menu) return;
+    this.cancelMenu();
+    this.menuCell=Rico.getParentByTagName(Rico.eventElement(e),'div');
+    this.highlightEnabled=false;
+    if (this.hideScroll) this.scrollDiv.style.overflow="hidden";
+    if (this.menu.buildGridMenu) this.menu.buildGridMenu(this.menuCell);
+    this.menu.showmenu(e,this.closeMenu.bind(this));
+  },
+
+  closeMenu: function() {
+    if (this.hideScroll) this.scrollDiv.style.overflow="";
+    this.highlightEnabled=true;
+  },
+
+  sizeDivs: function() {
+    if (this.outerDiv.offsetParent.style.display=='none') return;
+    this.baseSizeDivs();
+    var maxHt=Math.max(this.options.maxHt || this.availHt(), 50);
+    var totHt=Math.min(this.hdrHt+this.dataHt, maxHt);
+    Rico.log('sizeDivs '+this.tableId+': hdrHt='+this.hdrHt+' dataHt='+this.dataHt);
+    this.dataHt=totHt-this.hdrHt;
+    if (this.scrWi>0) this.dataHt+=this.options.scrollBarWidth;
+    this.scrollDiv.style.height=this.dataHt+'px';
+    this.frozenTabs.style.height=this.scrollDiv.clientHeight+'px';
+    var divAdjust=2;
+    this.innerDiv.style.width=(this.scrWi-this.options.scrollBarWidth+divAdjust)+'px';
+    //this.innerDiv.style.height=(this.hdrHt+1)+'px';
+    totHt+=divAdjust;
+    this.resizeDiv.style.height=totHt+'px';
+    
+    //this.outerDiv.style.height=(totHt+this.options.scrollBarWidth)+'px';
+    this.handleScroll();
+  },
+  
+/**
+ * Copies all rows to a new window as a simple html table.
+ */
+  printVisible: function(exportType) {
+    this.showMsg(Rico.getPhraseById('exportInProgress'));
+    Rico.runLater(10,this,'_printVisible',exportType);  // allow message to paint
+  },
+
+  _printVisible: function(exportType) {
+    this.exportStart();
+    var exportStyles=this.getExportStyles(this.tbody[0]);
+    for(var r=0; r < this.pageSize; r++) {
+      if (this.columns[0].cell(r).style.display=='none') continue;
+      var exportText='';
+      for (var c=0; c<this.columns.length; c++) {
+        var col=this.columns[c];
+        if (col.visible) {
+          var v=col.getFormattedValue(r, !this.options.exportImgTags, !this.options.exportFormFields, 'NoExport');
+          if (col.format.exportPrefix) v=col.format.exportPrefix+v;
+          if (v=='') v='&nbsp;';
+          exportText+="<td style='"+this.exportStyle(col.cell(r),exportStyles)+"'>"+v+"</td>";
+        }
+      }
+      this.exportRows.push(exportText);
+    }
+    this.exportFinish(exportType);
+  },
+
+  /**
+   * Hide a row in the grid.
+   * sizeDivs() should be called after this function has completed.
+   */
+  hideRow: function(rownum) {
+    if (this.columns[0].cell(rownum).style.display=='none') return;
+    for (var i=0; i<this.columns.length; i++)
+      this.columns[i].cell(rownum).style.display='none';
+  },
+
+  /**
+   * Unhide a row in the grid.
+   * sizeDivs() should be called after this function has completed.
+   */
+  showRow: function(rownum) {
+    if (this.columns[0].cell(rownum).style.display=='') return;
+    for (var i=0; i<this.columns.length; i++)
+      this.columns[i].cell(rownum).style.display='';
+  },
+
+  /**
+   * Search for rows that contain SearchString in column ColIdx.
+   * If ShowMatch is false, then matching rows are hidden, if true then mismatching rows are hidden.
+   */
+  searchRows: function(ColIdx,SearchString,ShowMatch) {\r
+    if (!SearchString) return;\r
+    var re=new RegExp(SearchString);\r
+    var rowcnt=this.columns[ColIdx].numRows();\r
+    for(var r=0; r<rowcnt; r++) {\r
+      var txt=this.cell(r,ColIdx).innerHTML;\r
+      var matched=(txt.match(re) != null);\r
+      if (matched != ShowMatch) this.hideRow(r);\r
+    }
+    this.sizeDivs();
+    this.handleScroll();\r
+  },\r
+\r
+  /**
+   * Unhide all rows in the grid
+   */
+  showAllRows: function() {
+    for (var i=0; i<this.pageSize; i++)
+      this.showRow(i);
+    this.sizeDivs();\r
+  },
+  
+  openPopup: function(elem,popupobj) {
+    while (elem && !Rico.hasClass(elem,'ricoLG_cell'))
+      elem=elem.parentNode;
+    if (!elem) return false;
+    var td=Rico.getParentByTagName(elem,'td');
+  
+    var newLeft=Math.floor(td.offsetLeft-this.scrollDiv.scrollLeft+td.offsetWidth/2);
+    if (this.direction == 'rtl') {
+      if (newLeft > this.width) newLeft-=this.width;
+    } else {
+      if (newLeft+this.width+this.options.margin > this.scrollDiv.clientWidth) newLeft-=this.width;
+    }
+    popupobj.divPopup.style.visibility="hidden";
+    popupobj.divPopup.style.display="block";
+    var contentHt=popupobj.divPopup.offsetHeight;
+    var newTop=Math.floor(elem.offsetTop-this.scrollDiv.scrollTop+elem.offsetHeight/2);
+    if (newTop+contentHt+popupobj.options.margin > this.scrollDiv.clientHeight)
+      newTop=Math.max(newTop-contentHt,0);
+    popupobj.openPopup(this.frzWi+newLeft,this.hdrHt+newTop);
+    popupobj.divPopup.style.visibility ="visible";
+    return elem;
+  }
+
+}
+
+if (Rico.Menu) {
+Rico.extend(Rico.Menu.prototype, {
+
+showSimpleMenu: function(e,hideFunc) {
+  Rico.eventStop(e);
+  this.hideFunc=hideFunc;
+  if (this.div.childNodes.length==0) {
+    this.cancelmenu();
+    return false;
+  }
+  var elem=Rico.eventElement(e);
+  this.grid.openPopup(elem,this);
+  return elem;
+},
+
+showSimpleSubMenu: function(a,submenu) {
+  if (this.openSubMenu) this.hideSubMenu();
+  this.openSubMenu=submenu;
+  this.openMenuAnchor=a;
+  if (a.className=='ricoSubMenu') a.className='ricoSubMenuOpen';
+  var top=parseInt(this.div.style.top,10);
+  var left=parseInt(this.div.style.left,10);
+  submenu.openPopup(left+a.offsetWidth,top+a.offsetTop);
+  submenu.div.style.visibility ="visible";
+}
+
+});
+}
+
+
+Rico.SimpleGridColumn = function(grid,colIdx,hdrInfo,tabIdx) {
+  this.initialize(grid,colIdx,hdrInfo,tabIdx);
+}
+
+Rico.SimpleGridColumn.prototype = {
+/**
+ * @class Implements a SimpleGrid column
+ * @extends Rico.TableColumnBase
+ * @constructs
+ */
+initialize: function(grid,colIdx,hdrInfo,tabIdx) {
+  Rico.extend(this, new Rico.TableColumnBase());
+  this.baseInit(grid,colIdx,hdrInfo,tabIdx);
+},
+
+setUnfiltered: function() {
+  this.filterRows=null;
+},
+
+filterChange: function(e) {\r
+  var selbox=Rico.eventElement(e);
+  if (selbox.value==this.liveGrid.options.FilterAllToken)\r
+    this.setUnfiltered();\r
+  else
+    this.filterRows=this.filterHash[selbox.value];
+  this.liveGrid.applyFilters();
+},
+
+filterKeypress: function(e) {\r
+  var txtbox=Rico.eventElement(e);
+  if (typeof this.lastKeyFilter != 'string') this.lastKeyFilter='';\r
+  if (this.lastKeyFilter==txtbox.value) return;\r
+  var v=txtbox.value;\r
+  Rico.log("filterKeypress: "+this.index+' '+v);\r
+  this.lastKeyFilter=v;
+  if (v) {
+    v=v.replace('\\','\\\\');
+    v=v.replace('(','\\(').replace(')','\\)');
+    v=v.replace('.','\\.');
+    if (this.format.filterUI.indexOf('^') > 0) v='^'+v;
+    var re=new RegExp(v,'i');\r
+    this.filterRows=[];
+    var n=this.numRows();
+    for (var i=0; i<n; i++) {
+      var celltxt=Rico.getInnerText(this.cell(i));
+      if (celltxt.match(re)) this.filterRows.push(i);
+    }
+  } else {
+    this.setUnfiltered();\r
+  }
+  this.liveGrid.applyFilters();
+}
+
+}
diff --git a/minsrc/ricoTree.js b/minsrc/ricoTree.js
new file mode 100644 (file)
index 0000000..3ed6b8c
--- /dev/null
@@ -0,0 +1,309 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+//  Rico Tree Control
+//  Requires prototype.js and ricoCommon.js
+//  Data for the tree can be obtained via AJAX requests
+//  Each record in the AJAX response should contain 5 or 6 cells:
+//   cells[0]=parent node id
+//   cells[1]=node id
+//   cells[2]=description
+//   cells[3]=L/zero (leaf), C/non-zero (container)
+//   cells[4]=0->not selectable, 1->selectable (use default action), otherwise the node is selectable and cells[4] contains the action
+//   cells[5]=leafIcon (optional)
+
+
+Rico.TreeControl = function(id,url,options) {
+  this.initialize(id,url,options);
+};
+
+Rico.TreeControl.prototype = {
+/**
+ * @class Implements a pop-up tree control.
+ * @extends Rico.Popup
+ * @constructs
+ * @param id unique identifier
+ * @param url data source
+ * @param options object may contain any of the following:<dl>
+ *   <dt>nodeIdDisplay</dt><dd> first, last, tooltip, or none? default=none</dd>
+ *   <dt>showCheckBox </dt><dd> show checkbox next to each item? default=false</dd>
+ *   <dt>showFolders  </dt><dd> show folder icons? default=false</dd>
+ *   <dt>showPlusMinus</dt><dd> show +/- icons to open/close branches? default=true</dd>
+ *   <dt>showLines    </dt><dd> show vertical lines connecting each level? default=true</dd>
+ *   <dt>defaultAction</dt><dd> Rico event handle to call when user clicks on an item, default is to call returnValue method</dd>
+ *   <dt>height       </dt><dd> control height? default=300px</dd>
+ *   <dt>width        </dt><dd> control width? default=300px</dd>
+ *   <dt>leafIcon     </dt><dd> class name to display img, default="rico-icon rico-doc" ('none'=no leaf icon)</dd>
+ *</dl>
+ */
+  initialize: function(id,url,options) {
+    Rico.extend(this, new Rico.Popup());
+    Rico.extend(this.options, {
+      ignoreClicks:true,
+      nodeIdDisplay:'none',
+      showCheckBox: false,
+      showFolders: false,
+      showPlusMinus: true,
+      showLines: true,
+      defaultAction: Rico.eventHandle(this,'nodeClick'),
+      height: '300px',
+      width: '300px',
+      leafIcon: 'rico-icon rico-doc'
+    });
+    Rico.extend(this.options, options || {});
+    this.id=id;
+    this.dataSource=url;
+    this.close=this.closePopup;
+    this.hoverSet = new Rico.HoverSet([]);
+  },
+
+  atLoad : function() {
+    this.treeDiv=document.createElement("div");
+    this.treeDiv.id=this.id;
+    this.treeDiv.className='ricoTree';
+    if (Rico.theme.treeContent) Rico.addClass(this.treeDiv,Rico.theme.treeContent);
+    this.treeDiv.style.height=this.options.height;
+    this.treeDiv.style.width=this.options.width;
+    this.createContainer();
+    this.content.className=Rico.theme.tree || 'ricoTreeContainer';
+    this.content.appendChild(this.treeDiv);
+    if (this.options.showCheckBox) {
+      this.buttonDiv=document.createElement("div");
+      this.buttonDiv.style.width=this.options.width;
+      this.buttonDiv.className='ricoTreeButtons';
+      if (Rico.getStyle(this.container,'position')=='absolute') {
+        var span=document.createElement("span");
+        span.innerHTML=RicoTranslate.getPhraseById('treeSave');
+        Rico.setStyle(span,{'float':'left',cursor:'pointer'});
+        this.buttonDiv.appendChild(span);
+        Rico.eventBind(span, 'click', Rico.eventHandle(this,'saveSelection'));
+      }
+      var span=document.createElement("span");
+      span.innerHTML=RicoTranslate.getPhraseById('treeClear');
+      Rico.setStyle(span,{'float':'right',cursor:'pointer'});
+      this.buttonDiv.appendChild(span);
+      this.content.appendChild(this.buttonDiv);
+      Rico.eventBind(span, 'click', Rico.eventHandle(this,'clrCheckBoxEvent'));
+    }
+    this.close();
+  },
+
+  setTreeDiv: function(divId) {
+    this.treeDiv = Rico.$(divId);
+    this.openPopup = function() {};
+  },
+
+  open: function() {
+    this.openPopup();
+    if (this.treeDiv.childNodes.length == 0 && this.dataSource) {
+      this.loadXMLDoc();
+    }
+  },
+
+  loadXMLDoc: function(branchPin) {
+    var parms = { id: this.id };
+    if (branchPin) parms.Parent=branchPin;
+    Rico.log('Tree loadXMLDoc: '+this.id);
+    var self=this;
+    new Rico.ajaxRequest(this.dataSource, {parameters:parms, method:'get', onComplete: function(request) { self.processResponse(request); } });
+  },
+
+  domID: function(nodeID,part) {
+    return 'RicoTree_'+part+'_'+this.id+'_'+nodeID;
+  },
+
+  processResponse: function(request) {
+    var response = request.responseXML.getElementsByTagName("ajax-response");
+    if (response == null || response.length != 1) return;
+    var rowsElement = response[0].getElementsByTagName('rows')[0];
+    var trs = rowsElement.getElementsByTagName("tr");
+    var rowdata=[];
+    for (var i=0; i < trs.length; i++) {
+      var cells = trs[i].getElementsByTagName("td");
+      if (cells.length < 5) continue;
+      var content=[];
+      content[5]=this.options.leafIcon;
+      for (var j=0; j<cells.length; j++) {
+        content[j]=Rico.getContentAsString(cells[j],true);
+      }
+      content[3] = content[3].match(/^0|L$/i) ? 0 : 1;
+      content[4] = parseInt(content[4]);
+      rowdata.push(content);
+    }
+    for (var i=0; i < rowdata.length; i++) {
+      var moreChildren=(i < rowdata.length-1) && (rowdata[i][0]==rowdata[i+1][0]);
+      this.addNode(rowdata[i][0],rowdata[i][1],rowdata[i][2],rowdata[i][3],rowdata[i][4],rowdata[i][5],!moreChildren);
+    }
+  },
+
+  addNode: function(parentId, nodeId, nodeDesc, isContainer, isSelectable, leafIcon, isLast) {
+    var parentNode=Rico.$(this.domID(parentId,'Parent'));
+    var parentChildren=Rico.$(this.domID(parentId,'Children'));
+    var level=parentNode ? parentNode.TreeLevel+1 : 0;
+    //alert("addNode at level " + level + " (" + nodeId + ")")
+    var tab = document.createElement("table");
+    var div = document.createElement("div");
+    div.id=this.domID(nodeId,'Children');
+    div.className='ricoTreeBranch';
+    div.style.display=parentNode ? 'none' : '';
+    tab.border=0;
+    tab.cellSpacing=0;
+    tab.cellPadding=0;
+    tab.id=this.domID(nodeId,'Parent');
+    tab.TreeLevel=level;
+    tab.TreeContainer=isContainer;
+    tab.TreeFetchedChildren=this.dataSource ? false : true;
+    var row=tab.insertRow(0);
+    var td=[];
+    for (var i=0; i<level-1; i++) {
+      td[i]=row.insertCell(-1);
+    }
+    if (level>1) {
+      var tdParent=parentNode.getElementsByTagName('td');
+      for (var i=0; i<level-2; i++) {
+        td[i].innerHTML=tdParent[i].innerHTML;
+      }
+      var img = document.createElement("div");
+      img.className='rico-icon rico-tree-'+(parentChildren.nextSibling && this.options.showLines ? "nodeline" : "nodeblank");
+      td[level-2].appendChild(img);
+    }
+    if (level>0) {
+      var suffix=isLast && this.options.showLines ? 'last' : '';
+      var prefix=this.options.showLines ? 'node' : '';
+      if (this.options.showPlusMinus && isContainer) {
+        var img = document.createElement("div");
+        img.name=nodeId;
+        img.style.cursor='pointer';
+        Rico.eventBind(img, 'click', Rico.eventHandle(this,'clickBranch'));
+        img.className='rico-icon rico-tree-'+prefix+"p"+suffix;
+        row.insertCell(-1).appendChild(img);
+      } else if (this.options.showLines) {
+        var img = document.createElement("div");
+        img.className='rico-icon rico-tree-node'+suffix;
+        row.insertCell(-1).appendChild(img);
+      }
+      if (this.options.showFolders && (isContainer || (leafIcon && leafIcon!='none'))) {
+        var img = document.createElement("div");
+        if (!isContainer) {
+          img.className=leafIcon;
+        } else {
+          img.name=nodeId;
+          img.style.cursor='pointer';
+          Rico.eventBind(img, 'click', Rico.eventHandle(this,'clickBranch'));
+          img.className='rico-icon rico-folderclosed';
+        }
+        row.insertCell(-1).appendChild(img);
+      }
+    }
+    if (isSelectable && this.options.showCheckBox) {
+      var chkbx=document.createElement("input");
+      chkbx.type="checkbox";
+      chkbx.value=nodeId;
+      row.insertCell(-1).appendChild(chkbx);
+    }
+
+    if (isSelectable && !this.options.showCheckBox) {
+      var span=document.createElement('a');
+      if (typeof isSelectable=='string') {
+        span.href=isSelectable;
+      } else {
+        span.href='javascript:void(0)';
+        Rico.eventBind(span, 'click', this.options.defaultAction);
+      }
+      this.hoverSet.add(span);
+    } else {
+      var span=document.createElement('p');
+    }
+    span.id=this.domID(nodeId,'Desc');
+    span.className='ricoTreeLevel'+level;
+    switch (this.options.nodeIdDisplay) {
+      case 'last': nodeDesc+=' ('+nodeId+')'; break;
+      case 'first': nodeDesc=nodeId+' - '+nodeDesc; break;
+      case 'tooltip': span.title=nodeId; break;
+    }
+       span.appendChild(document.createTextNode(nodeDesc));
+    row.insertCell(-1).appendChild(span);
+
+    var parent=parentChildren || this.treeDiv;
+    parent.appendChild(tab);
+    parent.appendChild(div);
+  },
+
+  nodeClick: function(e) {
+    var node=Rico.eventElement(e);
+    if (this.returnValue) {
+      var t=this.domID('','Desc');
+      this.returnValue(node.id.substr(t.length),node.innerHTML);
+    }
+    this.close();
+  },
+
+  saveSelection: function(e) {
+    if (this.returnValue) {
+      this.returnValue(this.getCheckedItems());
+    }
+    this.close();
+  },
+
+  getCheckedItems: function() {
+    var inp=this.treeDiv.getElementsByTagName('input');
+    var vals=[];
+    for (var i=0; i<inp.length; i++) {
+      if (inp[i].type=='checkbox' && inp[i].checked) {
+        vals.push(inp[i].value);
+      }
+    }
+    return vals;
+  },
+
+  setCheckBoxes: function(val) {
+    var inp=this.treeDiv.getElementsByTagName('input');
+    for (var i=0; i<inp.length; i++) {
+      if (inp[i].type=='checkbox') {
+        inp[i].checked=val;
+      }
+    }
+  },
+
+  clrCheckBoxEvent: function(e) {
+    Rico.eventStop(e);
+    this.setCheckBoxes(false);
+  },
+
+  clickBranch: function(e) {
+    var node=Rico.eventElement(e);
+    var tab=Rico.getParentByTagName(node,'table');
+    if (!tab || !tab.TreeContainer) return;
+    var a=tab.id.split('_');
+    a[1]='Children';
+    var childDiv=Rico.$(a.join('_'));
+    Rico.toggle(childDiv);
+    if (node.tagName=='DIV') {
+      var v=Rico.visible(childDiv);
+      if (node.className.match(/node(p|m)(last)?$/)) {
+        node.className=node.className.replace(/nodep|nodem/,'node'+(v ? 'm' : 'p'));
+      } else if (node.className.match(/folder(open|closed)$/)) {
+        node.className=node.className.replace(/folder(open|closed)/,'folder'+(v ? 'open' : 'closed'));
+      } else if (node.className.match(/\b(m|p)$/)) {
+        node.className=node.className.replace(/(p|m)$/,v ? 'm' : 'p');
+      }
+    }
+    if (!tab.TreeFetchedChildren) {
+      tab.TreeFetchedChildren=1;
+      this.loadXMLDoc(node.name);
+    }
+  }
+
+};
diff --git a/minsrc/ricoUI.js b/minsrc/ricoUI.js
new file mode 100644 (file)
index 0000000..e79a3ac
--- /dev/null
@@ -0,0 +1,1398 @@
+/*
+ *  (c) 2005-2009 Richard Cowin (http://openrico.org)
+ *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
+ *
+ *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
+ *  file except in compliance with the License. You may obtain a copy of the License at
+ *
+ *         http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software distributed under the
+ *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ *  either express or implied. See the License for the specific language governing permissions
+ *  and limitations under the License.
+ */
+
+Rico.applyShadow = function(elem,shadowFlag) {
+  if (typeof shadowFlag=='undefined') shadowFlag=true;
+  if (shadowFlag) Rico.addClass(elem,'ricoShadow');
+  return elem;
+};
+
+// ensure popups/windows get closed in the right order when the user hits escape key
+Rico._OpenPopupList = [];
+Rico._RemoveOpenPopup = function(popup) {
+  if (popup.openIndex >= 0 && popup.openIndex < Rico._OpenPopupList.length) Rico._OpenPopupList.splice(popup.openIndex,1);
+  popup.openIndex = -1;
+};
+Rico._AddOpenPopup = function(popup) {
+  popup.openIndex = Rico._OpenPopupList.push(popup) - 1;
+};
+Rico._checkEscKey = function(e) {
+  if (Rico.eventKey(e) != 27) return true;
+  while (Rico._OpenPopupList.length > 0) {
+    var popup = Rico._OpenPopupList.pop();
+    if (popup && popup.visible()) {
+      popup.openIndex = -1;
+      Rico.eventStop(e);
+      popup.closeFunc();
+      return false;
+    }
+  }
+  return true;
+};
+Rico.eventBind(document,"keyup", Rico.eventHandle(Rico,'_checkEscKey'));
+
+
+Rico.Popup = function(containerDiv,options) {
+  this.initialize(containerDiv,options);
+};
+
+Rico.Popup.prototype = {
+/**
+ * @class Class to manage pop-up div windows.
+ * @constructs
+ * @param options object may contain any of the following:<dl>
+ *   <dt>hideOnClick </dt><dd> hide popup when mouse button is clicked? default=true</dd>
+ *   <dt>ignoreClicks</dt><dd> if true, mouse clicks within the popup are not allowed to bubble up to parent elements</dd>
+ *   <dt>position    </dt><dd> defaults to absolute, use "auto" to auto-detect</dd>
+ *   <dt>shadow      </dt><dd> display shadow with popup? default=true</dd>
+ *   <dt>zIndex      </dt><dd> which layer? default=1</dd>
+ *   <dt>canDrag     </dt><dd> boolean value (or function that returns a boolean) indicating if it is ok to drag/reposition popup, default=false</dd>
+ *   <dt>onClose     </dt><dd> function to call when the popup is closed</dd>
+ *</dl>
+ * @param containerDiv if supplied, then setDiv() is called at the end of initialization
+ */
+  initialize: function(containerDiv,options) {
+    this.options = {
+      hideOnClick   : false,
+      ignoreClicks  : false,
+      position      : 'absolute',
+      shadow        : true,
+      zIndex        : 2,
+      canDrag       : false,
+      dragElement   : false,
+      closeFunc     : false
+    };
+    this.openIndex=-1;
+    if (containerDiv) this.setDiv(containerDiv,options);
+  },
+
+  createContainer: function(options) {
+    this.setDiv(document.createElement('div'), options);
+    if (options && options.parent) {
+      options.parent.appendChild(this.container);
+    } else {
+      document.getElementsByTagName("body")[0].appendChild(this.container);
+    }
+  },
+
+/**
+ * Apply popup behavior to a div that already exists in the DOM
+ * @param containerDiv div element (or element id) in the DOM. If null, then the div is created automatically.
+ */
+  setDiv: function(containerDiv,options) {
+    Rico.extend(this.options, options || {});
+    this.container=Rico.$(containerDiv);
+    if (this.options.position == 'auto') {
+      this.position=Rico.getStyle(this.container,'position').toLowerCase();
+    } else {
+      this.position=this.container.style.position=this.options.position;
+    }
+    if (this.position != 'absolute') {
+      this.content=this.container;
+      return;
+    }
+    if (this.options.zIndex >= 0) this.container.style.zIndex=this.options.zIndex;
+    if (this.options.closeFunc) {
+      this.closeFunc=this.options.closeFunc;
+    } else {
+      var self=this;
+      this.closeFunc=function() { self.closePopup(); };
+    }
+    //this.container.style.overflow='hidden';
+    this.container.style.top='0px';
+    this.container.style.left='0px';
+    this.container.style.display='none';
+
+    this.content=document.createElement('div');
+    while (this.container.firstChild) {
+      this.content.appendChild(this.container.firstChild);
+    }
+    this.container.appendChild(this.content);
+    this.content.className='RicoPopupContent';
+    this.content.style.position='relative';
+
+    if (Rico.isIE && Rico.ieVersion < 7) {
+      // create iframe shim
+      this.ifr = document.createElement('iframe');
+      this.ifr.style.position="absolute";
+      this.ifr.style.top     = '0px';
+      this.ifr.style.left    = '0px';
+      this.ifr.style.width   = '2000px';
+      this.ifr.style.height  = '2000px';
+      this.ifr.style.zIndex  = -1;
+      this.ifr.frameBorder   = 0;
+      this.ifr.src="javascript:false;";
+      this.content.appendChild(this.ifr);
+    }
+    Rico.applyShadow(this.container,this.options.shadow);
+
+    if (this.options.hideOnClick)
+      Rico.eventBind(document,"click", Rico.eventHandle(this,'_docClick'));
+    this.dragEnabled=false;
+    this.mousedownHandler = Rico.eventHandle(this,'_startDrag');
+    this.dragHandler = Rico.eventHandle(this,'_drag');
+    this.dropHandler = Rico.eventHandle(this,'_endDrag');
+    if (this.options.canDrag) this.enableDragging();
+    if (this.options.ignoreClicks || this.options.canDrag) this.ignoreClicks();
+  },
+
+  clearContent: function() {
+    this.content.innerHTML="";
+  },
+
+  setContent: function(content) {
+    this.content.innerHTML=content;
+  },
+
+  enableDragging: function() {
+    if (!this.dragEnabled && this.options.dragElement) {
+      Rico.eventBind(this.options.dragElement, "mousedown", this.mousedownHandler);
+      this.dragEnabled=true;
+    }
+    return this.dragEnabled;
+  },
+
+  disableDragging: function() {
+    if (!this.dragEnabled) return;
+    Rico.eventUnbind(this.options.dragElement, "mousedown", this.mousedownHandler);
+    this.dragEnabled=false;
+  },
+
+  setZ: function(zIndex) {
+    this.container.style.zIndex=zIndex;
+  },
+
+/** @private */
+  ignoreClicks: function() {
+    Rico.eventBind(this.container,"click", Rico.eventHandle(this,'_ignoreClick'));
+  },
+
+  _ignoreClick: function(e) {
+    if (e.stopPropagation)
+      e.stopPropagation();
+    else
+      e.cancelBubble = true;
+    return true;
+  },
+
+  _docClick: function(e) {
+    this.closeFunc();
+    return true;
+  },
+
+/**
+ * Move popup to specified position
+ */
+  move: function(left,top) {
+    if (typeof left=='number') this.container.style.left=left+'px';
+    if (typeof top=='number') this.container.style.top=top+'px';
+  },
+
+  _startDrag : function(event){
+    var elem=Rico.eventElement(event);
+    this.container.style.cursor='move';
+    this.lastMouse = Rico.eventClient(event);
+    Rico.eventBind(document, "mousemove", this.dragHandler);
+    Rico.eventBind(document, "mouseup", this.dropHandler);
+    Rico.eventStop(event);
+  },
+
+  _drag : function(event){
+    var newMouse = Rico.eventClient(event);
+    var newLeft = parseInt(this.container.style.left,10) + newMouse.x - this.lastMouse.x;
+    var newTop = parseInt(this.container.style.top,10) + newMouse.y - this.lastMouse.y;
+    this.move(newLeft, newTop);
+    this.lastMouse = newMouse;
+    Rico.eventStop(event);
+  },
+
+  _endDrag : function(){
+    this.container.style.cursor='';
+    Rico.eventUnbind(document, "mousemove", this.dragHandler);
+    Rico.eventUnbind(document, "mouseup", this.dropHandler);
+  },
+
+/**
+ * Display popup at specified position
+ */
+  openPopup: function(left,top) {
+    this.container.style.display=this.position=='absolute' ? "block" : Rico.isIE && Rico.ieVersion<8 ? "inline" : "inline-block";
+    if (typeof left=='number') this.container.style.left=left+'px';
+    if (typeof top=='number') this.container.style.top=top+'px';
+    if (this.container.id) Rico.log('openPopup '+this.container.id+' at '+left+','+top);
+    Rico._AddOpenPopup(this);
+  },
+
+  centerPopup: function() {
+    this.openPopup();
+    var msgWidth=this.container.offsetWidth;
+    var msgHeight=this.container.offsetHeight;
+    var divwi=this.container.parentNode.offsetWidth;
+    var divht=this.container.parentNode.offsetHeight;
+    this.move(parseInt(Math.max((divwi-msgWidth)/2,0),10), parseInt(Math.max((divht-msgHeight)/2,0),10));
+  },
+
+  visible: function() {
+    return Rico.visible(this.container);
+  },
+
+/**
+ * Hide popup
+ */
+  closePopup: function() {
+    Rico._RemoveOpenPopup(this);
+    if (!Rico.visible(this.container)) return;
+    if (this.container.id) Rico.log('closePopup '+this.container.id);
+    if (this.dragEnabled) this._endDrag();
+    this.container.style.display="none";
+    if (this.options.onClose) this.options.onClose();
+  }
+
+};
+
+Rico.closeButton = function(handle) {
+  var a = document.createElement('a');
+  a.className='RicoCloseAnchor';
+  if (Rico.theme.closeAnchor) Rico.addClass(a,Rico.theme.closeAnchor);
+  var span = a.appendChild(document.createElement('span'));
+  span.title=Rico.getPhraseById('close');
+  new Rico.HoverSet([a]);
+  Rico.addClass(span,Rico.theme.close || 'rico-icon RicoClose');
+  Rico.eventBind(a,"click", handle);
+  return a;
+};
+
+Rico.floatButton = function(buttonName, handle, title) {
+  var a=document.createElement("a");
+  a.className='RicoButtonAnchor'
+  Rico.addClass(a,Rico.theme.buttonAnchor || 'RicoButtonAnchorNative');
+  var span=a.appendChild(document.createElement("span"));
+  if (title) span.title=title;
+  span.className=Rico.theme[buttonName.toLowerCase()] || 'rico-icon Rico'+buttonName;
+  Rico.eventBind(a,"click", handle, false);
+  new Rico.HoverSet([a]);
+  return a
+}
+
+Rico.clearButton = function(handle) {
+  var span=document.createElement("span");
+  span.title=Rico.getPhraseById('clear');
+  span.className='ricoClear';
+  Rico.addClass(span, Rico.theme.clear || 'rico-icon ricoClearNative');
+  Rico.eventBind(span,"click", handle);
+  return span;
+}
+
+Rico.Window = function(title, options, contentParam) {
+  this.initialize(title, options, contentParam);
+};
+
+Rico.Window.prototype = {
+
+/**
+ * Create popup div with a title bar.
+ */
+  initialize: function(title, options, contentParam) {
+    options=options || {overflow:'auto'};
+    Rico.extend(this, new Rico.Popup());
+
+    this.titleDiv = document.createElement('div');
+    this.options.canDrag=true;
+    this.options.dragElement=this.titleDiv;
+    this.createContainer(options);
+    this.content.appendChild(this.titleDiv);
+    contentParam=Rico.$(contentParam);
+    this.contentDiv=contentParam || document.createElement('div');
+    this.content.appendChild(this.contentDiv);
+
+    // create title area
+    this.titleDiv.className='ricoTitle';
+    if (Rico.theme.dialogTitle) Rico.addClass(this.titleDiv,Rico.theme.dialogTitle);
+    this.titleDiv.style.position='relative';
+    this.titleContent = document.createElement('span');
+    this.titleContent.className='ricoTitleSpan';
+    this.titleDiv.appendChild(this.titleContent);
+    this.titleDiv.appendChild(Rico.closeButton(Rico.eventHandle(this,'closePopup')));
+    if (!title && contentParam) {
+      title=contentParam.title;
+      contentParam.title='';
+    }
+    this.setTitle(title || '&nbsp;');
+
+    // create content area
+    this.contentDiv.className='ricoContent';
+    if (Rico.theme.dialogContent) Rico.addClass(this.contentDiv,Rico.theme.dialogContent);
+    this.contentDiv.style.position='relative';
+    if (options.height) this.contentDiv.style.height=options.height;
+    if (options.width) this.contentDiv.style.width=options.width;
+    if (options.overflow) this.contentDiv.style.overflow=options.overflow;
+    Rico.addClass(this.content,'ricoWindow');
+    if (Rico.theme.dialog) Rico.addClass(this.content,Rico.theme.dialog);
+    if (Rico.isIE) {
+      // fix float'ed content in IE
+      this.titleDiv.style.zoom=1;
+      this.contentDiv.style.zoom=1;
+    }
+
+    this.content=this.contentDiv;
+  },
+
+  setTitle: function(title) {
+    this.titleContent.innerHTML=title;
+  }
+
+}
+
+
+Rico.Menu = function(options) {
+  this.initialize(options);
+}
+
+Rico.Menu.prototype = {
+/**
+ * @class Implements popup menus and submenus
+ * @extends Rico.Popup
+ * @constructs
+ */
+  initialize: function(options) {
+    Rico.extend(this, new Rico.Popup());
+    Rico.extend(this.options, {
+      width        : "15em",
+      arrowColor   : "b",   // for submenus: b=black, w=white
+      showDisabled : false,
+      hideOnClick  : true
+    });
+    if (typeof options=='string')
+      this.options.width=options;
+    else
+      Rico.extend(this.options, options || {});
+    this.hideFunc=null;
+    this.highlightElem=null;
+  },
+
+  createDiv: function(parentNode) {
+    if (this.container) return;
+    var self=this;
+    var options={ closeFunc: function() { self.cancelmenu(); } };
+    if (parentNode) options.parent=parentNode;
+    this.createContainer(options);
+    this.content.className = Rico.isWebKit ? 'ricoMenuSafari' : 'ricoMenu';
+    this.content.style.width=this.options.width;
+    this.direction=Rico.getStyle(this.container,'direction') || 'ltr';
+    this.direction=this.direction.toLowerCase();  // ltr or rtl
+    this.hidemenu();
+    this.itemCount=0;
+  },
+
+  showmenu: function(e,hideFunc){
+    Rico.eventStop(e);
+    this.hideFunc=hideFunc;
+    if (this.content.childNodes.length==0) {
+      this.cancelmenu();
+      return false;
+    }
+    var mousePos = Rico.eventClient(e);
+    this.openmenu(mousePos.x,mousePos.y,0,0);
+  },
+
+  openmenu: function(x,y,clickItemWi,clickItemHt,noOffset) {
+    var newLeft=x + (noOffset ? 0 : Rico.docScrollLeft());
+    this.container.style.visibility="hidden";
+    this.container.style.display="block";
+    var w=this.container.offsetWidth;
+    var cw=this.content.offsetWidth;
+    //window.status='openmenu: newLeft='+newLeft+' width='+w+' clickItemWi='+clickItemWi+' windowWi='+Rico.windowWidth();
+    if (this.direction == 'rtl') {
+      if (newLeft > w+clickItemWi) newLeft-=cw+clickItemWi;
+    } else {
+      if (x+w > Rico.windowWidth()) newLeft-=cw+clickItemWi-2;
+    }
+    var scrTop=Rico.docScrollTop();
+    var newTop=y + (noOffset ? 0 : scrTop);
+    if (y+this.container.offsetHeight-scrTop > Rico.windowHeight())
+      newTop=Math.max(newTop-this.content.offsetHeight+clickItemHt,0);
+    this.openPopup(newLeft,newTop);
+    this.container.style.visibility ="visible";
+    return false;
+  },
+
+  clearMenu: function() {
+    this.clearContent();
+    this.defaultAction=null;
+    this.itemCount=0;
+  },
+
+  addMenuHeading: function(hdg) {
+    var el=document.createElement('div');
+    el.innerHTML=hdg;
+    el.className='ricoMenuHeading';
+    this.content.appendChild(el);
+  },
+
+  addMenuBreak: function() {
+    var brk=document.createElement('div');
+    brk.className="ricoMenuBreak";
+    this.content.appendChild(brk);
+  },
+
+  addSubMenuItem: function(menutext, submenu, translate) {
+    var dir=this.direction=='rtl' ? 'left' : 'right';
+    var a=this.addMenuItem(menutext,null,true,null,translate);
+    a.className='ricoSubMenu';
+    var arrowdiv = a.appendChild(document.createElement('div'));
+    arrowdiv.className='rico-icon rico-'+dir+'-'+this.options.arrowColor;
+    Rico.setStyle(arrowdiv,{position:'absolute',top:'2px'});
+    arrowdiv.style[dir]='0px';
+    a.RicoSubmenu=submenu;
+    Rico.eventBind(a,"mouseover", Rico.eventHandle(this,'showSubMenu'));
+    Rico.eventBind(a,"mouseout", Rico.eventHandle(this,'subMenuOut'));
+  },
+
+  showSubMenu: function(e) {
+    if (this.openSubMenu) this.hideSubMenu();
+    var a=Rico.eventElement(e);
+    if (!a.RicoSubmenu) a=a.parentNode; // event can happen on arrow div
+    if (!a.RicoSubmenu) return;
+    this.openSubMenu=a.RicoSubmenu;
+    this.openMenuAnchor=a;
+    if (Rico.hasClass(a,'ricoSubMenu')) {
+      Rico.removeClass(a,'ricoSubMenu');
+      Rico.addClass(a,'ricoSubMenuOpen');
+    }
+    a.RicoSubmenu.openmenu(parseInt(this.container.style.left)+a.offsetWidth, parseInt(this.container.style.top)+a.offsetTop, a.offsetWidth-2, a.offsetHeight+2,true);
+  },
+
+  subMenuOut: function(e) {
+    if (!this.openSubMenu) return;
+    Rico.eventStop(e);
+    var elem=Rico.eventElement(e);
+    var reltg = Rico.eventRelatedTarget(e) || e.toElement;
+    try {
+      while (reltg != null && reltg != this.openSubMenu.div)
+        reltg=reltg.parentNode;
+    } catch(err) {}
+    if (reltg == this.openSubMenu.div) return;
+    this.hideSubMenu();
+  },
+
+  hideSubMenu: function() {
+    if (this.openMenuAnchor) {
+      Rico.removeClass(this.openMenuAnchor,'ricoSubMenuOpen');
+      Rico.addClass(this.openMenuAnchor,'ricoSubMenu');
+      this.openMenuAnchor=null;
+    }
+    if (this.openSubMenu) {
+      this.openSubMenu.hidemenu();
+      this.openSubMenu=null;
+    }
+  },
+
+  addMenuItemId: function(phraseId,action,enabled,title,target) {
+    if ( arguments.length < 3 ) enabled=true;
+    this.addMenuItem(Rico.getPhraseById(phraseId),action,enabled,title,target);
+  },
+
+// if action is a string, then it is assumed to be a URL and the target parm can be used indicate which window gets the content
+// action can also be a function
+// action can also be a Rico.eventHandle, but set target='event' in this case
+  addMenuItem: function(menutext,action,enabled,title,target) {
+    if (arguments.length >= 3 && !enabled && !this.options.showDisabled) return null;
+    this.itemCount++;
+    var a = document.createElement(typeof action=='string' ? 'a' : 'div');
+    if ( arguments.length < 3 || enabled ) {
+      if (typeof action=='string') {
+        a.href = action;
+        if (target) a.target = target;
+      } else if (target=='event') {
+        Rico.eventBind(a,"click", action);
+      } else {
+        a.onclick=action;
+      }
+      a.className = 'enabled';
+      if (this.defaultAction==null) this.defaultAction=action;
+    } else {
+      a.disabled = true;
+      a.className = 'disabled';
+    }
+    a.innerHTML = menutext;
+    if (typeof title=='string')
+      a.title = title;
+    a=this.content.appendChild(a);
+    Rico.eventBind(a,"mouseover", Rico.eventHandle(this,'mouseOver'));
+    Rico.eventBind(a,"mouseout", Rico.eventHandle(this,'mouseOut'));
+    return a;
+  },
+
+  mouseOver: function(e) {
+    if (this.highlightElem && this.highlightElem.className=='enabled-hover') {
+      // required for Safari
+      this.highlightElem.className='enabled';
+      this.highlightElem=null;
+    }
+    var elem=Rico.eventElement(e);
+    if (this.openMenuAnchor && this.openMenuAnchor!=elem)
+      this.hideSubMenu();
+    if (elem.className=='enabled') {
+      elem.className='enabled-hover';
+      this.highlightElem=elem;
+    }
+  },
+
+  mouseOut: function(e) {
+    var elem=Rico.eventElement(e);
+    if (elem.className=='enabled-hover') elem.className='enabled';
+    if (this.highlightElem==elem) this.highlightElem=null;
+  },
+
+  cancelmenu: function() {
+    if (!this.visible()) return;
+    if (this.hideFunc) this.hideFunc();
+    this.hideFunc=null;
+    this.hidemenu();
+  },
+
+  hidemenu: function() {
+    if (this.openSubMenu) this.openSubMenu.hidemenu();
+    this.closePopup();
+  }
+
+}
+
+
+Rico.SelectionSet = function(selectionSet, options) {
+  this.initialize(selectionSet, options);
+}
+
+Rico.SelectionSet.prototype = {
+/**
+ * @class
+ * @constructs
+ * @param selectionSet collection of DOM elements (or a CSS selection string)
+ * @param options object may contain any of the following:<dl>
+ *   <dt>selectedClass</dt><dd>class name to add when element is selected, default is "selected"</dd>
+ *   <dt>selectNode   </dt><dd>optional function that returns the element to be selected</dd>
+ *   <dt>onSelect     </dt><dd>optional function that gets called when element is selected</dd>
+ *   <dt>onFirstSelect</dt><dd>optional function that gets called the first time element is selected</dd>
+ *   <dt>noDefault    </dt><dd>when true, no element in the set is initially selected, default is false</dd>
+ *   <dt>selectedIndex</dt><dd>index of the element that should be initially selected, default is 0</dd>
+ *   <dt>cookieName   </dt><dd>optional name of cookie to use to remember selected element. If specified, and the cookie exists, then the cookie value overrides selectedIndex.</dd>
+ *   <dt>cookieDays   </dt><dd>specifies how long cookie should persist (in days). If unspecified, then the cookie persists for the current session.</dd>
+ *   <dt>cookiePath   </dt><dd>optional cookie path</dd>
+ *   <dt>cookieDomain </dt><dd>optional cookie domain</dd>
+ *</dl>
+ */
+  initialize: function(selectionSet, options){
+    Rico.log('SelectionSet#initialize');
+    this.options = options || {};
+    if (typeof selectionSet == 'string')
+      selectionSet = Rico.select(selectionSet);
+    this.previouslySelected = [];
+    this.selectionSet = [];
+    this.selectedClassName = this.options.selectedClass || Rico.theme.selected || "selected";
+    this.selectNode = this.options.selectNode || function(e){return e;};
+    this.onSelect = this.options.onSelect;
+    this.onFirstSelect = this.options.onFirstSelect;
+    var self=this;
+    this.clickHandler = function(idx) { self.selectIndex(idx); };
+    this.selectedIndex=-1;
+    for (var i=0; i<selectionSet.length; i++)
+      this.add(selectionSet[i]);
+    if (!this.options.noDefault) {
+      var cookieIndex=this.options.cookieName ? this.getCookie() : 0;
+      this.selectIndex(cookieIndex || this.options.selectedIndex || 0);
+    }
+  },
+  getCookie: function() {
+    var cookie = Rico.getCookie(this.options.cookieName);
+    if (!cookie) return 0;
+    var index = parseInt(cookie);
+    return index < this.selectionSet.length ? index : 0;
+  },
+  reset: function(){
+    this.previouslySelected = [];
+    this._notifySelected(this.selectedIndex);
+  },
+  clearSelected: function() {
+    if (this.selected)
+      Rico.removeClass(this.selectNode(this.selected), this.selectedClassName);
+  },
+  getIndex: function(element) {
+    for (var i=0; i<this.selectionSet.length; i++) {
+      if (element == this.selectionSet[i]) return i;
+    }
+    return -1;
+  },
+  select: function(element){
+    if (this.selected == element) return;
+    var i=this.getIndex(element);
+    if (i >= 0) this.selectIndex(i);
+  },
+  _notifySelected: function(index){
+    if (index < 0) return;
+    var element = this.selectionSet[index];
+    if (this.options.cookieName)
+      Rico.setCookie(this.options.cookieName, index, this.options.cookieDays, this.options.cookiePath, this.options.cookieDomain);
+    if (this.onFirstSelect && !this.previouslySelected[index]){
+      this.onFirstSelect(element, index);
+      this.previouslySelected[index] = true;
+    }
+    if (this.onSelect)
+      try{
+        this.onSelect(index);
+      } catch (e) {};
+  },
+  selectIndex: function(index){
+    if (this.selectedIndex == index || index >= this.selectionSet.length) return;
+    this.clearSelected();
+    this._notifySelected(index);
+    this.selectedIndex = index;
+    this.selected=this.selectionSet[index].element;
+    Rico.addClass(this.selectNode(this.selected), this.selectedClassName);
+  },
+  nextSelectIndex: function(){
+    return (this.selectedIndex + 1) % this.selectionSet.length;
+  },
+  nextSelectItem: function(){
+    return this.selectionSet[this.nextSelectIndex()];
+  },
+  selectNext: function(){
+    this.selectIndex(this.nextSelectIndex());
+  },
+  add: function(item){
+    var index=this.selectionSet.length;
+    this.selectionSet[index] = new Rico._SelectionItem(item,index,this.clickHandler);
+  },
+  remove: function(item){
+    if (item==this.selected) this.clearSelected();
+    var i=this.getIndex(item);
+    if (i < 0) return;
+    this.selectionSet[i].remove();
+    this.selectionSet.splice(i,1);
+  },
+  removeAll: function(){
+    this.clearSelected();
+    while (this.selectionSet.length > 0) {
+      this.selectionSet.pop().remove();
+    }
+  }
+};
+
+
+Rico._SelectionItem=function(element,index,callback) {
+  this.add(element,index,callback);
+};
+
+Rico._SelectionItem.prototype = {
+  add: function(element,index,callback) {
+    this.element=element;
+    this.index=index;
+    this.callback=callback;
+    this.handle=Rico.eventHandle(this,'click');
+    Rico.eventBind(element, "click", this.handle);
+  },
+
+  click: function(ev) {
+    this.callback(this.index);
+  },
+
+  remove: function() {
+    Rico.eventUnbind(this.element, "click", this.handle);
+  }
+};
+
+
+Rico.HoverSet = function(hoverSet, options) {
+  this.initialize(hoverSet, options);
+};
+
+Rico.HoverSet.prototype = {
+/**
+ * @class
+ * @constructs
+ * @param hoverSet collection of DOM elements
+ * @param options object may contain any of the following:<dl>
+ *   <dt>hoverClass</dt><dd> class name to add when mouse is over element, default is "hover"</dd>
+ *   <dt>hoverNodes</dt><dd> optional function to select/filter which nodes are in the set</dd>
+ *</dl>
+ */
+  initialize: function(hoverSet, options){
+    Rico.log('HoverSet#initialize');
+    options = options || {};
+    this.hoverClass = options.hoverClass || Rico.theme.hover || "hover";
+    this.hoverFunc = options.hoverNodes || function(e){return [e];};
+    this.hoverSet=[];
+    if (!hoverSet) return;
+    for (var i=0; i<hoverSet.length; i++)
+      this.add(hoverSet[i]);
+  },
+  add: function(item) {
+    this.hoverSet.push(new Rico._HoverItem(item,this.hoverFunc,this.hoverClass));
+  },
+  removeAll: function(){
+    while (this.hoverSet.length > 0) {
+      this.hoverSet.pop().remove();
+    }
+  }
+};
+
+
+Rico._HoverItem=function(element,selectFunc,hoverClass) {
+  this.add(element,selectFunc,hoverClass);
+};
+
+Rico._HoverItem.prototype = {
+  add: function(element,selectFunc,hoverClass) {
+    this.element=element;
+    this.selectFunc=selectFunc;
+    this.hoverClass=hoverClass;
+    this.movehandle=Rico.eventHandle(this,'move');
+    this.outhandle=Rico.eventHandle(this,'mouseout');
+    Rico.eventBind(element, "mousemove", this.movehandle);
+    Rico.eventBind(element, "mouseout", this.outhandle);
+  },
+
+  move: function(ev) {
+    var elems=this.selectFunc(this.element);
+    for (var i=0; i<elems.length; i++)
+      Rico.addClass(elems[i],this.hoverClass);
+  },
+
+  mouseout: function(ev) {
+    var elems=this.selectFunc(this.element);
+    for (var i=0; i<elems.length; i++)
+      Rico.removeClass(elems[i],this.hoverClass);
+  },
+
+  remove: function() {
+    Rico.eventUnbind(element, "mousemove", this.movehandle);
+    Rico.eventUnbind(element, "mouseout", this.outhandle);
+  }
+};
+
+
+/** @namespace */
+Rico.Effect = {};
+Rico.Effect.easeIn = function(step){
+  return Math.sqrt(step);
+};
+Rico.Effect.easeOut = function(step){
+  return step*step;
+};
+
+
+/** @class core methods for transition effects */
+Rico.ContentTransitionBase = function() {};
+Rico.ContentTransitionBase.prototype = {
+  initBase: function(titles, contents, options) {
+    Rico.log('ContentTransitionBase#initBase');
+    if (typeof titles == 'string')
+      titles = Rico.select(titles);
+    if (typeof contents == 'string')
+      contents = Rico.select(contents);
+
+    this.options = options || {};
+    this.titles = titles;
+    this.contents = contents;
+    this.hoverSet = new Rico.HoverSet(titles, options);
+    for (var i=0; i<contents.length; i++) {
+      if (contents[i]) Rico.hide(contents[i]);
+    }
+    var self=this;
+    this.selectionSet = new Rico.SelectionSet(titles, Rico.extend(options, { onSelect: function(idx) { self._finishSelect(idx); } }));
+  },
+  reset: function(){
+    this.selectionSet.reset();
+  },
+  select: function(index) {
+    this.selectionSet.selectIndex(index);
+  },
+  _finishSelect: function(index) {
+    Rico.log('ContentTransitionBase#_finishSelect');
+    var panel = this.contents[index];
+    if (!panel) {
+      alert('Internal error: no panel @index='+index);
+      return;
+    }
+    if ( this.selected == panel) return;
+    if (this.transition){
+      if (this.selected){
+        this.transition(panel);
+      } else {
+        panel.style.display='block';
+      }
+    } else {
+      if (this.selected) Rico.hide(this.selected);
+      panel.style.display='block';
+    }
+    this.selected = panel;
+  },
+  addBase: function(title, content){
+    this.titles.push(title);
+    this.contents.push(content);
+    this.hoverSet.add(title);
+    this.selectionSet.add(title);
+    Rico.hide(content);
+    //this.selectionSet.select(title);
+  },
+  removeBase: function(title){},
+  removeAll: function(){
+    this.hoverSet.removeAll();
+    this.selectionSet.removeAll();
+  }
+};
+
+
+/**
+ * @class Implements accordion effect
+ * @see Rico.ContentTransitionBase#initialize for construction parameters
+ * @extends Rico.ContentTransitionBase
+ */
+Rico.Accordion = function(element, options) {
+  this.initialize(element, options);
+};
+
+Rico.Accordion.prototype = Rico.extend(new Rico.ContentTransitionBase(),
+/** @lends Rico.Accordion# */
+{
+  initialize: function(element, options) {
+    element=Rico.$(element);
+    element.style.overflow='hidden';
+    element.className=options.accClass || Rico.theme.accordion || "Rico_accordion";
+    if (typeof options.panelWidth=='number') options.panelWidth+="px";
+    if (options.panelWidth) element.style.width = options.panelWidth;
+    var panels=Rico.getDirectChildrenByTag(element,'div');
+    var items,titles=[], contents=[];
+    for (var i=0; i<panels.length; i++) {
+      items=Rico.getDirectChildrenByTag(panels[i],'div');
+      if (items.length>=2) {
+        items[0].className=options.titleClass || Rico.theme.accTitle || "Rico_accTitle";
+        items[1].className=options.contentClass || Rico.theme.accContent || "Rico_accContent";
+        titles.push(items[0]);
+        contents.push(items[1]);
+        var a=Rico.wrapChildren(items[0],'','','a');
+        a.href="javascript:void(0)";
+      }
+    }
+    Rico.log('creating Rico.Accordion for '+element.id+' with '+titles.length+' panels');
+    this.initBase(titles, contents, options);
+    this.selected.style.height = this.options.panelHeight + "px";
+    this.totSteps=(typeof options.duration =='number' ? options.duration : 200)/25;
+  },
+  transition: function(p){
+    if (!this.options.noAnimate) {
+      this.closing=this.selected;
+      this.opening=p;
+      this.curStep=0;
+      var self=this;
+      this.timer=setInterval(function() { self.step(); },25);
+    } else {
+      p.style.height = this.options.panelHeight + "px";
+      if (this.selected) Rico.hide(this.selected);
+      p.style.display='block';
+    }
+  },
+  step: function() {
+    this.curStep++;
+    var oheight=Math.round(this.curStep/this.totSteps*this.options.panelHeight);
+    this.opening.style.height=oheight+'px';
+    this.closing.style.height=(this.options.panelHeight - oheight)+'px';
+    if (this.curStep==1) {
+      this.opening.style.paddingTop=this.opening.style.paddingBottom='0px';
+      this.opening.style.display='block';
+    }
+    if (this.curStep==this.totSteps) {
+      clearInterval(this.timer);
+      this.opening.style.paddingTop=this.opening.style.paddingBottom='';
+      Rico.hide(this.closing);
+    }
+  },
+  setPanelHeight: function(h) {
+    this.options.panelHeight = h;
+    this.selected.style.height = this.options.panelHeight + "px";
+  }
+});
+
+
+/**
+ * @class Implements tabbed panel effect
+ * @see Rico.ContentTransitionBase#initialize for construction parameters
+ * @extends Rico.ContentTransitionBase
+ */
+Rico.TabbedPanel = function(element, options) {
+  this.initialize(element, options);
+};
+
+Rico.TabbedPanel.prototype = Rico.extend(new Rico.ContentTransitionBase(),
+{
+  initialize: function(element, options) {
+    element=Rico.$(element);
+    options=options || {};
+    if (typeof options.panelWidth=='number') options.panelWidth+="px";
+    if (typeof options.panelHeight=='number') options.panelHeight+="px";
+    element.className=options.tabClass || Rico.theme.tabPanel || "Rico_tabPanel";
+    if (options.panelWidth) element.style.width = options.panelWidth;
+    var items = [];
+    var allKids = element.childNodes;
+    for( var i = 0 ; i < allKids.length ; i++ ) {
+      if (allKids[i] && allKids[i].tagName && allKids[i].tagName.match(/^div|ul$/i))
+        items.push(allKids[i]);
+    }
+    if (items.length < 2) return;
+    var childTag=items[0].tagName.toLowerCase()=='ul' ? 'li' : 'div';
+    items[0].className=options.navContainerClass || Rico.theme.tabNavContainer || "Rico_tabNavContainer";
+    items[0].style.listStyle='none';
+    items[1].className=options.contentContainerClass || Rico.theme.tabContentContainer || "Rico_tabContentContainer";
+    var titles=Rico.getDirectChildrenByTag(items[0], childTag);
+    var contents=Rico.getDirectChildrenByTag(items[1],'div');
+    if (!options.corners) options.corners='top';
+    for (var i=0; i<titles.length; i++) {
+      titles[i].className=options.titleClass || Rico.theme.tabTitle || "Rico_tabTitle";
+      var a=Rico.wrapChildren(titles[i],'','','a');
+      a.href="javascript:void(0)";
+      contents[i].className=options.contentClass || Rico.theme.tabContent || "Rico_tabContent";
+      if (options.panelHeight) contents[i].style.overflow='auto';
+      if (options.corners!='none') {
+        if (options.panelHdrWidth) titles[i].style.width=options.panelHdrWidth;
+        Rico.Corner.round(titles[i], Rico.theme.tabCornerOptions || options);
+      }
+    }
+    options.selectedClass=Rico.theme.tabSelected || 'selected';
+    this.initBase(titles, contents, options);
+    if (this.selected) this.transition(this.selected);
+  },
+  transition: function(p){
+    Rico.log('TabbedPanel#transition '+typeof(p));
+    if (this.selected) Rico.hide(this.selected);
+    Rico.show(p);
+    if (this.options.panelHeight) p.style.height = this.options.panelHeight;
+  }
+});
+
+
+/**
+ * @namespace
+ */
+Rico.Corner = {
+
+   round: function(e, options) {
+      e = Rico.$(e);
+      this.options = {
+         corners : "all",
+         bgColor : "fromParent",
+         compact : false,
+         nativeCorners: false  // only use native corners?
+      };
+      Rico.extend(this.options, options || {});
+      if (typeof(Rico.getStyle(e,'border-radius'))=='string')
+        this._roundCornersStdCss(e);
+      else if (typeof(Rico.getStyle(e,'-webkit-border-radius'))=='string')
+        this._roundCornersWebKit(e);
+      else if (typeof(Rico.getStyle(e,'-moz-border-radius'))=='string')
+        this._roundCornersMoz(e);
+      else if (!this.options.nativeCorners)
+        this._roundCornersImpl(e);
+   },
+
+    _roundCornersStdCss: function(e) {
+      var radius=this.options.compact ? '4px' : '8px';
+      if (this._hasString(this.options.corners, "all"))
+        Rico.setStyle(e, {borderRadius:radius});
+      else {
+        if (this._hasString(this.options.corners, "top", "tl")) Rico.setStyle(e, {borderTopLeftRadius:radius});
+        if (this._hasString(this.options.corners, "top", "tr")) Rico.setStyle(e, {borderTopRightRadius:radius});
+        if (this._hasString(this.options.corners, "bottom", "bl")) Rico.setStyle(e, {borderBottomLeftRadius:radius});
+        if (this._hasString(this.options.corners, "bottom", "br")) Rico.setStyle(e, {borderBottomRightRadius:radius});
+      }
+   },
+
+   _roundCornersWebKit: function(e) {
+      var radius=this.options.compact ? '4px' : '8px';
+      if (this._hasString(this.options.corners, "all"))
+        Rico.setStyle(e, {WebkitBorderRadius:radius});
+      else {
+        if (this._hasString(this.options.corners, "top", "tl")) Rico.setStyle(e, {WebkitBorderTopLeftRadius:radius});
+        if (this._hasString(this.options.corners, "top", "tr")) Rico.setStyle(e, {WebkitBorderTopRightRadius:radius});
+        if (this._hasString(this.options.corners, "bottom", "bl")) Rico.setStyle(e, {WebkitBorderBottomLeftRadius:radius});
+        if (this._hasString(this.options.corners, "bottom", "br")) Rico.setStyle(e, {WebkitBorderBottomRightRadius:radius});
+      }
+   },
+
+   _roundCornersMoz: function(e) {
+      var radius=this.options.compact ? '4px' : '8px';
+      if (this._hasString(this.options.corners, "all"))
+        Rico.setStyle(e, {MozBorderRadius:radius});
+      else {
+        if (this._hasString(this.options.corners, "top", "tl")) Rico.setStyle(e, {MozBorderRadiusTopleft:radius});
+        if (this._hasString(this.options.corners, "top", "tr")) Rico.setStyle(e, {MozBorderRadiusTopright:radius});
+        if (this._hasString(this.options.corners, "bottom", "bl")) Rico.setStyle(e, {MozBorderRadiusBottomleft:radius});
+        if (this._hasString(this.options.corners, "bottom", "br")) Rico.setStyle(e, {MozBorderRadiusBottomright:radius});
+      }
+   },
+
+  _roundCornersImpl: function(e) {
+      var bgColor = this.options.bgColor == "fromParent" ? this._background(e.parentNode) : this.options.bgColor;
+      e.style.position='relative';
+      //this.options.numSlices = this.options.compact ? 2 : 4;
+      if (this._hasString(this.options.corners, "all", "top", "tl")) this._createCorner(e,'top','left',bgColor);
+      if (this._hasString(this.options.corners, "all", "top", "tr")) this._createCorner(e,'top','right',bgColor);
+      if (this._hasString(this.options.corners, "all", "bottom", "bl")) this._createCorner(e,'bottom','left',bgColor);
+      if (this._hasString(this.options.corners, "all", "bottom", "br")) this._createCorner(e,'bottom','right',bgColor);
+   },
+
+  _createCorner: function(elem,tb,lr,bgColor) {
+    //alert('Corner: '+tb+' '+lr+' bgColor='+typeof(bgColor));
+    var corner = document.createElement("div");
+    corner.className='ricoCorner';
+    Rico.setStyle(corner,{width:'6px', height:'5px'});
+    var borderStyle = Rico.getStyle(elem,'border-'+tb+'-style');
+    var borderColor = borderStyle=='none' ? bgColor : Rico.getStyle(elem,'border-'+tb+'-color');
+    //alert('Corner: '+tb+' '+borderStyle+borderColor+' '+);
+    var pos=borderStyle=='none' ? '0px' : '-1px';
+    corner.style[tb]=pos;
+    corner.style[lr]=Rico.isIE && Rico.ieVersion<7 && lr=='right' && borderStyle!='none' ? '-2px' : '-1px';
+    //corner.style[lr]='-1px';
+    elem.appendChild(corner);
+    var marginSizes = [ 0, 2, 3, 4, 4 ];
+    if (tb=='bottom') marginSizes.reverse();
+    var borderVal= borderStyle=='none' ? '0px none' : '1px solid '+borderColor;
+    var d= lr=='left' ? 'Right' : 'Left';
+    for (var i=0; i<marginSizes.length; i++) {
+      var slice = document.createElement("div");
+      Rico.setStyle(slice,{backgroundColor:bgColor,height:'1px'});
+      slice.style['margin'+d]=marginSizes[i]+'px';
+      slice.style['border'+d]=borderVal;
+      corner.appendChild(slice);
+    }
+  },
+
+  _background: function(elem) {
+     try {
+       var actualColor = Rico.getStyle(elem, "backgroundColor");
+
+       // if color is tranparent, check parent
+       // Safari returns "rgba(0, 0, 0, 0)", which means transparent
+       if ( actualColor.match(/^(transparent|rgba\(0,\s*0,\s*0,\s*0\))$/i) && elem.parentNode )
+          return this._background(elem.parentNode);
+
+       return actualColor == null ? "#ffffff" : actualColor;
+     } catch(err) {
+       return "#ffffff";
+     }
+   },
+
+   _hasString: function(str) {
+     for(var i=1 ; i<arguments.length ; i++) {
+       if (str.indexOf(arguments[i]) >= 0) return true;
+     }
+     return false;
+   }
+
+};
+
+Rico.toColorPart = function(c) {
+  return Rico.zFill(c, 2, 16);
+};
+
+
+Rico.Color = function(red, green, blue) {
+  this.initialize(red, green, blue);
+};
+
+Rico.Color.prototype = {
+/**
+ * @class Methods to manipulate color values.
+ * @constructs
+ * @param red integer (0-255)
+ * @param green integer (0-255)
+ * @param blue integer (0-255)
+ */
+   initialize: function(red, green, blue) {
+      this.rgb = { r: red, g : green, b : blue };
+   },
+
+   setRed: function(r) {
+      this.rgb.r = r;
+   },
+
+   setGreen: function(g) {
+      this.rgb.g = g;
+   },
+
+   setBlue: function(b) {
+      this.rgb.b = b;
+   },
+
+   setHue: function(h) {
+
+      // get an HSB model, and set the new hue...
+      var hsb = this.asHSB();
+      hsb.h = h;
+
+      // convert back to RGB...
+      this.rgb = Rico.Color.HSBtoRGB(hsb.h, hsb.s, hsb.b);
+   },
+
+   setSaturation: function(s) {
+      // get an HSB model, and set the new hue...
+      var hsb = this.asHSB();
+      hsb.s = s;
+
+      // convert back to RGB and set values...
+      this.rgb = Rico.Color.HSBtoRGB(hsb.h, hsb.s, hsb.b);
+   },
+
+   setBrightness: function(b) {
+      // get an HSB model, and set the new hue...
+      var hsb = this.asHSB();
+      hsb.b = b;
+
+      // convert back to RGB and set values...
+      this.rgb = Rico.Color.HSBtoRGB( hsb.h, hsb.s, hsb.b );
+   },
+
+   darken: function(percent) {
+      var hsb  = this.asHSB();
+      this.rgb = Rico.Color.HSBtoRGB(hsb.h, hsb.s, Math.max(hsb.b - percent,0));
+   },
+
+   brighten: function(percent) {
+      var hsb  = this.asHSB();
+      this.rgb = Rico.Color.HSBtoRGB(hsb.h, hsb.s, Math.min(hsb.b + percent,1));
+   },
+
+   blend: function(other) {
+      this.rgb.r = Math.floor((this.rgb.r + other.rgb.r)/2);
+      this.rgb.g = Math.floor((this.rgb.g + other.rgb.g)/2);
+      this.rgb.b = Math.floor((this.rgb.b + other.rgb.b)/2);
+   },
+
+   isBright: function() {
+      var hsb = this.asHSB();
+      return this.asHSB().b > 0.5;
+   },
+
+   isDark: function() {
+      return ! this.isBright();
+   },
+
+   asRGB: function() {
+      return "rgb(" + this.rgb.r + "," + this.rgb.g + "," + this.rgb.b + ")";
+   },
+
+   asHex: function() {
+      return "#" + Rico.toColorPart(this.rgb.r) + Rico.toColorPart(this.rgb.g) + Rico.toColorPart(this.rgb.b);
+   },
+
+   asHSB: function() {
+      return Rico.Color.RGBtoHSB(this.rgb.r, this.rgb.g, this.rgb.b);
+   },
+
+   toString: function() {
+      return this.asHex();
+   }
+
+};
+
+/**
+ * Factory method for creating a color from an RGB string
+ * @param hexCode a 3 or 6 digit hex string, optionally preceded by a # symbol
+ * @returns a Rico.Color object
+ */
+Rico.Color.createFromHex = function(hexCode) {
+  if(hexCode.length==4) {
+    var shortHexCode = hexCode;
+    hexCode = '#';
+    for(var i=1;i<4;i++)
+      hexCode += (shortHexCode.charAt(i) + shortHexCode.charAt(i));
+  }
+  if ( hexCode.indexOf('#') == 0 )
+    hexCode = hexCode.substring(1);
+  if (!hexCode.match(/^[0-9A-Fa-f]{6}$/)) return null;
+  var red   = hexCode.substring(0,2);
+  var green = hexCode.substring(2,4);
+  var blue  = hexCode.substring(4,6);
+  return new Rico.Color( parseInt(red,16), parseInt(green,16), parseInt(blue,16) );
+};
+
+/**
+ * Retrieves the background color of an HTML element
+ * @param elem the DOM element whose background color should be retreived
+ * @returns a Rico.Color object
+ */
+Rico.Color.createColorFromBackground = function(elem) {
+
+   if (!elem.style) return new Rico.Color(255,255,255);
+   var actualColor = Rico.getStyle(elem, "background-color");
+
+   // if color is tranparent, check parent
+   // Safari returns "rgba(0, 0, 0, 0)", which means transparent
+   if ( actualColor.match(/^(transparent|rgba\(0,\s*0,\s*0,\s*0\))$/i) && elem.parentNode )
+      return Rico.Color.createColorFromBackground(elem.parentNode);
+
+   if (actualColor == null) return new Rico.Color(255,255,255);
+
+   if ( actualColor.indexOf("rgb(") == 0 ) {
+      var colors = actualColor.substring(4, actualColor.length - 1 );
+      var colorArray = colors.split(",");
+      return new Rico.Color( parseInt( colorArray[0],10 ),
+                             parseInt( colorArray[1],10 ),
+                             parseInt( colorArray[2],10 )  );
+
+   }
+   else if ( actualColor.indexOf("#") == 0 ) {
+      return Rico.Color.createFromHex(actualColor);
+   }
+   else
+      return new Rico.Color(255,255,255);
+};
+
+/**
+ * Converts hue/saturation/brightness to RGB
+ * @returns a 3-element object: r=red, g=green, b=blue.
+ */
+Rico.Color.HSBtoRGB = function(hue, saturation, brightness) {
+
+  var red   = 0;
+  var green = 0;
+  var blue  = 0;
+
+  if (saturation == 0) {
+     red = parseInt(brightness * 255.0 + 0.5,10);
+     green = red;
+     blue = red;
+  }
+  else {
+      var h = (hue - Math.floor(hue)) * 6.0;
+      var f = h - Math.floor(h);
+      var p = brightness * (1.0 - saturation);
+      var q = brightness * (1.0 - saturation * f);
+      var t = brightness * (1.0 - (saturation * (1.0 - f)));
+
+      switch (parseInt(h,10)) {
+         case 0:
+            red   = (brightness * 255.0 + 0.5);
+            green = (t * 255.0 + 0.5);
+            blue  = (p * 255.0 + 0.5);
+            break;
+         case 1:
+            red   = (q * 255.0 + 0.5);
+            green = (brightness * 255.0 + 0.5);
+            blue  = (p * 255.0 + 0.5);
+            break;
+         case 2:
+            red   = (p * 255.0 + 0.5);
+            green = (brightness * 255.0 + 0.5);
+            blue  = (t * 255.0 + 0.5);
+            break;
+         case 3:
+            red   = (p * 255.0 + 0.5);
+            green = (q * 255.0 + 0.5);
+            blue  = (brightness * 255.0 + 0.5);
+            break;
+         case 4:
+            red   = (t * 255.0 + 0.5);
+            green = (p * 255.0 + 0.5);
+            blue  = (brightness * 255.0 + 0.5);
+            break;
+          case 5:
+            red   = (brightness * 255.0 + 0.5);
+            green = (p * 255.0 + 0.5);
+            blue  = (q * 255.0 + 0.5);
+            break;
+      }
+  }
+
+   return { r : parseInt(red,10), g : parseInt(green,10) , b : parseInt(blue,10) };
+};
+
+/**
+ * Converts RGB value to hue/saturation/brightness
+ * @param r integer (0-255)
+ * @param g integer (0-255)
+ * @param b integer (0-255)
+ * @returns a 3-element object: h=hue, s=saturation, b=brightness.
+ * (unlike some HSB documentation which states hue should be a value 0-360, this routine returns hue values from 0 to 1.0)
+ */
+Rico.Color.RGBtoHSB = function(r, g, b) {
+
+   var hue;
+   var saturation;
+   var brightness;
+
+   var cmax = (r > g) ? r : g;
+   if (b > cmax)
+      cmax = b;
+
+   var cmin = (r < g) ? r : g;
+   if (b < cmin)
+      cmin = b;
+
+   brightness = cmax / 255.0;
+   if (cmax != 0)
+      saturation = (cmax - cmin)/cmax;
+   else
+      saturation = 0;
+
+   if (saturation == 0)
+      hue = 0;
+   else {
+      var redc   = (cmax - r)/(cmax - cmin);
+      var greenc = (cmax - g)/(cmax - cmin);
+      var bluec  = (cmax - b)/(cmax - cmin);
+
+      if (r == cmax)
+         hue = bluec - greenc;
+      else if (g == cmax)
+         hue = 2.0 + redc - bluec;
+      else
+         hue = 4.0 + greenc - redc;
+
+      hue = hue / 6.0;
+      if (hue < 0)
+         hue = hue + 1.0;
+   }
+
+   return { h : hue, s : saturation, b : brightness };
+};
+
+/**
+ * Returns a new XML document object
+ */
+Rico.createXmlDocument = function() {
+  if (document.implementation && document.implementation.createDocument) {
+    var doc = document.implementation.createDocument("", "", null);
+    // some older versions of Moz did not support the readyState property
+    // and the onreadystate event so we patch it!
+    if (doc.readyState == null) {
+      doc.readyState = 1;
+      doc.addEventListener("load", function () {
+        doc.readyState = 4;
+        if (typeof doc.onreadystatechange == "function") {
+          doc.onreadystatechange();
+        }
+      }, false);
+    }
+    return doc;
+  }
+
+  if (window.ActiveXObject)
+      return Rico.tryFunctions(
+        function() { return new ActiveXObject('MSXML2.DomDocument');   },
+        function() { return new ActiveXObject('Microsoft.DomDocument');},
+        function() { return new ActiveXObject('MSXML.DomDocument');    },
+        function() { return new ActiveXObject('MSXML3.DomDocument');   }
+      ) || false;
+  return null;
+}
index 0a835da..93cb6f0 100644 (file)
@@ -17,21 +17,23 @@ class LoadRicoClient
 \r
 ' Properties\r
 \r
-Public jsDir     ' directory containing Rico's javascript files\r
-Public cssDir    ' directory containing Rico's css files\r
-Public imgDir    ' directory containing Rico's image files\r
-Public transDir\r
-Public ricoLib,ricoTheme,ricoLogging\r
+Public jsDir      ' directory containing Rico's javascript files\r
+Public cssDir     ' directory containing Rico's css files\r
+Public imgDir     ' directory containing Rico's image files\r
+Public imgHeading ' background image for grid headings and window titles, used in grayedout theme\r
+Public transDir   ' directory containing ricoLocale_xx files, set to jsDir/translations when jsDir is set\r
+Public ricoLib    ' base library - prototype, jquery, etc\r
+Public ricoLogging       ' enable console logging?\r
 Public grid_striping     ' apply row striping to LiveGrids?\r
 Public LoadBaseLib       ' load base Javascript library (prototype, jQuery, etc)?\r
-Public jQuery_theme_path\r
-Public SupportedLangs    ' comma-separated list of 2-character codes, representing the files that can be found in js/translations directory\r
+Public jQuery_theme_path ' url to jqueryui themes\r
+Public SupportedLangs    ' comma-separated list of 2-character codes, representing the files that can be found in js/translations directory. Defaults to ones delivered with Rico.\r
 \r
 ' Constructor\r
 \r
 Private Sub Class_Initialize\r
   ricoLib="proto_min"         ' default base library\r
-  ricoTheme="j-ui-lightness"  ' jquery themes start with j-, rico themes start with r-\r
+  ricoTheme="j-ui-lightness"\r
   ricoLogging=false\r
   grid_striping = true\r
   LoadBaseLib = true\r
@@ -40,6 +42,21 @@ Private Sub Class_Initialize
 End Sub\r
 \r
 \r
+' jquery themes start with j-, rico themes start with r-\r
+Private m_ricoTheme         \r
+Public Property Get ricoTheme()\r
+  ricoTheme = m_ricoTheme\r
+End Property\r
+Public Property Let ricoTheme(theme)\r
+  Dim prefix\r
+  prefix=left(theme,2)\r
+  if prefix="r-" or prefix="j-" then\r
+    m_ricoTheme = theme\r
+    if theme="r-grayedout" then imgHeading="grayedout.gif"\r
+  end if\r
+End Property \r
+\r
+\r
 Public Sub SetRicoClientPaths(ByVal jsPath, cssPath, imgPath)\r
   jsDir = jsPath\r
   cssDir= cssPath\r
@@ -80,8 +97,9 @@ sub SetConfig()
   Response.Write vbLf & "Rico_CONFIG = {"\r
   if ricoLogging then Response.Write vbLf & "enableLogging: true,"\r
   Response.Write vbLf & "jsDir: '" & jsDir & "',"\r
-  Response.Write vbLf & "cssDir: '" & cssDir & "',"\r
-  Response.Write vbLf & "imgDir: '" & imgDir & "'"\r
+  if not IsEmpty(imgHeading) then Response.Write vbLf & "imgHeading: '" & imgDir & imgHeading & "',"\r
+  Response.Write vbLf & "imgResize: '" & imgDir & "resize.gif',"\r
+  Response.Write vbLf & "imgIcons: '" & imgDir & "ricoIcons.gif'"\r
   Response.Write vbLf & "};"\r
   Response.Write vbLf & "</script>"\r
 end sub\r
index 80f782d..b4ed29a 100644 (file)
@@ -647,7 +647,6 @@ Private sub TableDisplay()
   response.write vbLf & "<div id='" & gridID & "'></div>"\r
 \r
   response.write vbLf & "<script type='text/javascript'>"\r
-  response.write vbLf & "Rico.acceptLanguage('" & Request.ServerVariables("HTTP_ACCEPT_LANGUAGE") & "');"\r
   response.write vbLf & "var " & gridID & " = {};"\r
   response.write vbLf & optionsVar & " = {"\r
   for each o in options\r
index b081e3d..f722156 100644 (file)
@@ -653,7 +653,6 @@ class TableEditClass {
     echo "</p>";\r
     echo "\n<div id='".$this->gridID."'></div>";\r
     echo "\n<script type='text/javascript'>";\r
-    echo "\nRico.acceptLanguage('" . $_SERVER["HTTP_ACCEPT_LANGUAGE"] . "');";\r
     echo "\nvar ".$this->gridID." = {};";\r
     echo "\n".$this->optionsVar." = {";\r
     foreach ($this->options as $o => $value) {\r
index 4a02c77..4a4ae12 100644 (file)
@@ -70,6 +70,11 @@ div.ricoLG_selection {
   border-style:none;\r
 }\r
 \r
+.ricoLG_FilterRow  div.ricoLG_cell {\r
+  height:1.9em !important;\r
+  line-height: 1.9em !important;\r
+}\r
+\r
 div.ricoLG_messageDiv {\r
   border:2px solid #523A0B;\r
   background-color:#ffffee;\r
index 05916ee..b31a553 100644 (file)
@@ -14,7 +14,6 @@ div.ricoLG_outerDiv div.ricoLG_cell, div.ricoLG_outerDiv div.ricoLG_cell *, div.
 \r
 tr.ricoLG_hdg th, tr.ricoLG_hdg td, table.ricoLiveGrid thead td, table.ricoLiveGrid thead th {\r
   background-color: #FFF !important;\r
-  background: url(../images/grayedout.gif) #FFF repeat-x scroll center left;\r
 }\r
 \r
 .ricoLG_bottom div.ricoLG_cell, table.ricoLG_top, table.ricoLG_top thead th, table.ricoLG_top thead td, table.ricoLiveGrid td, table.ricoLiveGrid th, .ricoLG_bottom th, .ricoLG_bottom td {\r
@@ -76,7 +75,7 @@ caption {
   background-color: #FCFCFC;\r
 }\r
 .ricoTitle  {\r
-  background: url(../images/grayedout.gif) #FFF repeat-x scroll center left;\r
+  background-color: #FFF !important;\r
   border-bottom: 1px solid #CCC;\r
   color: #000;\r
   font-weight: bold;\r
@@ -106,7 +105,7 @@ caption {
 }\r
 \r
 .Rico_accTitle {\r
-  background: url(../images/grayedout.gif) #FFF repeat-x scroll center left;\r
+  background-color: #FFF !important;\r
   font-size: 11px;\r
   height: 1.7em;\r
   font-weight : normal;\r
@@ -145,13 +144,13 @@ div.ricoCalContainer tfoot td {
   color:#FFFFFF;\r
 }\r
 span.Rico_leftArrow {\r
-  background: url(../images/left_w.gif) no-repeat bottom left;\r
+  background-position: -10px -1077px;\r
 }\r
 span.Rico_rightArrow {\r
-  background: url(../images/right_w.gif) no-repeat bottom left;\r
+  background-position: -10px -1139px;\r
 }\r
-div.ricoCalContainer span.RicoClose {\r
-  background: url(../images/close_w.gif) no-repeat top right;\r
+.RicoCalHeading span.RicoClose {\r
+  background-position: -10px -155px;\r
 }\r
 div.ricoCalContainer table.ricoCalTab thead th a.hover {\r
   background-color: #AAA !important;\r
index cb14a1d..f876431 100644 (file)
@@ -56,11 +56,9 @@ div.ricoLG_highlightDiv {
   line-height: 1.6em;\r
   white-space: nowrap;\r
 }\r
-\r
-.ricoLG_FilterRow  div.ricoLG_cell {\r
-  vertical-align: middle;\r
+.ricoLG_FilterRow  div.ricoLG_cell * {\r
+  vertical-align: top;\r
 }\r
-\r
 .ricoLG_FilterRow select {\r
   width:100%; \r
 }\r
@@ -110,36 +108,6 @@ table.ricoLG_top thead div.ricoLG_Resize {
   cursor:e-resize;\r
 }\r
 \r
-.ricoLG_HdrIcon {\r
-  padding-left:2px;\r
-  padding-right:2px;\r
-}\r
-\r
-.ricoLG_sortAsc {\r
-  background: url(../images/sort_asc.gif) no-repeat center center;\r
-  width: 11px;\r
-  height: 11px;\r
-}\r
-\r
-.ricoLG_sortDesc {\r
-  background: url(../images/sort_desc.gif) no-repeat center center;\r
-  width: 11px;\r
-  height: 11px;\r
-}\r
-\r
-.ricoClearNative {\r
-  background: url(../images/delete.gif) no-repeat center center;\r
-  width: 17px;\r
-  height: 17px;\r
-  margin-top: -8px;\r
-}\r
-\r
-.ricoClear {\r
-  display:-moz-inline-box;\r
-  display:inline-block;\r
-  cursor: pointer;\r
-}\r
-\r
 .ricoLG_bottom div.ricoLG_cell {\r
   border-style: solid;\r
   border-color: silver;\r
@@ -216,14 +184,10 @@ span.ricoSessionTimer {
   color:white;\r
 }\r
 \r
-table.ricoLG_top thead input {\r
+table.ricoLG_top thead option, table.ricoLG_top thead select, table.ricoLG_top thead input {\r
   font-weight: normal;\r
   font-size: 8pt !important;\r
-}\r
-\r
-table.ricoLG_top thead option, table.ricoLG_top thead select {\r
-  font-weight: normal;\r
-  font-size: 9pt !important;\r
+  padding: 1px;\r
 }\r
 \r
 /* grid column chooser */\r
@@ -449,28 +413,6 @@ a.RicoButton {
   width: auto !important;\r
 }\r
 \r
-.RicoCloseAnchor {\r
-  cursor: pointer;\r
-  position: absolute;\r
-  top: 1px;\r
-  right: 2px;\r
-  border: 0px none !important;\r
-}\r
-\r
-span.RicoCloseW {\r
-  display: block;\r
-  height: 15px;\r
-  width: 16px;\r
-  background: url(../images/close_w.gif) no-repeat top right;\r
-}\r
-\r
-span.RicoCloseB, span.RicoClose {\r
-  display: block;\r
-  height: 15px;\r
-  width: 16px;\r
-  background: url(../images/close_b.gif) no-repeat top right;\r
-}\r
-\r
 a.RicoButtonAnchor {\r
   padding: 1px 0;\r
   margin: 1px;\r
@@ -494,20 +436,6 @@ a.RicoButtonAnchorNative:hover {
   background-color: #CCC;\r
 }\r
 \r
-span.RicoCheckmark {\r
-  display: block;\r
-  height: 17px;\r
-  width: 17px;\r
-  background: url(../images/checkmark.gif) no-repeat top right;\r
-}\r
-\r
-span.RicoCancel {\r
-  display: block;\r
-  height: 17px;\r
-  width: 17px;\r
-  background: url(../images/delete.gif) no-repeat top right;\r
-}\r
-\r
 * html .ui-dialog-titlebar .ui-dialog-titlebar-close {\r
   top: 16px;   /* required by IE6 */\r
 }\r
@@ -620,7 +548,7 @@ span.RicoCalYearHeading {
 }\r
 \r
 span.Rico_leftArrow {\r
-  background: url(../images/left_b.gif) no-repeat bottom right;\r
+  background-position: -10px -1046px;\r
   position: absolute;\r
   top: 4px;\r
   left: 0px;\r
@@ -637,7 +565,7 @@ a.Rico_leftArrow {
 }\r
 \r
 span.Rico_rightArrow {\r
-  background: url(../images/right_b.gif) no-repeat bottom left;\r
+  background-position: -10px -1108px;\r
   position: absolute;\r
   top: 4px;\r
   right: 0px;\r
@@ -855,3 +783,209 @@ div.ricoCorner div {
   height: 1%;\r
   position: relative;\r
 }\r
+\r
+.rico-calarrow {\r
+  width: 37px;\r
+  height: 18px;\r
+  display: inline-block;\r
+  background-position: -10px -10px;\r
+}\r
+.rico-calendaricon {\r
+  width: 17px;\r
+  height: 17px;\r
+  display: inline-block;\r
+  background-position: -10px -48px;\r
+}\r
+.rico-close-b {\r
+  background-position: -10px -122px;\r
+}\r
+.rico-close-w {\r
+  background-position: -10px -155px;\r
+}\r
+.rico-delete-b {\r
+  background-position: -10px -188px;\r
+}\r
+.rico-delete-w {\r
+  background-position: -10px -225px;\r
+}\r
+.rico-doc {\r
+  background-position: -10px -262px;\r
+}\r
+.rico-dotbutton {\r
+  width: 17px;\r
+  height: 17px;\r
+  display: inline-block;\r
+  background-position: -10px -304px;\r
+}\r
+.rico-drop {\r
+  background-position: -10px -341px;\r
+}\r
+.rico-folderclosed {\r
+  width: 24px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -403px;\r
+}\r
+.rico-folderopen {\r
+  width: 24px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -445px;\r
+}\r
+.rico-info {\r
+  width: 15px;\r
+  height: 15px;\r
+  display: inline-block;\r
+  background-position: -10px -487px;\r
+}\r
+.rico-link {\r
+  background-position: -10px -522px;\r
+}\r
+.rico-tree-m {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -564px;\r
+}\r
+.rico-tree-node {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -606px;\r
+}\r
+.rico-tree-nodelast {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -648px;\r
+}\r
+.rico-tree-nodeline {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -690px;\r
+}\r
+.rico-tree-nodem {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -732px;\r
+}\r
+.rico-tree-nodemlast {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -774px;\r
+}\r
+.rico-tree-nodep {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -816px;\r
+}\r
+.rico-tree-nodeplast {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -858px;\r
+}\r
+.rico-tree-p {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background-position: -10px -900px;\r
+}\r
+.rico-tree-nodeblank {\r
+  width: 16px;\r
+  height: 22px;\r
+  display: block;\r
+  background: none !important;\r
+}\r
+.rico-removeFilter {\r
+  background-position: -10px -942px;\r
+}\r
+.rico-FilterCollapse {\r
+  background-position: -10px -974px;\r
+}\r
+.rico-FilterExpand {\r
+  background-position: -10px -1010px;\r
+}\r
+.rico-left-b {\r
+  background-position: -10px -1046px;\r
+  width: 10px;\r
+  height: 11px;\r
+}\r
+.rico-left-w {\r
+  background-position: -10px -1077px;\r
+  width: 10px;\r
+  height: 11px;\r
+}\r
+.rico-right-b {\r
+  background-position: -10px -1108px;\r
+  width: 10px;\r
+  height: 11px;\r
+}\r
+.rico-right-w {\r
+  background-position: -10px -1139px;\r
+  width: 10px;\r
+  height: 11px;\r
+}\r
+\r
+.ricoLG_filterCol {\r
+  margin: 0px 2px;\r
+  width: 13px;\r
+  height: 12px;\r
+  background-position: -10px -371px;\r
+}\r
+.ricoLG_sortAsc {\r
+  margin: 0px 2px;\r
+  width: 11px;\r
+  height: 11px;\r
+  background-position: -10px -1168px;\r
+}\r
+.ricoLG_sortDesc {\r
+  margin: 0px 2px;\r
+  width: 11px;\r
+  height: 11px;\r
+  background-position: -10px -1198px;\r
+}\r
+\r
+.ricoClearNative {\r
+  width: 17px;\r
+  height: 17px;\r
+  background-position: -10px -186px;\r
+}\r
+span.ricoClear {\r
+  display:-moz-inline-box;\r
+  display:inline-block;\r
+  cursor: pointer;\r
+}\r
+\r
+span.RicoCheckmark {\r
+  display: block;\r
+  height: 17px;\r
+  width: 17px;\r
+  background-position: -10px -85px;\r
+}\r
+\r
+span.RicoCancel {\r
+  display: block;\r
+  height: 17px;\r
+  width: 17px;\r
+  background-position: -10px -188px;\r
+}\r
+\r
+\r
+.RicoCloseAnchor {\r
+  cursor: pointer;\r
+  position: absolute;\r
+  top: 1px;\r
+  right: 2px;\r
+  border: 0px none !important;\r
+}\r
+span.RicoClose {\r
+  display: block;\r
+  height: 15px;\r
+  width: 16px;\r
+  background-position: -10px -122px;\r
+}\r
index 9dc1e54..8d8dbb5 100644 (file)
@@ -38,7 +38,7 @@ div.ricoLG_selection {
   color: #ffffff;
 }
 .ricoLG_top .ricoClearNative {
-  background: url(../images/delete_w.gif) no-repeat center center;
+  background-position: -10px -223px;
 }
 
 .ricoWindow {
@@ -135,13 +135,13 @@ div.ricoCalContainer .ricoCalMenu {
   border-right:1px solid #2C5755;
 }
 span.Rico_leftArrow {
-  background: url(../images/left_w.gif) no-repeat bottom left;
+  background-position: -10px -1077px;
 }
 span.Rico_rightArrow {
-  background: url(../images/right_w.gif) no-repeat bottom left;
+  background-position: -10px -1139px;
 }
 span.RicoClose {
-  background: url(../images/close_w.gif) no-repeat top right;
+  background-position: -10px -155px;
 }
 div.ricoCalContainer table.ricoCalTab thead th a.hover {
   background-color: #4C7775 !important;
index 51ae738..aec9247 100644 (file)
@@ -45,7 +45,7 @@ div.ricoLG_outerDiv a:hover {
 }\r
 \r
 div.ricoLG_outerDiv tbody a:visited {\r
-       color:#444;\r
+  color:#444;\r
 }\r
 \r
 div.ricoLG_messageDiv {\r
@@ -55,14 +55,14 @@ div.ricoLG_messageDiv {
 }\r
 \r
 div.ricoLG_selection {\r
-       background-color: #923106;\r
-       color: #ffffff;\r
+  background-color: #923106;\r
+  color: #ffffff;\r
 }\r
 \r
 .ricoLG_top .ricoClearNative {\r
-  background: url(../images/delete_w.gif) no-repeat center center;\r
+  background-position: -10px -223px;\r
 }\r
-       \r
+\r
 a.RicoButton {\r
   border: 1px solid #923106;\r
   color: #923106;\r
@@ -72,10 +72,10 @@ a.RicoButton.hover {
 }\r
 \r
 caption {\r
-       text-align: left;\r
-       font-size: 100%;\r
-       padding: .75em;\r
-       color: #000;\r
+  text-align: left;\r
+  font-size: 100%;\r
+  padding: .75em;\r
+  color: #000;\r
 }\r
 \r
 .ricoWindow {\r
@@ -162,13 +162,13 @@ div.ricoCalContainer .ricoCalMenu {
   border-right:1px solid #923106;\r
 }\r
 span.Rico_leftArrow {\r
-  background: url(../images/left_w.gif) no-repeat bottom left;\r
+  background-position: -10px -1077px;\r
 }\r
 span.Rico_rightArrow {\r
-  background: url(../images/right_w.gif) no-repeat bottom left;\r
+  background-position: -10px -1139px;\r
 }\r
 span.RicoClose {\r
-  background: url(../images/close_w.gif) no-repeat top right;\r
+  background-position: -10px -155px;\r
 }\r
 div.ricoCalContainer table.ricoCalTab thead th a.hover {\r
   background-color: #c26136 !important;\r
diff --git a/ricoClient/images/calarrow.png b/ricoClient/images/calarrow.png
deleted file mode 100644 (file)
index acdcfed..0000000
Binary files a/ricoClient/images/calarrow.png and /dev/null differ
diff --git a/ricoClient/images/calendaricon.gif b/ricoClient/images/calendaricon.gif
deleted file mode 100644 (file)
index abdf6ac..0000000
Binary files a/ricoClient/images/calendaricon.gif and /dev/null differ
diff --git a/ricoClient/images/checkmark.gif b/ricoClient/images/checkmark.gif
deleted file mode 100644 (file)
index 88a19a1..0000000
Binary files a/ricoClient/images/checkmark.gif and /dev/null differ
diff --git a/ricoClient/images/close_b.gif b/ricoClient/images/close_b.gif
deleted file mode 100644 (file)
index cf32976..0000000
Binary files a/ricoClient/images/close_b.gif and /dev/null differ
diff --git a/ricoClient/images/close_w.gif b/ricoClient/images/close_w.gif
deleted file mode 100644 (file)
index 3ebc825..0000000
Binary files a/ricoClient/images/close_w.gif and /dev/null differ
diff --git a/ricoClient/images/delete.gif b/ricoClient/images/delete.gif
deleted file mode 100644 (file)
index d5c6057..0000000
Binary files a/ricoClient/images/delete.gif and /dev/null differ
diff --git a/ricoClient/images/delete_w.gif b/ricoClient/images/delete_w.gif
deleted file mode 100644 (file)
index 86acb65..0000000
Binary files a/ricoClient/images/delete_w.gif and /dev/null differ
diff --git a/ricoClient/images/divider.gif b/ricoClient/images/divider.gif
deleted file mode 100644 (file)
index d9863d4..0000000
Binary files a/ricoClient/images/divider.gif and /dev/null differ
diff --git a/ricoClient/images/doc.gif b/ricoClient/images/doc.gif
deleted file mode 100644 (file)
index 2ff706c..0000000
Binary files a/ricoClient/images/doc.gif and /dev/null differ
diff --git a/ricoClient/images/dotbutton.gif b/ricoClient/images/dotbutton.gif
deleted file mode 100644 (file)
index 20c85cc..0000000
Binary files a/ricoClient/images/dotbutton.gif and /dev/null differ
diff --git a/ricoClient/images/drop.gif b/ricoClient/images/drop.gif
deleted file mode 100644 (file)
index afce892..0000000
Binary files a/ricoClient/images/drop.gif and /dev/null differ
diff --git a/ricoClient/images/filtercol.gif b/ricoClient/images/filtercol.gif
deleted file mode 100644 (file)
index 2e93828..0000000
Binary files a/ricoClient/images/filtercol.gif and /dev/null differ
diff --git a/ricoClient/images/folderclosed.gif b/ricoClient/images/folderclosed.gif
deleted file mode 100644 (file)
index 73594c3..0000000
Binary files a/ricoClient/images/folderclosed.gif and /dev/null differ
diff --git a/ricoClient/images/folderopen.gif b/ricoClient/images/folderopen.gif
deleted file mode 100644 (file)
index f44db49..0000000
Binary files a/ricoClient/images/folderopen.gif and /dev/null differ
diff --git a/ricoClient/images/info_icon.gif b/ricoClient/images/info_icon.gif
deleted file mode 100644 (file)
index f0d2107..0000000
Binary files a/ricoClient/images/info_icon.gif and /dev/null differ
diff --git a/ricoClient/images/left_b.gif b/ricoClient/images/left_b.gif
deleted file mode 100644 (file)
index f652075..0000000
Binary files a/ricoClient/images/left_b.gif and /dev/null differ
diff --git a/ricoClient/images/left_w.gif b/ricoClient/images/left_w.gif
deleted file mode 100644 (file)
index ed221f7..0000000
Binary files a/ricoClient/images/left_w.gif and /dev/null differ
diff --git a/ricoClient/images/link.gif b/ricoClient/images/link.gif
deleted file mode 100644 (file)
index 35b0fe7..0000000
Binary files a/ricoClient/images/link.gif and /dev/null differ
diff --git a/ricoClient/images/m.gif b/ricoClient/images/m.gif
deleted file mode 100644 (file)
index 4baccbb..0000000
Binary files a/ricoClient/images/m.gif and /dev/null differ
diff --git a/ricoClient/images/node.gif b/ricoClient/images/node.gif
deleted file mode 100644 (file)
index f90699e..0000000
Binary files a/ricoClient/images/node.gif and /dev/null differ
diff --git a/ricoClient/images/nodeblank.gif b/ricoClient/images/nodeblank.gif
deleted file mode 100644 (file)
index fc9cead..0000000
Binary files a/ricoClient/images/nodeblank.gif and /dev/null differ
diff --git a/ricoClient/images/nodelast.gif b/ricoClient/images/nodelast.gif
deleted file mode 100644 (file)
index 49f794b..0000000
Binary files a/ricoClient/images/nodelast.gif and /dev/null differ
diff --git a/ricoClient/images/nodeline.gif b/ricoClient/images/nodeline.gif
deleted file mode 100644 (file)
index 8e62441..0000000
Binary files a/ricoClient/images/nodeline.gif and /dev/null differ
diff --git a/ricoClient/images/nodem.gif b/ricoClient/images/nodem.gif
deleted file mode 100644 (file)
index fcc2d37..0000000
Binary files a/ricoClient/images/nodem.gif and /dev/null differ
diff --git a/ricoClient/images/nodemlast.gif b/ricoClient/images/nodemlast.gif
deleted file mode 100644 (file)
index 11ae43a..0000000
Binary files a/ricoClient/images/nodemlast.gif and /dev/null differ
diff --git a/ricoClient/images/nodep.gif b/ricoClient/images/nodep.gif
deleted file mode 100644 (file)
index 5b68013..0000000
Binary files a/ricoClient/images/nodep.gif and /dev/null differ
diff --git a/ricoClient/images/nodeplast.gif b/ricoClient/images/nodeplast.gif
deleted file mode 100644 (file)
index b87f003..0000000
Binary files a/ricoClient/images/nodeplast.gif and /dev/null differ
diff --git a/ricoClient/images/p.gif b/ricoClient/images/p.gif
deleted file mode 100644 (file)
index 3b49d67..0000000
Binary files a/ricoClient/images/p.gif and /dev/null differ
diff --git a/ricoClient/images/removeFilter.gif b/ricoClient/images/removeFilter.gif
deleted file mode 100644 (file)
index c46c54a..0000000
Binary files a/ricoClient/images/removeFilter.gif and /dev/null differ
diff --git a/ricoClient/images/ricoIcons.gif b/ricoClient/images/ricoIcons.gif
new file mode 100644 (file)
index 0000000..cc67808
Binary files /dev/null and b/ricoClient/images/ricoIcons.gif differ
diff --git a/ricoClient/images/right_b.gif b/ricoClient/images/right_b.gif
deleted file mode 100644 (file)
index 5517f2b..0000000
Binary files a/ricoClient/images/right_b.gif and /dev/null differ
diff --git a/ricoClient/images/right_w.gif b/ricoClient/images/right_w.gif
deleted file mode 100644 (file)
index cb7b214..0000000
Binary files a/ricoClient/images/right_w.gif and /dev/null differ
diff --git a/ricoClient/images/sort_asc.gif b/ricoClient/images/sort_asc.gif
deleted file mode 100644 (file)
index 6330232..0000000
Binary files a/ricoClient/images/sort_asc.gif and /dev/null differ
diff --git a/ricoClient/images/sort_desc.gif b/ricoClient/images/sort_desc.gif
deleted file mode 100644 (file)
index b3a681c..0000000
Binary files a/ricoClient/images/sort_desc.gif and /dev/null differ
diff --git a/ricoClient/images/tableFilterCollapse.gif b/ricoClient/images/tableFilterCollapse.gif
deleted file mode 100644 (file)
index 3209312..0000000
Binary files a/ricoClient/images/tableFilterCollapse.gif and /dev/null differ
diff --git a/ricoClient/images/tableFilterExpand.gif b/ricoClient/images/tableFilterExpand.gif
deleted file mode 100644 (file)
index 84ae691..0000000
Binary files a/ricoClient/images/tableFilterExpand.gif and /dev/null differ
diff --git a/ricoClient/js/baselibs/README.TXT b/ricoClient/js/baselibs/README.TXT
deleted file mode 100644 (file)
index 32689cb..0000000
+++ /dev/null
@@ -1,5 +0,0 @@
-The files in this directory are included in the Rico distribution\r
-for the purpose of demonstrating the examples. Each of these libraries \r
-has licensing terms that are different from Rico itself. Make sure you\r
-understand the terms before deploying any library into a production \r
-environment.
\ No newline at end of file
diff --git a/ricoClient/js/baselibs/glow.core-1.7.js b/ricoClient/js/baselibs/glow.core-1.7.js
deleted file mode 100644 (file)
index e6d6e0e..0000000
+++ /dev/null
@@ -1,20 +0,0 @@
-/*     
-       Copyright 2009 British Broadcasting Corporation
-
-       Licensed under the Apache License, Version 2.0 (the "License");
-       you may not use this file except in compliance with the License.
-       You may obtain a copy of the License at
-
-          http://www.apache.org/licenses/LICENSE-2.0
-
-       Unless required by applicable law or agreed to in writing, software
-       distributed under the License is distributed on an "AS IS" BASIS,
-       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-       See the License for the specific language governing permissions and
-       limitations under the License.
-*/
-(function(){var f={glow:true},b=/([$^\\\/()|?+*\[\]{}.-])/g,a=navigator.userAgent.toLowerCase(),k="1.7.3",r=0,g=[],m=0,q=[],o=0,s=false,n={VERSION:k,UID:"glow"+Math.floor(Math.random()*(1<<30)),isDomReady:window.gloader&&gloader.isReady,isReady:window.gloader&&gloader.isReady,env:function(){var u=[0,NaN],d=(/opera[\s\/]([\w\.]+)/.exec(a)||u)[1],v=d?NaN:(/msie ([\w\.]+)/.exec(a)||u)[1],x=(/rv:([\w\.]+).*gecko\//.exec(a)||u)[1],e=(/applewebkit\/([\w\.]+)/.exec(a)||u)[1],t=(/khtml\/([\w\.]+)/.exec(a)||u)[1],w=parseFloat;return{gecko:w(x),ie:w(v),opera:w(d),webkit:w(e),khtml:w(t),version:v||x||e||d||t,standardsMode:document.compatMode!="BackCompat"&&(!v||v>=6)};}(),module:function(u){var t=2,w=u.depends[0]||[],d=w.length,e=u.name,v=window.glow;if(u.library[1]!=n.VERSION){throw new Error("Cannot register "+e+": Version mismatch");}if(w[2]){for(;t<d;t++){if(!f[w[t]]){throw new Error("Module "+w[t]+" required before "+e);}}}u.builder(n);f[e]=true;return n;},ready:function(d){if(this.isReady){d();}else{q[o++]=d;}return this;},_readyBlockers:{},_addReadyBlock:function(d){if(!n._readyBlockers[d]){n._readyBlockers[d]=true;n.isReady=false;r++;}return n;},_removeReadyBlock:function(d){if(n._readyBlockers[d]){n._readyBlockers[d]=false;r--;if(!r){n.isReady=true;p();}}return n;},onDomReady:function(d){if(this.isDomReady){d();}else{g[m++]=d;}},lang:{trim:function(d){return d.replace(/^\s*((?:[\S\s]*\S)?)\s*$/,"$1");},toArray:function(e){if(e.constructor==Array){return e;}var u=[],t=0,d=e.length;for(;t<d;t++){u[t]=e[t];}return u;},apply:function(d,t){for(var e in t){d[e]=t[e];}return d;},map:function(e,x,w){if(Array.prototype.map){return Array.prototype.map.call(e,x,w||e);}if(!x.call){throw new TypeError();}var d=e.length,v=[],u=w||e,t=0;for(;t<d;t++){if(t in e){v[t]=x.call(u,e[t],t,e);}}return v;},replace:(function(){var d="g".replace(/g/,function(){return"l";})!="l",e=String.prototype.replace;return function(y,w,t){var z,v,x,u;if(!d||typeof(t)!="function"){return e.call(y,w,t);}if(!(w instanceof RegExp)){z=y.indexOf(w);return z==-1?y:e.call(y,w,t.call(null,w,z,y));}u=[];x=w.lastIndex=0;while((v=w.exec(y))!=null){z=v.index;u[u.length]=y.slice(x,z);u[u.length]=t.apply(null,v);if(w.global){x=w.lastIndex;}else{x=z+v[0].length;break;}}u[u.length]=y.slice(x);return u.join("");};})(),interpolate:function(t,w,u){var d,v,e,x;u=u||{};if(u.escapeHtml){if(!n.dom){throw new Error("glow.lang.interpolate - glow.dom is needed for escapeHtml");}x=n.dom.create("<div></div>");}if(u.delimiter==undefined){d=/\{[^{}]+\}/g;}else{v=u.delimiter.substr(0,1).replace(b,"\\$1");e=u.delimiter.substr(1,1).replace(b,"\\$1")||v;d=new RegExp(v+"[^"+v+e+"]+"+e,"g");}return t.replace(d,function(D){var A=D.slice(1,-1),C=A.split("."),B,z=0,y=C.length;if(A in w){B=w[A];}else{B=w;for(;z<y;z++){if(C[z] in B){B=B[C[z]];}else{return D;}}}if(u.escapeHtml){B=x.text(B).html();}return B;});},hasOwnProperty:{}.hasOwnProperty?function(d,e){return d.hasOwnProperty(e);}:function(x,y){var u=x[y],w=x.__proto__,t=w?w[y]:{};if(u!==t){return true;}var e=n.lang.hasOwnProperty(w,y),d=w[y]={},v=(x[y]!==d);delete w[y];if(e){w[name]=d;}return v;},extend:function(e,u,d){var t=function(){},v;t.prototype=u.prototype;v=new t();e.prototype=v;v.constructor=e;e.base=u;if(d){n.lang.apply(e.prototype,d);}},clone:function(t){var d,u,e;t=t.valueOf();if(typeof t!=="object"){return t;}else{if(t[0]||t.concat){e=[];d=t.length;while(d--){e[d]=arguments.callee(t[d]);}}else{e={};for(d in t){e[d]=arguments.callee(t[d]);}}return e;}}}},h=n.env,l=document;function c(){n.isDomReady=true;for(var d=0;d<m;d++){g[d]();}}function p(){if(s){return;}s=true;for(var d=0;d<o;){q[d]();d++;if(r){break;}}q=q.slice(d);o=o-d;s=false;}(function(){if(n.isDomReady){return;}n._addReadyBlock("glow_domReady");if(h.ie){if(typeof window.frameElement!="undefined"){l.attachEvent("onreadystatechange",function(){if(l.readyState=="complete"){l.detachEvent("onreadystatechange",arguments.callee);c();n._removeReadyBlock("glow_domReady");}});}else{(function(){try{l.documentElement.doScroll("left");}catch(u){setTimeout(arguments.callee,0);return;}c();n._removeReadyBlock("glow_domReady");})();}}else{if(n.env.webkit<525.13&&typeof l.readyState!="undefined"){var e=function(){if(/loaded|complete/.test(l.readyState)){c();n._removeReadyBlock("glow_domReady");}else{setTimeout(e,0);}};e();}else{var t=function(){if(t.fired){return;}t.fired=true;c();n._removeReadyBlock("glow_domReady");};if(l.addEventListener){l.addEventListener("DOMContentLoaded",t,false);}var d=window.onload;window.onload=function(){if(d){d();}t();};}}})();n.isSupported=!(h.ie<6||(h.gecko<1.9&&!/^1\.8\.1/.test(h.version))||h.opera<9||h.webkit<412);if(!n.isSupported){n._addReadyBlock("glow_browserSupport");}if(window.gloader){gloader.library({name:"glow",version:"1.7.3",builder:function(){return n;}});}else{if(window.glow){throw new Error("Glow global object already exists");}else{window.glow=n;}}if(n.ie){try{document.execCommand("BackgroundImageCache",false,true);}catch(j){}}})();
-/*@cc_on @*/
-/*@if (@_jscript_version > 5.5)@*/
-(window.gloader||glow).module({name:"glow.i18n",library:["glow","1.7.3"],depends:[["glow","1.7.3"]],builder:function(r){var A;var t={l:/^[a-z]$/,lv:/^[a-z]{2,3}$/,s:/^[A-Z][a-z]{3}$/,r:/^[A-Z]{2}|[0-9]{3}$/,v:/^[a-z0-9]{4,}$/};var s=1,k=2,l=4,j=8,p=s+k+l+j,f=s+l+j,x=s+k+j,c=s+j,y=s+k+l,e=s+l,d=s+k;var m={l:s,s:k,r:l,v:j},I=["l","s","r","v"],F={l:0,s:1,r:2,v:3};var C={};var b={};var o=w(document.documentElement.lang||"en")||w("en");function D(K){for(var J in t){if(t[J].test(K)){return J;}}return"";}function w(V){if(!V.split){V="";}var N=V.split("-"),Q=N.length,R=[],K={l:"",s:"",r:"",v:""},J=0,O=J,U=0,P,S;for(var M=0,T=I.length;M<T;M++){O=J;P=I[M];S=F[P];while((D(N[O]).indexOf(P)==-1)&&(O<Q)){O++;}if(O<Q){R[S]=N[O];U+=m[P];K[P]=N[O];N[O]="*";J=O;}}var L=R.join("-").replace(/-+/g,"-");if((L=="")||(L.substring(0,1)=="-")){return false;}else{return{canonical:L,mask:U,subtags:K};}}function g(K,M,J){var L;if((J&~K.mask)==0){L=K.subtags.l;if(k&J){L=L+"-"+K.subtags.s;}if(l&J){L=L+"-"+K.subtags.r;}if(j&J){L=L+"-"+K.subtags.v;}if(M(L)){return L;}}return false;}function a(K,N,J,L){var M;switch(K.mask){case f:if((M=g(K,N,f))){break;}case e:if((M=g(K,N,e))){break;}case p:if((M=g(K,N,p))){break;}case y:if((M=g(K,N,y))){break;}case x:if((M=g(K,N,x))){break;}case d:if((M=g(K,N,d))){break;}case c:if((M=g(K,N,c))){break;}case s:if((M=g(K,N,s))){break;}default:if(N("en")){M="en";}else{M=null;}}if(M==null){L();}else{J(M);}}function v(L){var J=o,K=w(L);if(K){o=K;o.next=J;}return A;}function H(){o=o.next||o;return A;}function B(){return o.canonical;}function h(M,L,Q){var J=w(L),P,O,K;if(J){P=C[J.canonical]=C[J.canonical]||{};O=P[M]=P[M]||{};K=b[M]=b[M]||{};for(var N in Q){O[N]=Q[N];K[N]=1;}}return A;}function E(K,J){var L={},S=J||{},N=b[K]||{},O=o,T,Q;function P(U){if(C[U]&&C[U][K]&&C[U][K][Q]){return true;}else{return false;}}function R(U){L[Q]=C[U][K][Q];}function M(){L[Q]="[Error! No "+K+"."+Q+" on "+O.canonical+"]";}if(S.locale!=undefined){T=w(S.locale);if(T){O=T;}}for(Q in N){a(O,P,R,M);}return L;}function G(K,L){for(var J in L){h(J,K,L[J]);}return A;}function u(L,M){var K=M||{},J=w(L);if(K.module){if(K.label){return q(J,K.module,K.label);}else{return n(J,K.module);}}else{return z(J);}return null;}function q(L,N,M){var J;function P(Q){if(C[Q]&&C[Q][N]&&C[Q][N][M]){return true;}else{return false;}}function O(Q){J=Q;}function K(){J="**error** - no negotiated value exists";}a(L,P,O,K);return J;}function n(L,O){var K=b[O]||{},N={},M;function Q(R){if(C[R]&&C[R][O]&&C[R][O][M]){return true;}else{return false;}}function P(R){N[M]=R;}function J(){N[M]="**error** - no negotiated value exists";}for(M in K){a(L,Q,P,J);}return N;}function z(K){var N={},M,L;function P(Q){if(C[Q]&&C[Q][M]&&C[Q][M][L]){return true;}else{return false;}}function O(Q){N[M][L]=Q;}function J(){N[M][L]="**error** - no negotiated value exists";}for(M in b){N[M]={};for(L in b[M]){a(K,P,O,J);}}return N;}r.i18n=A={setLocale:v,revertLocale:H,getLocale:B,addLocaleModule:h,getLocaleModule:E,addLocalePack:G,checkLocale:u};G("en",{PROPERTIES:{LANGUAGE:"English",DIR:"ltr"}});}});(window.gloader||glow).module({name:"glow.dom",library:["glow","1.7.3"],depends:[],builder:function(u){var n=u.env,k=u.lang,s={tagName:/^(\w+|\*)/,combinator:/^\s*([>]?)\s*/,classNameOrId:(n.webkit<417)?new RegExp("^([\\.#])((?:(?![\\.#\\[:\\s\\\\]).|\\\\.)+)"):/^([\.#])((?:[^\.#\[:\\\s]+|\\.)+)/},X=/([$^\\\/()|?+*\[\]{}.-])/g,B={},R={checked:"checked","class":"className",disabled:"disabled","for":"htmlFor",maxlength:"maxLength"},c={checked:true,disabled:true},ah={maxlength:function(r){return r.toString()=="2147483647"?undefined:r;}},ad=1,x="_unique"+u.UID,ai="_uniqueData"+u.UID,ag=1,L=[],I={black:0,silver:12632256,gray:8421504,white:16777215,maroon:8388608,red:16711680,purple:8388736,fuchsia:16711935,green:32768,lime:65280,olive:8421376,yellow:16776960,navy:128,blue:255,teal:32896,aqua:65535,orange:16753920},D=/height|top/,t=/^rgb\(([\d\.]+)(%?),\s*([\d\.]+)(%?),\s*([\d\.]+)(%?)/i,A=/^(?:(width|height)|(border-(top|bottom|left|right)-width))$/,C=/width|height|top$|bottom$|left$|right$|spacing$|indent$|font-size/,T,d,K,H,aa=window,l=document,V,G,w,P=l.createElement("div"),y=[1,"<table>","</table>"],ab=[0,"",""],O=n.webkit<526?[0,"","</div>",true]:[1,"b<div>","</div>"],a=[3,"<table><tbody><tr>","</tr></tbody></table>"],E={caption:y,thead:y,th:a,colgroup:y,tbody:y,tr:[2,"<table><tbody>","</tbody></table>"],td:a,tfoot:y,option:[1,"<select>","</select>"],legend:[1,"<fieldset>","</fieldset>"],link:O,script:O,style:O};if(n.ie){window.attachEvent("onunload",function(){P=null;});}u.ready(function(){V=l.body;G=l.documentElement;});(function(){var r=l.createElement("div");r.a=1;w=!!r.cloneNode(true).a;})();function af(r){for(var aj=r.firstChild;aj;aj=aj.nextSibling){if(aj.nodeType==1){return aj;}}return null;}function v(r){return new RegExp(["(^|\\s)",r.replace(X,"\\$1"),"($|\\s)"].join(""),"g");}function N(ap){var ao=[],al=(/^\s*<([^\s>]+)/.exec(ap)||[,"div"])[1],aj=E[al]||ab,am,ak,an=0;P.innerHTML=(aj[1]+ap+aj[2]);ak=P;am=aj[0];while(am--){ak=ak.lastChild;}while(ak.firstChild){ao[an++]=ak.removeChild(ak.firstChild);}ak=null;return ao;}function p(al){var ak=[],aj=0;for(;al[aj];aj++){ak[aj]=al[aj];}return ak;}function e(am,aj){for(var al=this,r=0,ak=al.length;r<ak;r++){aj.call(al[r],am.call?am.call(al[r],r):am);}return al;}if(document.all){T=function(ak,aj){var al=0,r=ak.length,am=aj.length;if(typeof aj.length=="number"){for(;al<am;al++){ak[r++]=aj[al];}}else{for(;aj[al];al++){ak[r++]=aj[al];}}};}else{T=function(ak,aj){var al=0,r=ak.length;for(;aj[al];al++){ak[r++]=aj[al];}};}function M(r){return(r.ownerDocument&&!r.ownerDocument.body)||(r.documentElement&&!r.documentElement.body);}if(n.ie){d=function(ak){if(ak.length==1){return ak;}var am=[],aj=0,al=0;for(;ak[al];al++){if(ak[al].getAttribute(x)!=ad&&ak[al].nodeType==1){am[aj++]=ak[al];}ak[al].setAttribute(x,ad);}for(al=0;ak[al];al++){ak[al].removeAttribute(x);}ad++;return am;};}else{d=function(ak){if(ak.length==1){return ak;}var am=[],aj=0,al=0;for(;ak[al];al++){if(ak[al][x]!=ad&&ak[al].nodeType==1){am[aj++]=ak[al];}ak[al][x]=ad;}ad++;return am;};}if(document.all){H=function(aj,al){var am=[],ak=0;for(;al[ak];ak++){if(aj=="*"&&al[ak].all&&!M(al[ak])){T(am,al[ak].all);}else{T(am,al[ak].getElementsByTagName(aj));}}return am;};}else{H=function(ak,am){var an=[],al=0,aj=am.length;for(;al<aj;al++){T(an,am[al].getElementsByTagName(ak));}return an;};}function b(am){var al=[],an=am.childNodes,ak=0,aj=0;for(;an[ak];ak++){if(an[ak].nodeType==1&&an[ak].nodeName!="!"){al[aj++]=an[ak];}}return al;}var U=["border-left-width","border-right-width","padding-left","padding-right"],Q=["border-top-width","border-bottom-width","padding-top","padding-bottom"];function f(ap,an){var am,aj=n.standardsMode?G:V,ao=(an=="width"),ak=ao?"Width":"Height",al;if(ap.window){am=n.webkit<522.11?(ao?ap.innerWidth:ap.innerHeight):n.webkit?(ao?V.clientWidth:ap.innerHeight):n.opera<9.5?(ao?V.clientWidth:V.clientHeight):(ao?aj.clientWidth:aj.clientHeight);}else{if(ap.getElementById){am=Math.max(V["scroll"+ak],G["scroll"+ak]);}else{al=ao?U:Q;am=ap["offset"+ak]-parseInt(Z(ap,al));}}return am;}function ae(r){if(n.ie<6){return r.document.body;}else{return r.ownerDocument.body;}}function j(al,am,ak){if(typeof am=="number"||/\d$/.test(am)){am+="px";}for(var aj=0,r=al.length;aj<r;aj++){al[aj].style[ak]=am;}}function W(r){if(r=="float"){return n.ie?"styleFloat":"cssFloat";}return k.replace(r,/-(\w)/g,function(aj,ak){return ak.toUpperCase();});}function z(ap,an){var am,ao=ap.style,al=ao.display,aj=ao.visibility,ak=ao.position;ao.visibility="hidden";ao.position="absolute";ao.display="block";if(!h(ap)){ao.position=ak;am=z(ap.parentNode,an);ao.display=al;ao.visibility=aj;}else{am=an();ao.display=al;ao.position=ak;ao.visibility=aj;}return am;}function h(r){return r.offsetWidth||r.offsetHeight;}function Z(ar,al){var ak,au=0,ap=0,ao=al.length,an=l.defaultView&&(l.defaultView.getComputedStyle(ar,null)||l.defaultView.getComputedStyle),am=ar.currentStyle,at,aq,aj=al.push||A.exec(al)||[];if(al.push){for(;ap<ao;ap++){au+=parseInt(Z(ar,al[ap]),10)||0;}return au+"px";}if(aj[1]){if(!h(ar)){return z(ar,function(){return f(ar,aj[1])+"px";});}return f(ar,aj[1])+"px";}else{if(aj[2]&&u.env.ie&&Z(ar,"border-"+aj[3]+"-style")=="none"){return"0";}else{if(an){if(typeof an=="function"){at=ar.style.display;ak=z(ar,function(){if(al=="display"){ar.style.display=at;if(!l.defaultView.getComputedStyle(ar,null)){return"none";}ar.style.display="block";}return Z(ar,al);});}else{if(n.webkit>500&&n.webkit<526&&al=="margin-right"&&an.getPropertyValue("position")!="absolute"){al="margin-left";}ak=an.getPropertyValue(al);}}else{if(am){if(al=="opacity"){aq=/alpha\(opacity=([^\)]+)\)/.exec(am.filter);return aq?String(parseInt(aq[1],10)/100):"1";}ak=String(am[W(al)]);if(/^-?[\d\.]+(?!px)[%a-z]+$/i.test(ak)&&al!="font-size"){ak=J(ar,ak,D.test(al))+"px";}}}}}if(al.indexOf("color")!=-1){ak=S(ak).toString();}else{if(ak.indexOf("url")==0){ak=ak.replace(/\"/g,"");}}return ak;}function J(ao,aq,am){var ak=am?"top":"left",an=am?"Top":"Left",ar=ao.style,al=ar[ak],ap=ao.runtimeStyle[ak],aj;ao.runtimeStyle[ak]=ao.currentStyle[ak];ar[ak]=aq;aj=ar["pixel"+an];ar[ak]=al;ao.runtimeStyle[ak]=ap;return aj;}function S(ak){if(/^(transparent|rgba\(0, ?0, ?0, ?0\))$/.test(ak)){return"transparent";}var ao,aj,ap,aq,al,an=Math.round,ar=parseInt,am=parseFloat;if(ao=t.exec(ak)){aj=ao[2]?an(((am(ao[1])/100)*255)):ar(ao[1]);ap=ao[4]?an(((am(ao[3])/100)*255)):ar(ao[3]);aq=ao[6]?an(((am(ao[5])/100)*255)):ar(ao[5]);}else{if(typeof ak=="number"){al=ak;}else{if(ak.charAt(0)=="#"){if(ak.length=="4"){ak="#"+ak.charAt(1)+ak.charAt(1)+ak.charAt(2)+ak.charAt(2)+ak.charAt(3)+ak.charAt(3);}al=ar(ak.slice(1),16);}else{al=I[ak];}}aj=(al)>>16;ap=(al&65280)>>8;aq=(al&255);}ak=new String("rgb("+aj+", "+ap+", "+aq+")");ak.r=aj;ak.g=ap;ak.b=aq;return ak;}function m(an){var am="",ak=an.childNodes,al=0,aj=ak.length;for(;al<aj;al++){if(ak[al].nodeType==3){am+=ak[al].nodeValue;}else{if(ak[al].nodeType==1){am+=m(ak[al]);}}}return am;}function q(ao,al){var ak=[],aj=0,r,am=0,an=ao.length;for(;am<an;am++){r=ao[am];while(r=r[al+"Sibling"]){if(r.nodeType==1&&r.nodeName!="!"){ak[aj++]=r;break;}}}return Y.get(ak);}function F(aj){var r=aj.offsetParent;while(r&&Y.get(r).css("position")=="static"){r=r.offsetParent;}if(!r&&Y.get(G).css("position")!="static"){r=G;}return r||null;}function ac(am,al){var ak,aj="scroll"+(al?"Left":"Top");if(am.window){ak=am.document.documentElement[aj]||(al?am.pageXOffset:am.pageYOffset)||0;}else{ak=am[aj];}return ak;}function o(ak,aj,r){if(ak.window){ak.scrollTo(aj?r:ac(ak,true),!aj?r:ac(ak,false));}else{ak["scroll"+(aj?"Left":"Top")]=r;}}function g(r,al,ak){var aj=r.length;if(ak!==undefined){while(aj--){o(r[aj],al,ak);}return r;}else{return ac(r[0],al);}}var Y={};Y.get=function(){var am=new u.dom.NodeList(),al=0,ak=arguments,aj=ak.length;for(;al<aj;al++){if(typeof ak[al]=="string"){am.push(new u.dom.NodeList().push(l).get(ak[al]));}else{am.push(ak[al]);}}return am;};Y.create=function(an,am){var r=[],aj=0,al=0,ak;am=u.lang.apply({interpolate:null,escapeHtml:false},am||{});if(am.interpolate){an=k.interpolate(an,am.interpolate,{escapeHtml:am.escapeHtml});}ak=N(an);for(;ak[aj];aj++){if(ak[aj].nodeType==1&&ak[aj].nodeName!="!"){r[al++]=ak[aj];}else{if(ak[aj].nodeType==3&&k.trim(ak[aj].nodeValue)!==""){throw new Error("glow.dom.create - Text must be wrapped in an element");}}}return new Y.NodeList().push(r);};Y.parseCssColor=function(r){var aj=S(r);return{r:aj.r,g:aj.g,b:aj.b};};Y.NodeList=function(){this.length=0;};Y.NodeList.prototype={item:function(r){return this[r];},push:function(){var ak=arguments,aj=ak.length,al=0,ao,r,am=this,an=Array.prototype.push;for(;al<aj;al++){if(!ak[al]){continue;}else{if(ak[al].nodeType==1||ak[al].nodeType==9||ak[al].document){an.call(am,ak[al]);}else{if(ak[al][0]){for(ao=0,r=ak[al].length;ao<r;ao++){an.call(am,ak[al][ao]);}}}}}return am;},each:function(al){for(var r=0,ak=this,aj=ak.length;r<aj;r++){al.call(ak[r],r,ak);}return ak;},eq:function(al){var ak=this,r=0,aj=ak.length;if(!al.push){al=[al];}if(al.length!=ak.length){return false;}for(;r<aj;r++){if(ak[r]!=al[r]){return false;}}return true;},isWithin:function(al){if(al.push){al=al[0];}if(!al||!this.length){return false;}var ak=this,r=0,aj=ak.length,am;if(al.contains&&n.webkit>=521){for(;r<aj;r++){if(!(al.contains(ak[r])&&ak[r]!=al)){return false;}}}else{if(ak[0].compareDocumentPosition){for(;r<aj;r++){if(!(ak[r].compareDocumentPosition(al)&8)){return false;}}}else{for(;r<aj;r++){am=ak[r];while(am=am.parentNode){if(am==al){break;}}if(!am){return false;}}}}return true;},attr:function(ak){var am=this,aj=arguments,r=aj.length,al,an;if(am.length===0){return r>1?am:undefined;}if(typeof ak=="object"){for(al in ak){if(k.hasOwnProperty(ak,al)){am.attr(al,ak[al]);}}return am;}if(n.ie&&R[ak]){if(r>1){e.call(am,aj[1],function(ao){this[R[ak]]=ao;});return am;}an=am[0][R[ak]];if(c[ak]){return an?ak:undefined;}else{if(ah[ak]){return ah[ak](an);}}return an;}if(r>1){e.call(am,aj[1],function(ao){this.setAttribute(ak,ao);});return am;}return M(am[0])?am[0].getAttribute(ak):am[0].getAttribute(ak,2);},removeAttr:function(aj){var r=n.ie&&R[aj],am=this,ak=0,al=am.length;for(;ak<al;ak++){if(r){am[ak][r]="";}else{am[ak].removeAttribute(aj);}}return am;},hasAttr:function(al){var an=this[0],ak=an.attributes;if(M(an)&&n.ie){var ak=an.attributes,am=0,aj=ak.length;for(;am<aj;am++){if(ak[am].nodeName==al){return ak[am].specified;}}return false;}else{if(this[0].getAttributeNode){var r=this[0].getAttributeNode(al);return r?r.specified:false;}}return typeof ak[r]!="undefined";},prop:function(r,am){if(r.constructor===Object){var al=r,ak;for(ak in al){this.prop(ak,al[ak]);}return this;}if(am!==undefined){var aj=this.length;while(aj--){this[aj][r]=am;}return this;}if(!this[0]){return undefined;}return this[0][r];},hasClass:function(r){for(var aj=0,ak=this.length;aj<ak;aj++){if((" "+this[aj].className+" ").indexOf(" "+r+" ")!=-1){return true;}}return false;},addClass:function(r){for(var aj=0,ak=this.length;aj<ak;aj++){if((" "+this[aj].className+" ").indexOf(" "+r+" ")==-1){this[aj].className+=((this[aj].className)?" ":"")+r;}}return this;},removeClass:function(r){var ak=v(r),am=this,aj=0,al=am.length;for(;aj<al;aj++){am[aj].className=am[aj].className.replace(ak," ");}return am;},toggleClass:function(aj){var al=this.length,ak,r=" "+aj+" ";while(al--){ak=" "+this[al].className+" ";if(ak.indexOf(r)!=-1){this[al].className=ak.replace(r," ");}else{this[al].className+=" "+aj;}}return this;},val:function(){function al(ap){var am=ap.type,an=ap.checked,ar=ap.value,at=[],ao=0;if(am=="radio"){return an?ar:"";}else{if(am=="checkbox"){return an?ar:"";}else{if(am=="select-one"){return ap.selectedIndex>-1?ap.options[ap.selectedIndex].value:"";}else{if(am=="select-multiple"){for(var aq=ap.options.length;ao<aq;ao++){if(ap.options[ao].selected){at[at.length]=ap.options[ao].value;}}return at;}else{return ar;}}}}}function aj(an){var av={},au={},aq=an.elements,at=0,ap=aq.length,am,ax,ar,ao,aw;for(;at<ap;at++){ax=aq[at];aw=ax.nodeName.toLowerCase();am=ax.name;if(aw=="fieldset"||aw=="object"||!am){continue;}if(ax.type=="checkbox"&&!ax.checked){if(!am in av){av[am]=undefined;}}else{if(ax.type=="radio"){if(au[am]){au[am][au[am].length]=ax;}else{au[am]=[ax];}}else{var ay=al(ax);if(am in av){if(av[am].push){av[am][av[am].length]=ay;}else{av[am]=[av[am],ay];}}else{av[am]=ay;}}}}for(at in au){ar=0;for(ap=au[at].length;ar<ap;ar++){ao=au[at][ar];am=ao.name;if(ao.checked){av[ao.name]=ao.value;break;}}if(!am in av){av[am]=undefined;}}return av;}function r(an,aw){var am,ax,at={},aq,ar=0,ao,au,av,ap;for(am in aw){ax=an[am];if(ax&&ax[0]&&!ax.options){aw[am]=aw[am]&&aw[am].push?aw[am]:[aw[am]];at.radios=[];at.checkboxesSelects=[];at.multiSelects=[];at.other=[];for(ar=0;ax[ar];ar++){ap=ax[ar].type;if(ap=="radio"){aq="radios";}else{if(ap=="select-one"||ap=="checkbox"){aq="checkboxesSelects";}else{if(ap=="select-multiple"){aq="multiSelects";}else{aq="other";}}}at[aq][at[aq].length]=ax[ar];}for(ar=0;at.multiSelects[ar];ar++){aw[am]=ak(at.multiSelects[ar],aw[am]);}for(ar=0;at.checkboxesSelects[ar];ar++){ak(at.checkboxesSelects[ar],"");for(ao=0,au=aw[am].length;ao<au;ao++){if(ak(at.checkboxesSelects[ar],aw[am][ao])){aw[am].slice(ao,1);break;}}}for(ar=0;at.radios[ar];ar++){at.radios[ar].checked=false;av=false;for(ao=0,au=aw[am].length;ao<au;ao++){if(ak(at.radios[ar],aw[am][ao])){aw[am].slice(ao,1);av=true;break;}if(av){break;}}}for(ar=0;at.other[ar]&&aw[am][ar]!==undefined;ar++){ak(at.other[ar],aw[am][ar]);}}else{if(ax&&ax.nodeName){ak(ax,aw[am]);}}}}function ak(ao,aq){var ar=0,an,ap=0,av,am,au;if(ao.type=="select-one"){for(an=ao.options.length;ar<an;ar++){if(ao.options[ar].value==aq){ao.selectedIndex=ar;return true;}}return false;}else{if(ao.type=="select-multiple"){var at=!!aq.push;for(ar=0,an=ao.options.length;ar<an;ar++){am=ao.options[ar];au=am.value;if(at){am.selected=false;for(av=aq.length;ap<av;ap++){if(au==aq[ap]){am.selected=true;aq.splice(ap,1);break;}}}else{return am.selected=aq==au;}}return false;}else{if(ao.type=="radio"||ao.type=="checkbox"){ao.checked=aq==ao.value;return aq==ao.value;}else{ao.value=aq;return true;}}}}return function(){var am=arguments,aq=am[0],ap=this,an=0,ao=ap.length;if(am.length===0){return ap[0].nodeName=="FORM"?aj(ap[0]):al(ap[0]);}if(ap[0].nodeName=="FORM"){if(!typeof aq=="object"){throw"value for FORM must be object";}r(ap[0],aq);}else{for(;an<ao;an++){ak(ap[an],aq);}}return ap;};}(),slice:function(){return new Y.NodeList().push(Array.prototype.slice.apply(this,arguments));},sort:function(al){var ak=this,aj=0,r;if(!ak.length){return ak;}if(!al){if(typeof ak[0].sourceIndex=="number"){al=function(an,am){return an.sourceIndex-am.sourceIndex;};}else{if(ak[0].compareDocumentPosition){al=function(an,am){return 3-(an.compareDocumentPosition(am)&6);};}else{r=H("*",[l]);for(;r[aj];aj++){r[aj]._sourceIndex=aj;}al=function(an,am){return an._sourceIndex-am._sourceIndex;};}}}return Y.get([].sort.call(ak,al));},filter:function(am){var aj=[],r=0,ak=0,al=this.length;for(;ak<al;ak++){if(am.apply(this[ak],[ak])){aj[r++]=this[ak];}}return Y.get(aj);},children:function(){var aj=[],r=0,ak=0,an=0,am=this.length,al;for(;ak<am;ak++){aj=aj.concat(b(this[ak]));}return Y.get(aj);},parent:function(){var aj=[],r=0,ak=0,al=this.length;for(;ak<al;ak++){aj[r++]=this[ak].parentNode;}return Y.get(d(aj));},ancestors:function(){var aj=[],r=0,ak=0,al=this.length,am;while(ak<al){am=this[ak].parentNode;while(am&&am.nodeType==1){aj[r++]=am;am=am.parentNode;}ak++;}return Y.get(d(aj));},wrap:function(am){var al=this.length,r,ak,aj;if(typeof am=="string"){aj=Y.create(am);}else{aj=Y.get(am);}for(i=0;i<al;i++){ak=aj[0];while(ak){r=af(ak);if(r){ak=r;}else{break;}}if(this[i].parentNode){aj.insertBefore(this[i]);}if(i!=al-1){aj=aj.clone();}ak.appendChild(this[i]);}return this;},unwrap:function(){var aj,r=this.parent(),ak=r.length;for(i=0;i<ak;i++){aj=r.slice(i,i+1);if(!aj[0].parentNode){aj.children().remove();aj.destroy();}else{aj.children().insertBefore(aj);aj.destroy();}}return this;},next:function(){return q(this,"next");},prev:function(){return q(this,"previous");},is:function(aj){var ak=u.dom.get(aj),am=0,r=this.length,al,an;node:for(;am<r;am++){for(al=0,an=ak.length;al<an;al++){if(this[am]==ak[al]){continue node;}}return false;}return true;},text:function(){var r=arguments,aj=0,al=this,ak=al.length;if(r.length>0){for(;aj<ak;aj++){al[aj].innerHTML="";al[aj].appendChild(l.createTextNode(r[0]));}return al;}return al[0].innerText||al[0].textContent==undefined?m(al[0]):al[0].textContent;},empty:function(){var aj=0,r=this.length;for(;aj<r;aj++){while(this[aj].firstChild){this[aj].removeChild(this[aj].firstChild);}}return this;},remove:function(){for(var al=this,aj=0,ak=al.length,r;aj<ak;aj++){if(r=al[aj].parentNode){r.removeChild(al[aj]);}}return al;},destroy:function(){this.get("*").push(this).removeData();this.appendTo(P);P.innerHTML="";Array.prototype.splice.call(this,0,this.length);return this;},clone:function(am){var aj=[],al=this.length,r,ak;eventIdProp="__eventId"+u.UID;if(al===0){return new Y.NodeList();}while(al--){aj[al]=this[al].cloneNode(true);}r=Y.get(aj).get("*").push(aj);if(w&&!M(aj[0])){al=r.length;while(al--){r[al][eventIdProp]=null;}}ak=this.get("*").push(this);al=r.length;while(al--){r[al].removeAttribute(ai);u.dom.get(r[al]).data(u.dom.get(ak[al]).data());}if(am){if(!u.events){throw"glow.events required to clone event listeners";}u.events._copyListeners(this.get("*").push(this),r||Y.get(aj).get("*").push(aj));}return Y.get(aj);},html:function(r){var aj=0,ak=this.length;if(r!==undefined){return this.empty().append(r.toString());}return this[0]?this[0].innerHTML:"";},width:function(r){if(r==undefined){return f(this[0],"width");}j(this,r,"width");return this;},height:function(r){if(r==undefined){return f(this[0],"height");}j(this,r,"height");return this;},scrollLeft:function(r){return g(this,true,r);},scrollTop:function(r){return g(this,false,r);},show:function(){var ak=0,aj=this.length,r,al;for(;ak<aj;ak++){r=Y.get(this[ak]);al=r[0].style;if(r.css("display")=="none"){al.display="";al.visibility="visible";if(r.css("display")=="none"){al.display="block";}}}return this;},hide:function(){return this.css("display","none").css("visibility","hidden");},css:function(ao,al){var ak=this,an,aj=0,r=ak.length,am=ao;if(ao.constructor===Object){for(style in ao){this.css(style,ao[style]);}return ak;}else{if(al!=undefined){ao=W(ao);for(;aj<r;aj++){an=ak[aj].style;if(typeof al=="number"&&C.test(am)){al=al.toString()+"px";}if(ao=="opacity"&&n.ie){an.zoom="1";if(al===""){an.filter="";}else{an.filter="alpha(opacity="+Math.round(Number(al,10)*100)+")";}}else{an[ao]=al;}}return ak;}else{if(!r){return;}return Z(ak[0],ao);}}},offset:function(){var an=this[0],al={x:ac(window,true),y:ac(window,false)};if(!u.env.webkit&&an.getBoundingClientRect){var ap=an.getBoundingClientRect();return{top:ap.top+al.y-G.clientTop,left:ap.left+al.x-G.clientLeft};}else{var ao=an.offsetTop,r=an.offsetLeft,aj=an,aq,am=false,ak=an;while(an=an.offsetParent){r+=an.offsetLeft;ao+=an.offsetTop;if(Z(an,"position")=="fixed"){am=true;}if(n.gecko||n.webkit>500){r+=parseInt(Z(an,"border-left-width"))||0;ao+=parseInt(Z(an,"border-top-width"))||0;}if(an.nodeName.toLowerCase()!="body"){ak=an;}}an=aj;while((an=an.parentNode)&&(an!=V)&&(an!=G)){r-=an.scrollLeft;ao-=an.scrollTop;if(n.gecko&&Z(an,"overflow")!="visible"){r+=parseInt(Z(an,"border-left-width"));ao+=parseInt(Z(an,"border-top-width"));}}if(am){r+=al.x;ao+=al.y;}if((n.webkit<500&&(am||Z(ak,"position")=="absolute"))||(n.gecko&&Z(ak,"position")!="absolute")){r-=V.offsetLeft;ao-=V.offsetTop;}return{left:r,top:ao};}},position:function(){var aj=Y.get(F(this[0])),ao=!!aj[0],an=parseInt(this.css("margin-left"))||0,am=parseInt(this.css("margin-top"))||0,al=(ao&&parseInt(aj.css("border-left-width")))||0,r=(ao&&parseInt(aj.css("border-top-width")))||0,ap=this.offset(),ak=ao?aj.offset():{top:0,left:0};return{left:ap.left-ak.left-an-al,top:ap.top-ak.top-am-r};},append:function(an){var am=this,aj=0,ak=1,al=am.length,r;if(al==0){return am;}r=typeof an=="string"?p(N(an)):an.nodeType?[an]:p(an);for(;r[aj];aj++){am[0].appendChild(r[aj]);}for(;ak<al;ak++){for(aj=0;r[aj];aj++){am[ak].appendChild(r[aj].cloneNode(true));}}return am;},prepend:function(ao){var am=this,aj=0,ak=1,al=am.length,r,an;if(al==0){return am;}r=typeof ao=="string"?p(N(ao)):ao.nodeType?[ao]:p(ao);an=am[0].firstChild;for(;r[aj];aj++){am[0].insertBefore(r[aj],an);}for(;ak<al;ak++){an=am[ak].firstChild;for(aj=0;r[aj];aj++){am[ak].insertBefore(r[aj].cloneNode(true),an);}}return am;},appendTo:function(r){if(!(r instanceof Y.NodeList)){r=Y.get(r);}r.append(this);return this;},prependTo:function(r){if(!(r instanceof Y.NodeList)){r=Y.get(r);}r.prepend(this);return this;},after:function(ap){var ao=this,an=ao.length,ak,aj,al,am=1,r;if(an==0){return ao;}ak=typeof ap=="string"?Y.create(ap):ap instanceof Y.NodeList?ap:Y.get(ap);aj=ak.length;for(al=aj-1;al>=0;al--){ao[0].parentNode.insertBefore(ak[al],ao[0].nextSibling);}for(;am<an;am++){r=ak.clone();for(al=aj-1;al>=0;al--){ao[am].parentNode.insertBefore(r[al],ao[am].nextSibling);}}return ao;},before:function(ap){var ao=this,an=ao.length,al=0,am=1,ak,aj,r;if(an==0){return ao;}ak=typeof ap=="string"?Y.create(ap):ap instanceof Y.NodeList?ap:Y.get(ap);aj=ak.length;for(;al<aj;al++){ao[0].parentNode.insertBefore(ak[al],ao[0]);}for(;am<an;am++){r=ak.clone();for(al=0;al<aj;al++){ao[am].parentNode.insertBefore(r[al],ao[am]);}}return ao;},insertAfter:function(r){if(!(r instanceof Y.NodeList)){r=Y.get(r);}r.after(this);return this;},insertBefore:function(r){if(!(r instanceof Y.NodeList)){r=Y.get(r);}r.before(this);return this;},replaceWith:function(r){if(n.webkit<500){this.after(K).remove();Y.get("u.glow-placeholder").after(r).remove();}else{this.after(r).remove();}return this;},data:function(ak,al){if(typeof ak==="object"){for(var an in ak){this.data(an,ak[an]);}return this;}var r,am;switch(arguments.length){case 0:if(this[0]===undefined){return undefined;}r=this[0][ai]||ag++;return L[r]||(L[r]={});case 1:if(this[0]===undefined){return undefined;}r=this[0][ai];return r?L[r][ak]:undefined;case 2:for(var aj=this.length;aj--;){am=this[aj];if(!(r=am[ai])){r=ag++;am[ai]=r;L[r]={};}L[r][ak]=al;}return this;default:throw new Error("glow.dom.NodeList#data expects 2 or less arguments, not "+arguments.length+".");}},removeData:function(ak){var am,aj=this.length,r;while(aj--){am=this[aj];r=am[ai];if(r!==undefined){switch(arguments.length){case 0:L[r]=undefined;am[ai]=undefined;try{delete am[ai];}catch(al){am.removeAttribute&&am.removeAttribute(ai);}break;case 1:L[r][ak]=undefined;delete L[r][ak];break;default:throw new Error("glow.dom.NodeList#removeData expects 1 or less arguments, not "+arguments.length+".");}}}return this;},get:function(){function ak(az){if(B[az]){return B[az];}var ar=[],aC=0,at,aA,av,ax,aB,ay,au=true,aw=az;while(az&&az!=ay){aA="";av="";ay=az;if(ax=s.combinator.exec(az)){at=ax[1];az=az.slice(ax[0].length);}if(ax=s.tagName.exec(az)){aA=ax[1];az=az.slice(ax[0].length);}if(ax=s.classNameOrId.exec(az)){if(ax[1]=="#"){av=ax[2];az=az.slice(ax[0].length);}}if(!at){if(av&&au){ar[aC++]=[am,[av.replace(/\\/g,""),aA||"*",null]];}else{ar[aC++]=[H,[aA||"*",null]];if(av){ar[aC++]=[ap,[av.replace(/\\/g,""),null]];}}}else{if(at==">"){ar[aC++]=[al,[null]];if(av){ar[aC++]=[ap,[av.replace(/\\/g,""),null]];}if(aA&&aA!="*"){ar[aC++]=[aq,[aA,null]];}}}aB=true;while(aB){if(az.charAt(0)=="#"||az.charAt(0)=="."){if(ax=s.classNameOrId.exec(az)){if(az.charAt(0)=="#"){ar[aC++]=[ap,[ax[2].replace(/\\/g,""),null]];}else{ar[aC++]=[ao,[ax[2].replace(/\\/g,""),null]];}az=az.slice(ax[0].length);}else{throw new Error("Invalid Selector "+aw);}}else{aB=false;}}au=false;}if(az!==""){throw new Error("Invalid Selector "+aw);}return B[az]=ar;}function an(ar,av){var au=av;for(var at=0,aw=ar.length;at<aw;at++){ar[at][1][ar[at][1].length-1]=au;au=ar[at][0].apply(this,ar[at][1]);}return au;}function am(at,ax,au){var ar=[],aB=0,aw=[],aA=0,ay;for(var az=0,av=au.length;az<av;az++){if(au[az].getElementById){ay=au[az].getElementById(at);if(ay&&(ay.tagName==ax.toUpperCase()||ax=="*"||ay.tagName==ax)){ar[aB++]=ay;}}else{aw[aA++]=au[az];}}if(aw[0]){aw=H(ax,aw);aw=ap(at,aw);}return ar.concat(aw);}function al(au){var av=[],at=0,ar=au.length;for(;at<ar;at++){T(av,b(au[at]));}return av;}function ap(av,at){for(var ar=0,au=at.length;ar<au;ar++){if(at[ar].id==av){return[at[ar]];}}return[];}function aq(av,au){var ax=[],ar=0;for(var at=0,aw=au.length;at<aw;at++){if(au[at].tagName==av.toUpperCase()||au[at].tagName==av){ax[ar++]=au[at];}}return ax;}function ao(av,au){var ax=[],ar=0;for(var at=0,aw=au.length;at<aw;at++){if((" "+au[at].className+" ").indexOf(" "+av+" ")!=-1){ax[ar++]=au[at];}}return ax;}function aj(ay,au){var av;var ax=[];var aw=ay.split(",");for(var at=0,ar=aw.length;at<ar;at++){av=ak(u.lang.trim(aw[at]));ax=ax.concat(an(av,au));}return ax;}function r(at,av){at=at.length?at:[at];var aw=[];var ar;for(var au=0;at[au];au++){ar=u.dom.get(at[au]);for(var ax=0;av[ax];ax++){if(ar.isWithin(av[ax])){aw[aw.length]=ar[0];break;}}}return aw;}return function(av){if(!this.length){return this;}var au=[];for(var at=0,ar=arguments.length;at<ar;at++){if(typeof arguments[at]=="string"){au=au.concat(aj(arguments[at],this));}else{au=au.concat(r(arguments[at],this));}}return u.dom.get(d(au));};}()};K=Y.create('<u class="glow-placeholder"></u>');u.dom=Y;}});(window.gloader||glow).module({name:"glow.events",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.dom"]],builder:function(o){var k=o.dom.get;var C={};var x=1;var m=1;var n={};var b={};var p={};var l="__eventId"+o.UID;var g=l+"PreventDefault";var u=l+"StopPropagation";var E={};var f=1;var B={};var z={};var G=1;var d=2;var s=4;var h={TAB:"\t",SPACE:" ",ENTER:"\n",BACKTICK:"`"};var K={"96":223};var t={CAPSLOCK:20,NUMLOCK:144,SCROLLLOCK:145,BREAK:19,BACKTICK:223,BACKSPACE:8,PRINTSCREEN:44,MENU:93,SPACE:32,SHIFT:16,CTRL:17,ALT:18,ESC:27,TAB:9,META:91,RIGHTMETA:92,ENTER:13,F1:112,F2:113,F3:114,F4:115,F5:116,F6:117,F7:118,F8:119,F9:120,F10:121,F11:122,F12:123,INS:45,HOME:36,PAGEUP:33,DEL:46,END:35,PAGEDOWN:34,LEFT:37,UP:38,RIGHT:39,DOWN:40};var I={};for(var H in t){I[""+t[H]]=H;}var y="0123456789=;'\\/#,.-";function D(O){var N=z[O];if(!N){return false;}var M=B[N];if(!M){return false;}for(var L=0,r=M.length;L<r;L++){if(M[L][0]==O){M.splice(L,1);return true;}}return false;}function A(r){E[r]=C.addListener(document,"key"+r,function(R){var P=0;if(R.ctrlKey){P+=G;}if(R.altKey){P+=d;}if(R.shiftKey){P+=s;}var Q=R.chr?R.chr.toLowerCase():R.key?R.key.toLowerCase():R.keyCode;var N=P+":"+Q+":"+r;var O=B[N]?B[N].slice(0):[];if(R.shiftKey){var S=(P&~s)+":"+Q+":"+r;if(B[S]){for(var M=0,L=B[S].length;M<L;M++){O[O.length]=B[S][M];}}}if(!O){return;}for(var M=0,L=O.length;M<L;M++){if(O[M][2].call(O[M][3]||this,R)===false){R.preventDefault();}}return !R.defaultPrevented();});}function w(){var r;for(r in b){C.removeListener(r);}}var q;var J,e;function F(O,N,L){var r;L=!!L;if(o.env.opera){if(N.toLowerCase()=="resize"&&!J&&O==window){J=C.addListener(window.document.body,"resize",function(R){C.fire(window,"resize",R);});}else{if(N.toLowerCase()=="scroll"&&!e&&O==window){e=C.addListener(window.document,"scroll",function(R){C.fire(window,"scroll",R);});}}}var Q=function(T){if(!T){T=window.event;}var S=new C.Event(),R=N.toLowerCase();S.nativeEvent=T;S.source=T.target||T.srcElement;S.relatedTarget=T.relatedTarget||(R=="mouseover"?T.fromElement:T.toElement);S.button=o.env.ie?(T.button&1?0:T.button&2?2:1):T.button;if(T.pageX||T.pageY){S.pageX=T.pageX;S.pageY=T.pageY;}else{if(T.clientX||T.clientY){S.pageX=T.clientX+document.body.scrollLeft+document.documentElement.scrollLeft;S.pageY=T.clientY+document.body.scrollTop+document.documentElement.scrollTop;}}if(R=="mousewheel"){S.wheelDelta=T.wheelDelta?T.wheelDelta/120:T.detail?-T.detail/3:0;if(S.wheelDelta==0){return;}}if(R.indexOf("key")!=-1){S.altKey=!!T.altKey;S.ctrlKey=!!T.ctrlKey;S.shiftKey=!!T.shiftKey;if(N=="keydown"){q=T.keyCode;}S.charCode=T.keyCode&&T.charCode!==0?undefined:T.charCode;if(R=="keypress"){if(typeof(S.charCode)=="undefined"){S.charCode=T.keyCode;}if(o.env.opera&&S.charCode&&S.charCode==q&&y.indexOf(String.fromCharCode(S.charCode))==-1){S.charCode=undefined;S.keyCode=q;}}if(S.charCode&&S.charCode<=49){S.charCode=undefined;}if(S.charCode){S.chr=String.fromCharCode(S.charCode);}else{if(T.keyCode){S.charCode=undefined;S.keyCode=K[T.keyCode.toString()]||T.keyCode;S.key=I[S.keyCode];if(h[S.key]){S.chr=h[S.key];S.charCode=S.chr.charCodeAt(0);}}}if(S.chr){S.capsLock=S.chr.toUpperCase()!=S.chr?S.shiftKey:S.chr.toLowerCase()!=S.chr?!S.shiftKey:undefined;}}C.fire(this,N,S);if(S.defaultPrevented()){return false;}};if(O.addEventListener&&(!o.env.webkit||o.env.webkit>418)){if((N=="focus"||N=="blur")&&(o.env.opera)){O.parentNode.addEventListener(N,function(){},true);}O.addEventListener(N.toLowerCase()=="mousewheel"&&o.env.gecko?"DOMMouseScroll":N,Q,L);}else{var M="on"+N;var P=O[M];if(P){O[M]=function(){var R=P.apply(this,arguments),S=Q.apply(this,arguments);return(R!==false)&&(S!==false);};}else{O[M]=Q;}}O=null;}function a(M,N){var O=k(M),r=N?"mouseout":"mouseover",L=N?"mouseleave":"mouseenter";C.addListener(M,r,function(Q){var P=k(Q.relatedTarget);if(!P.eq(O)&&!P.isWithin(O)){return !C.fire(O[0],L,Q).defaultPrevented();}});}C._copyListeners=function(R,Q){var M=R.length,P,r,L,O,N;while(M--){if(R[M][l]){P=n[R[M][l]];for(r in P){L=0;O=P[r].length;for(;L<O;L++){N=P[r][L];C.addListener(Q[M],r,N[2],N[3]);}}}}};C.addListener=function(S,M,V,O){var T=false;if(!S){throw"no attachTo paramter passed to addListener";}if(typeof S=="string"){if(!o.dom){throw"glow.dom must be loaded to use a selector as the first argument to glow.events.addListener";}S=k(S);}if(o.dom&&S instanceof o.dom.NodeList){var U=[],R=S.length;while(R--){U[R]=C.addListener(S[R],M,V,O);}return U;}var N;if(!(N=S[l])){N=S[l]=m++;}var Q=x++;var P=[N,M,V,O,Q];b[Q]=P;var r=n[N];if(!r){r=n[N]={};}var L=r[M];if(!L){L=r[M]=[];}L[L.length]=P;if((S.addEventListener||S.attachEvent)&&!p[N+":"+M]){switch(M){case"mouseenter":a(S,false);return Q;case"mouseleave":a(S,true);return Q;case"focus":if(o.env.ie){c(S,true);return Q;}else{T=true;}break;case"blur":if(o.env.ie){c(S,false);return Q;}else{T=true;}break;}F(S,M,T);p[N+":"+M]=true;}return Q;};function c(M,N){var r=N?"focusin":"focusout",L=N?"focus":"blur";C.addListener(M,r,function(O){return !C.fire(M,L,O).defaultPrevented();});}C.removeListener=function(O){if(O&&O.toString().indexOf("k:")!=-1){return D(O);}if(O instanceof Array){var L=O.length;while(L--){C.removeListener(O[L]);}return true;}var N=b[O];if(!N){return false;}delete b[O];var M=n[N[0]][N[1]];for(var L=0,r=M.length;L<r;L++){if(M[L]==N){M.splice(L,1);break;}}if(!M.length){delete n[N[0]][N[1]];}var P=false;for(var L in n[N[0]]){P=true;break;}if(!P){delete n[N[0]];}return true;};C.removeAllListeners=function(Q){var N,M,P=[],O=0,r,L;if(typeof Q=="string"){Q=k(Q);}if(Q instanceof Array||Q instanceof o.dom.NodeList){N=Q.length;while(N--){C.removeAllListeners(Q[N]);}return C;}M=Q[l];if(!M){return C;}L=n[M];for(r in L){N=L[r].length;while(N--){P[O++]=L[r][N][4];}}if(P.length){C.removeListener(P);}return C;};C.fire=function(r,L,M){if(!r){throw"glow.events.fire: required parameter attachedTo not passed (name: "+L+")";}if(!L){throw"glow.events.fire: required parameter name not passed";}if(!M){M=new C.Event();}if(M.constructor===Object){M=new C.Event(M);}if(typeof r=="string"){if(!o.dom){throw"glow.dom must be loaded to use a selector as the first argument to glow.events.addListener";}r=k(r);}M.type=L;M.attachedTo=r;if(!M.source){M.source=r;}if(r instanceof o.dom.NodeList){r.each(function(N){v(r[N],M);});}else{v(r,M);}return M;};function v(R,Q){var L,r,M=r&&r[Q.type];(L=R[l])&&(r=n[L])&&(M=r[Q.type]);if(!M){return Q;}var N;var S=M.slice(0);for(var O=0,P=S.length;O<P;O++){N=S[O];if(N[2].call(N[3]||R,Q)===false){Q.preventDefault();}}}var j=/^((?:(?:ctrl|alt|shift)\+)*)(?:(\w+|.)|[\n\r])$/i;C.addKeyListener=function(Q,P,R,L){P.replace(/^key/i,"");P=P.toLowerCase();if(!(P=="press"||P=="down"||P=="up")){throw"event type must be press, down or up";}if(!E[P]){A(P);}var N=Q.match(j),S=0,T;if(!N){throw"key format not recognised";}if(N[1].toLowerCase().indexOf("ctrl")!=-1){S+=G;}if(N[1].toLowerCase().indexOf("alt")!=-1){S+=d;}if(N[1].toLowerCase().indexOf("shift")!=-1){S+=s;}var r=S+":"+(N[2]?N[2].toLowerCase():"\n")+":"+P;var M="k:"+f++;z[M]=r;var O=B[r];if(!O){O=B[r]=[];}O[O.length]=[M,P,R,L];return M;};C.Event=function(r){if(r){o.lang.apply(this,r);}};C.Event.prototype.preventDefault=function(){if(this[g]){return;}this[g]=true;if(this.nativeEvent&&this.nativeEvent.preventDefault){this.nativeEvent.preventDefault();this.nativeEvent.returnValue=false;}};C.Event.prototype.defaultPrevented=function(){return !!this[g];};C.Event.prototype.stopPropagation=function(){if(this[u]){return;}this[u]=true;var r=this.nativeEvent;if(r){r.cancelBubble=true;if(r.stopPropagation){r.stopPropagation();}}};C.Event.prototype.propagationStopped=function(){return !!this[u];};if(o.env.ie<8||o.env.webkit<500){C.addListener(window,"unload",w);}o.events=C;o.events.listenersByObjId=n;}});(window.gloader||glow).module({name:"glow.data",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.dom"]],builder:function(glow){var TYPES={UNDEFINED:"undefined",OBJECT:"object",NUMBER:"number",BOOLEAN:"boolean",STRING:"string",ARRAY:"array",FUNCTION:"function",NULL:"null"};var TEXT={AT:"@",EQ:"=",DOT:".",EMPTY:"",AND:"&",OPEN:"(",CLOSE:")"};var JSON={HASH:{START:"{",END:"}",SHOW_KEYS:true},ARRAY:{START:"[",END:"]",SHOW_KEYS:false},DATA_SEPARATOR:",",KEY_SEPARATOR:":",KEY_DELIMITER:'"',STRING_DELIMITER:'"',SAFE_PT1:/^[\],:{}\s]*$/,SAFE_PT2:/\\./g,SAFE_PT3:/\"[^\"\\\n\r]*\"|true|false|null|-?\d+(?:\.\d*)?(:?[eE][+\-]?\d+)?/g,SAFE_PT4:/(?:^|:|,)(?:\s*\[)+/g};var SLASHES={TEST:/[\b\n\r\t\\\f\"]/g,B:{PLAIN:"\b",ESC:"\\b"},N:{PLAIN:"\n",ESC:"\\n"},R:{PLAIN:"\r",ESC:"\\r"},T:{PLAIN:"\t",ESC:"\\t"},F:{PLAIN:"\f",ESC:"\\f"},SL:{PLAIN:"\\",ESC:"\\\\"},QU:{PLAIN:'"',ESC:'\\"'}};function _replaceSlashes(s){switch(s){case SLASHES.B.PLAIN:return SLASHES.B.ESC;case SLASHES.N.PLAIN:return SLASHES.N.ESC;case SLASHES.R.PLAIN:return SLASHES.R.ESC;case SLASHES.T.PLAIN:return SLASHES.T.ESC;case SLASHES.F.PLAIN:return SLASHES.F.ESC;case SLASHES.SL.PLAIN:return SLASHES.SL.ESC;case SLASHES.QU.PLAIN:return SLASHES.QU.ESC;default:return s;}}function _getType(object){if((typeof object)==TYPES.OBJECT){if(object==null){return TYPES.NULL;}else{return(object instanceof Array)?TYPES.ARRAY:TYPES.OBJECT;}}else{return(typeof object);}}glow.data={encodeUrl:function(object){var objectType=_getType(object);var paramsList=[];var listLength=0;if(objectType!=TYPES.OBJECT){throw new Error("glow.data.encodeUrl: cannot encode item");}else{for(var key in object){switch(_getType(object[key])){case TYPES.FUNCTION:case TYPES.OBJECT:throw new Error("glow.data.encodeUrl: cannot encode item");break;case TYPES.ARRAY:for(var i=0,l=object[key].length;i<l;i++){switch(_getType(object[key])[i]){case TYPES.FUNCTION:case TYPES.OBJECT:case TYPES.ARRAY:throw new Error("glow.data.encodeUrl: cannot encode item");break;default:paramsList[listLength++]=key+TEXT.EQ+encodeURIComponent(object[key][i]);}}break;default:paramsList[listLength++]=key+TEXT.EQ+encodeURIComponent(object[key]);}}return paramsList.join(TEXT.AND);}},decodeUrl:function(text){if(_getType(text)!=TYPES.STRING){throw new Error("glow.data.decodeUrl: cannot decode item");}else{if(text===""){return{};}}var result={};var keyValues=text.split(/[&;]/);var thisPair,key,value;for(var i=0,l=keyValues.length;i<l;i++){thisPair=keyValues[i].split(TEXT.EQ);if(thisPair.length!=2){throw new Error("glow.data.decodeUrl: cannot decode item");}else{key=glow.lang.trim(decodeURIComponent(thisPair[0]));value=glow.lang.trim(decodeURIComponent(thisPair[1]));switch(_getType(result[key])){case TYPES.ARRAY:result[key][result[key].length]=value;break;case TYPES.UNDEFINED:result[key]=value;break;default:result[key]=[result[key],value];}}}return result;},encodeJson:function(object,options){function _encode(object,options){if(_getType(object)==TYPES.ARRAY){var type=JSON.ARRAY;}else{var type=JSON.HASH;}var serial=[type.START];var len=1;var dataType;var notFirst=false;for(var key in object){dataType=_getType(object[key]);if(dataType!=TYPES.UNDEFINED){if(notFirst){serial[len++]=JSON.DATA_SEPARATOR;}notFirst=true;if(type.SHOW_KEYS){serial[len++]=JSON.KEY_DELIMITER;serial[len++]=key;serial[len++]=JSON.KEY_DELIMITER;serial[len++]=JSON.KEY_SEPARATOR;}switch(dataType){case TYPES.FUNCTION:throw new Error("glow.data.encodeJson: cannot encode item");break;case TYPES.STRING:default:serial[len++]=JSON.STRING_DELIMITER;serial[len++]=glow.lang.replace(object[key],SLASHES.TEST,_replaceSlashes);serial[len++]=JSON.STRING_DELIMITER;break;case TYPES.NUMBER:case TYPES.BOOLEAN:serial[len++]=object[key];break;case TYPES.OBJECT:case TYPES.ARRAY:serial[len++]=_encode(object[key],options);break;case TYPES.NULL:serial[len++]=TYPES.NULL;break;}}}serial[len++]=type.END;return serial.join(TEXT.EMPTY);}options=options||{};var type=_getType(object);if((type==TYPES.OBJECT)||(type==TYPES.ARRAY)){return _encode(object,options);}else{throw new Error("glow.data.encodeJson: cannot encode item");}},decodeJson:function(text,options){if(_getType(text)!=TYPES.STRING){throw new Error("glow.data.decodeJson: cannot decode item");}options=options||{};options.safeMode=options.safeMode||false;var canEval=true;if(options.safeMode){canEval=(JSON.SAFE_PT1.test(text.replace(JSON.SAFE_PT2,TEXT.AT).replace(JSON.SAFE_PT3,JSON.ARRAY.END).replace(JSON.SAFE_PT4,TEXT.EMPTY)));}if(canEval){try{return eval(TEXT.OPEN+text+TEXT.CLOSE);}catch(e){}}throw new Error("glow.data.decodeJson: cannot decode item");},escapeHTML:function(html){return glow.dom.create("<div></div>").text(html).html();}};}});(window.gloader||glow).module({name:"glow.net",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.data","glow.events"]],builder:function(h){var t={XML_ERR:"Cannot get response as XML, check the mime type of the data",POST_DEFAULT_CONTENT_TYPE:"application/x-www-form-urlencoded;"},u=/\+xml$/,p=[],e="c",o="_"+h.UID+"loadScriptCbs",g=h.dom.get,c=h.events,f=function(){},k=1;function s(){if(window.ActiveXObject){return(s=function(){return new ActiveXObject("Microsoft.XMLHTTP");})();}else{return(s=function(){return new XMLHttpRequest();})();}}function d(w){var r=h.lang.apply({onLoad:f,onError:f,onAbort:f,headers:{},async:true,useCache:false,data:null,defer:false,forceXml:false},w||{});if(!("X-Requested-With" in r.headers)){r.headers["X-Requested-With"]="XMLHttpRequest";}return r;}function v(r){return[r,(/\?/.test(r)?"&":"?"),"a",new Date().getTime(),parseInt(Math.random()*100000)].join("");}function j(C,r,z){var y=s(),A=z.data&&(typeof z.data=="string"?z.data:h.data.encodeUrl(z.data)),w,x=new q(y,z);if(!z.useCache){r=v(r);}y.open(C,r,z.async);for(w in z.headers){y.setRequestHeader(w,z.headers[w]);}function B(){x.send=f;if(z.async){if(z.timeout){x._timeout=setTimeout(function(){a(x);var E=new b(y,true,x);c.fire(x,"error",E);},z.timeout*1000);}y.onreadystatechange=function(){if(y.readyState==4){x._timeout&&clearTimeout(x._timeout);x.completed=true;var E=new b(y,false,x);if(E.wasSuccessful){c.fire(x,"load",E);}else{c.fire(x,"error",E);}y.onreadystatechange=new Function();}};y.send(A);return x;}else{y.send(A);x.completed=true;var D=new b(y,false,x);if(D.wasSuccessful){c.fire(x,"load",D);}else{c.fire(x,"error",D);}return D;}}x.send=B;return z.defer?x:B();}var m={};m.get=function(r,w){w=d(w);return j("GET",r,w);};m.post=function(r,w,x){x=d(x);x.data=w;if(!x.headers["Content-Type"]){x.headers["Content-Type"]=t.POST_DEFAULT_CONTENT_TYPE;}return j("POST",r,x);};m.send=function(y,r,w,x){w=w||"";x=d(x);x.data=w;return j(y,r,x);};m.put=function(r,w,x){x=d(x);x.data=w;if(!x.headers["Content-Type"]){x.headers["Content-Type"]=t.POST_DEFAULT_CONTENT_TYPE;}return j("PUT",r,x);};m.del=function(r,w){w=d(w);return j("DELETE",r,w);};m.loadScript=function(w,z){var x=p.length,r,A=e+x,z=d(z),y=new q(x,z),w=z.useCache?w:v(w),B=window[o]||(window[o]={});if(z.onLoad!=f){B[A]=function(){y._timeout&&clearTimeout(y._timeout);y.completed=true;z.onLoad.apply(this,arguments);y.destroy();r=B[A]=undefined;delete B[A];};w=h.lang.interpolate(w,{callback:o+"."+A});}r=p[x]=document.createElement("script");if(z.charset){r.charset=z.charset;}c.addListener(y,"abort",z.onAbort);h.ready(function(){if(z.timeout){y._timeout=setTimeout(function(){a(y);z.onError();},z.timeout*1000);}if(h.env.opera){setTimeout(function(){if(r){r.src=w;}},0);}else{r.src=w;}document.body.appendChild(r);});return y;};function a(w){var r=w.nativeRequest,x=w._callbackIndex;w._timeout&&clearTimeout(w._timeout);if(r){r.onreadystatechange=new Function();r.abort();}else{if(x){window[o][e+x]=f;h.dom.get(p[x]).destroy();}}}function q(y,w){this._timeout=null;this._forceXml=w.forceXml;if(w.forceXml&&y.overrideMimeType){y.overrideMimeType("application/xml");}this.complete=false;if(typeof y=="number"){this._callbackIndex=y;}else{this.nativeRequest=y;}var x=["Load","Error","Abort"],r=0;for(;r<3;r++){c.addListener(this,x[r].toLowerCase(),w["on"+x[r]]);}}q.prototype={send:function(){},abort:function(){if(!this.completed&&!c.fire(this,"abort").defaultPrevented()){a(this);}return this;},destroy:function(){var r=this;if(this._callbackIndex!==undefined){setTimeout(function(){g(p[r._callbackIndex]).destroy();p[r._callbackIndex]=undefined;delete p[r._callbackIndex];},0);}return this;}};function b(x,r,w){c.Event.call(this);this._request=w;this.nativeResponse=x;this.status=r?408:x.status==1223?204:x.status;this.timedOut=!!r;this.wasSuccessful=(this.status>=200&&this.status<300)||this.status==304||(this.status==0&&x.responseText);}function l(){var r=this.header("Content-Type");return u.test(r)||r==="";}h.lang.extend(b,c.Event,{text:function(){return this.nativeResponse.responseText;},xml:function(){var w=this.nativeResponse;if((h.env.ie&&l.call(this))||(this._request._forceXml&&!this._request.nativeRequest.overrideMimeType&&window.ActiveXObject)){var r=new ActiveXObject("Microsoft.XMLDOM");r.loadXML(w.responseText);return r;}else{if(!w.responseXML){throw new Error(t.XML_ERR);}return w.responseXML;}},json:function(r){return h.data.decodeJson(this.text(),{safeMode:r});},header:function(r){return this.nativeResponse.getResponseHeader(r);},statusText:function(){return this.timedOut?"Request Timeout":this.nativeResponse.statusText;}});var n=function(r,x,y,w){this.url=r;this.data=x;this.isGet=y;this.opts=w;};n.prototype={_send:function(){this._addIframe();this._addForm();this._addTimeout();this.onLoad=this._handleResponse;this._submitForm();},_addIframe:function(){this.iframe=h.dom.create('<iframe style="visibility: hidden; position: absolute; height: 0;"></iframe>');var r=this.iframe[0],w=this,x=function(){if(w.onLoad){w.onLoad();}};if(r.attachEvent){r.attachEvent("onload",x);}else{r.onload=x;}g("body").append(this.iframe);},_addForm:function(){var x=this._window().document;if(h.env.ie){x.open();x.write("<html><body></body></html>");x.close();}var w=this.form=x.createElement("form");w.setAttribute("action",this.url);w.setAttribute("method",this.isGet?"GET":"POST");var r=x.getElementsByTagName("body")[0];r.appendChild(w);this._addFormData();},_addFormData:function(){for(var x in this.data){if(!this.data.hasOwnProperty(x)){continue;}if(this.data[x] instanceof Array){var r=this.data[x].length;for(var w=0;w<r;w++){this._addHiddenInput(x,this.data[x][w]);}}else{this._addHiddenInput(x,this.data[x]);}}},_addHiddenInput:function(w,x){var r=this._window().document.createElement("input");r.type="hidden";r.name=w;r.value=x;this.form.appendChild(r);},_window:function(){var r=this.iframe[0];if(r.contentWindow){return r.contentWindow;}throw new Error("could not get contentWindow from iframe");},_addTimeout:function(){var r=this;this.timeout=setTimeout(function(){var w;if(r.opts.hasOwnProperty("onTimeout")){try{r.opts.onTimeout();}catch(x){w=x;}}r._cleanup();if(w){throw new Error("error in xDomainPost onTimeout callback: "+w);}},(this.opts.timeout||10)*1000);},_handleResponse:function(){var w,r,y=this._window();try{r=y.location.href;}catch(x){w=x;}if(r!="about:blank"||w){clearTimeout(this.timeout);this.onLoad=this._readHandler;if("_fullBlankUrl" in this.opts){y.location=this.opts._fullBlankUrl;}else{y.location=window.location.protocol+"//"+window.location.host+(this.opts.blankUrl||"/favicon.ico");}}},_readHandler:function(){var r;if(this.opts.hasOwnProperty("onLoad")){try{this.opts.onLoad(this._window().name);}catch(w){r=w;}}this._cleanup();if(r){throw new Error("error in xDomainPost onLoad callback: "+r);}},_cleanup:function(){this.iframe.remove();},_submitForm:function(){var r=this;this._window().setTimeout(function(){r.form.submit();},0);}};m.xDomainPost=function(r,y,x){var w=new n(r,y,false,x);w._send();};m.xDomainGet=function(r,x){var w=new n(r,{},true,x);w._send();};h.net=m;}});(window.gloader||glow).module({name:"glow.tweens",library:["glow","1.7.3"],depends:[],builder:function(b){function a(c){return function(d){return 1-c(1-d);};}b.tweens={linear:function(){return function(c){return c;};},easeIn:function(c){c=c||2;return function(d){return Math.pow(1,c-1)*Math.pow(d,c);};},easeOut:function(c){return a(this.easeIn(c));},easeBoth:function(c){return this.combine(this.easeIn(c),this.easeOut(c));},overshootIn:function(c){return a(this.overshootOut(c));},overshootOut:function(c){c=c||1.70158;return function(d){if(d==0||d==1){return d;}return((d-=1)*d*((c+1)*d+c)+1);};},overshootBoth:function(c){return this.combine(this.overshootIn(c),this.overshootOut(c));},bounceIn:function(){return a(this.bounceOut());},bounceOut:function(){return function(c){if(c<(1/2.75)){return 7.5625*c*c;}else{if(c<(2/2.75)){return(7.5625*(c-=(1.5/2.75))*c+0.75);}else{if(c<(2.5/2.75)){return(7.5625*(c-=(2.25/2.75))*c+0.9375);}else{return(7.5625*(c-=(2.625/2.75))*c+0.984375);}}}};},bounceBoth:function(){return this.combine(this.bounceIn(),this.bounceOut());},elasticIn:function(c,d){return a(this.elasticOut(c,d));},elasticOut:function(c,d){return function(e){if(e==0||e==1){return e;}if(!d){d=0.3;}if(!c||c<1){c=1;var f=d/4;}else{var f=d/(2*Math.PI)*Math.asin(1/c);}return c*Math.pow(2,-10*e)*Math.sin((e-f)*(2*Math.PI)/d)+1;};},elasticBoth:function(c,d){d=d||0.45;return this.combine(this.elasticIn(c,d),this.elasticOut(c,d));},combine:function(d,c){return function(e){if(e<0.5){return d(e*2)/2;}else{return c((e-0.5)*2)/2+0.5;}};}};}});(window.gloader||glow).module({name:"glow.anim",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.tweens","glow.events","glow.dom"]],builder:function(glow){var $=glow.dom.get,manager,events=glow.events,dom=glow.dom,get=dom.get,hasUnits=/width|height|top$|bottom$|left$|right$|spacing$|indent$|font-size/,noNegatives=/width|height|padding|opacity/,usesYAxis=/height|top/,getUnit=/(\D+)$/,testElement=dom.create('<div style="position:absolute;visibility:hidden"></div>');function addEventsFromOpts(instance,opts,eventProps){for(var i=0,len=eventProps.length;i<len;i++){if(opts[eventProps[i]]){events.addListener(instance,eventProps[i].slice(2).toLowerCase(),opts[eventProps[i]]);}}}(function(){var queue=[],queueLen=0,intervalTime=13,interval;manager={addToQueue:function(anim){queue[queueLen++]=anim;anim._playing=true;anim._timeAnchor=anim._timeAnchor||new Date().valueOf();if(!interval){this.startInterval();}},removeFromQueue:function(anim){for(var i=0;i<queueLen;i++){if(queue[i]==anim){queue.splice(i,1);anim._timeAnchor=null;anim._playing=false;if(--queueLen==0){this.stopInterval();}return;}}},startInterval:function(){interval=window.setInterval(this.processQueue,intervalTime);},stopInterval:function(){window.clearInterval(interval);interval=null;},processQueue:function(){var anim,i,now=new Date().valueOf();for(i=0;i<queueLen;i++){anim=queue[i];if(anim.position==anim.duration){manager.removeFromQueue(anim);i--;events.fire(anim,"complete");if(anim._opts.destroyOnComplete){anim.destroy();}continue;}if(anim.useSeconds){anim.position=(now-anim._timeAnchor)/1000;if(anim.position>anim.duration){anim.position=anim.duration;}}else{anim.position++;}anim.value=anim.tween(anim.position/anim.duration);events.fire(anim,"frame");}}};})();function convertCssUnit(element,fromValue,toUnit,axis){var elmStyle=testElement[0].style,axisProp=(axis=="x")?"width":"height",startPixelValue,toUnitPixelValue;elmStyle.margin=elmStyle.padding=elmStyle.border="0";startPixelValue=testElement.css(axisProp,fromValue).insertAfter(element)[axisProp]();toUnitPixelValue=testElement.css(axisProp,10+toUnit)[axisProp]()/10;testElement.remove();return startPixelValue/toUnitPixelValue;}function keepWithinRange(num,start,end){if(start!==undefined&&num<start){return start;}if(end!==undefined&&num>end){return end;}return num;}function buildAnimFunction(element,spec){var cssProp,r=["a=(function(){"],rLen=1,fromUnit,unitDefault=[0,"px"],to,from,unit,a;for(cssProp in spec){r[rLen++]='element.css("'+cssProp+'", ';if(typeof spec[cssProp]!="object"){to=spec[cssProp];}else{to=spec[cssProp].to;}if((from=spec[cssProp].from)===undefined){if(cssProp=="font-size"||cssProp=="background-position"){throw new Error("From value must be set for "+cssProp);}from=element.css(cssProp);}if(hasUnits.test(cssProp)){unit=(getUnit.exec(to)||unitDefault)[1];fromUnit=(getUnit.exec(from)||unitDefault)[1];from=parseFloat(from)||0;to=parseFloat(to)||0;if(from&&unit!=fromUnit){if(cssProp=="font-size"){throw new Error("Units must be the same for font-size");}from=convertCssUnit(element,from+fromUnit,unit,usesYAxis.test(cssProp)?"y":"x");}if(noNegatives.test(cssProp)){r[rLen++]="keepWithinRange(("+(to-from)+" * this.value) + "+from+', 0) + "'+unit+'"';}else{r[rLen++]="("+(to-from)+" * this.value) + "+from+' + "'+unit+'"';}}else{if(!(isNaN(from)||isNaN(to))){from=Number(from);to=Number(to);r[rLen++]="("+(to-from)+" * this.value) + "+from;}else{if(cssProp.indexOf("color")!=-1){to=dom.parseCssColor(to);if(!glow.lang.hasOwnProperty(from,"r")){from=dom.parseCssColor(from);}r[rLen++]='"rgb(" + keepWithinRange(Math.round('+(to.r-from.r)+" * this.value + "+from.r+'), 0, 255) + "," + keepWithinRange(Math.round('+(to.g-from.g)+" * this.value + "+from.g+'), 0, 255) + "," + keepWithinRange(Math.round('+(to.b-from.b)+" * this.value + "+from.b+'), 0, 255) + ")"';}else{if(cssProp=="background-position"){var vals={},fromTo=["from","to"],unit=(getUnit.exec(from)||unitDefault)[1];vals.fromOrig=from.toString().split(/\s/);vals.toOrig=to.toString().split(/\s/);if(vals.fromOrig[1]===undefined){vals.fromOrig[1]="50%";}if(vals.toOrig[1]===undefined){vals.toOrig[1]="50%";}for(var i=0;i<2;i++){vals[fromTo[i]+"X"]=parseFloat(vals[fromTo[i]+"Orig"][0]);vals[fromTo[i]+"Y"]=parseFloat(vals[fromTo[i]+"Orig"][1]);vals[fromTo[i]+"XUnit"]=(getUnit.exec(vals[fromTo[i]+"Orig"][0])||unitDefault)[1];vals[fromTo[i]+"YUnit"]=(getUnit.exec(vals[fromTo[i]+"Orig"][1])||unitDefault)[1];}if((vals.fromXUnit!==vals.toXUnit)||(vals.fromYUnit!==vals.toYUnit)){throw new Error("Mismatched axis units cannot be used for "+cssProp);}r[rLen++]="("+(vals.toX-vals.fromX)+" * this.value + "+vals.fromX+') + "'+vals.fromXUnit+' " + ('+(vals.toY-vals.fromY)+" * this.value + "+vals.fromY+') + "'+vals.fromYUnit+'"';}}}}r[rLen++]=");";}r[rLen++]="})";return eval(r.join(""));}var r={};r.css=function(element,duration,spec,opts){element=get(element);var anim=new r.Animation(duration,opts);if(element[0]){events.addListener(anim,"frame",buildAnimFunction(element,spec));}return anim;};slideElement=function slideElement(element,duration,action,opts){duration=duration||0.5;element=$(element);opts=glow.lang.apply({tween:glow.tweens.easeBoth(),onStart:function(){},onComplete:function(){}},opts);var i=0,thatlength=element.length,completeHeight,fromHeight,channels=[],timeline;for(;i<thatlength;i++){if(action=="up"||(action=="toggle"&&element.slice(i,i+1).height()>0)){element[i].style.overflow="hidden";if(glow.env.ie<8){element[i].style.zoom=1;}completeHeight=0;fromHeight=element.slice(i,i+1).height();}else{if(action=="down"||(action=="toggle"&&element.slice(i,i+1).height()==0)){fromHeight=element.slice(i,i+1).height();element[i].style.height="";completeHeight=element.slice(i,i+1).height();if(completeHeight===0){element[i].style.height="auto";completeHeight=element.slice(i,i+1).height();}element[i].style.height=fromHeight+"px";}}channels[i]=[glow.anim.css(element[i],duration,{height:{from:fromHeight,to:completeHeight}},{tween:opts.tween})];}timeline=new glow.anim.Timeline(channels,{destroyOnComplete:true});events.addListener(timeline,"complete",function(){element.each(function(){if(this.style.height.slice(0,1)!="0"){this.style.height="";if(glow.dom.get(this).height()===0){this.style.height="auto";}}});});events.addListener(timeline,"start",opts.onStart);events.addListener(timeline,"complete",opts.onComplete);return timeline.start();};r.slideDown=function(element,duration,opts){return slideElement(element,duration,"down",opts);};r.slideUp=function(element,duration,opts){return slideElement(element,duration,"up",opts);};r.slideToggle=function(element,duration,opts){return slideElement(element,duration,"toggle",opts);};r.fadeOut=function(element,duration,opts){return r.fadeTo(element,0,duration,opts);};r.fadeIn=function(element,duration,opts){return r.fadeTo(element,1,duration,opts);};r.fadeTo=function(element,opacity,duration,opts){duration=duration||0.5;element=$(element);opts=glow.lang.apply({tween:glow.tweens.easeBoth(),onStart:function(){},onComplete:function(){}},opts);var i=0,thatlength=element.length,channels=[],timeline;for(;i<thatlength;i++){channels[i]=[glow.anim.css(element[i],duration,{opacity:opacity},{tween:opts.tween})];}timeline=new glow.anim.Timeline(channels,{destroyOnComplete:true});events.addListener(timeline,"start",opts.onStart);events.addListener(timeline,"complete",opts.onComplete);return timeline.start();};r.highlight=function(element,highlightColour,duration,opts){element=$(element);duration=duration||1;highlightColour=highlightColour||"#ffff99";opts=glow.lang.apply({tween:glow.tweens.easeBoth(),onStart:function(){},onComplete:function(){}},opts);var i=0,transArray=[],elmsLength=element.length,completeColour,channels=[],timeline;for(;i<elmsLength;i++){completeColour=opts.completeColour||element.slice(i,i+1).css("background-color");if(completeColour=="transparent"||completeColour==""){completeColour="#fff";}channels[i]=[r.css(element[i],duration,{"background-color":{from:highlightColour,to:completeColour}},{tween:opts.tween})];}timeline=new glow.anim.Timeline(channels,{destroyOnComplete:true});events.addListener(timeline,"start",opts.onStart);events.addListener(timeline,"complete",opts.onComplete);return timeline.start();};var animationEventConstructorNames=["onStart","onStop","onComplete","onResume","onFrame"];r.Animation=function(duration,opts){this._opts=opts=glow.lang.apply({useSeconds:true,tween:glow.tweens.linear(),destroyOnComplete:false,onStart:null,onStop:null,onComplete:null,onResume:null,onFrame:null},opts);this._playing=false;this._timeAnchor=null;this.duration=duration;this.useSeconds=opts.useSeconds;this.tween=opts.tween;this.position=0;this.value=0;addEventsFromOpts(this,opts,animationEventConstructorNames);};r.Animation.prototype={start:function(){if(this._playing){this.stop();}var e=events.fire(this,"start");if(e.defaultPrevented()){return this;}this._timeAnchor=null;this.position=0;manager.addToQueue(this);return this;},stop:function(){if(this._playing){var e=events.fire(this,"stop");if(e.defaultPrevented()){return this;}manager.removeFromQueue(this);}return this;},destroy:function(){this.stop();events.removeAllListeners(this);return this;},resume:function(){if(!this._playing){var e=events.fire(this,"resume");if(e.defaultPrevented()){return this;}this._timeAnchor=new Date().valueOf()-(this.position*1000);manager.addToQueue(this);}return this;},isPlaying:function(){return this._playing;},goTo:function(pos){this._timeAnchor=new Date().valueOf()-((this.position=pos)*1000);this.value=this.tween(this.duration&&this.position/this.duration);events.fire(this,"frame");return this;}};var timelineEventConstructorNames=["onStart","onStop","onComplete","onResume"];r.Timeline=function(channels,opts){this._opts=opts=glow.lang.apply({loop:false,destroyOnComplete:false,onStart:null,onStop:null,onComplete:null,onResume:null},opts);this._channels=(channels[0]&&channels[0].push)?channels:[channels];this._channelPos=[];this._playing=false;this.loop=opts.loop;var i,j,iLen,jLen,channel,allChannels=this._channels,totalDuration=0,channelDuration;for(i=0,iLen=allChannels.length;i<iLen;i++){channel=allChannels[i];channelDuration=0;for(j=0,jLen=channel.length;j<jLen;j++){if(typeof channel[j]=="number"){channel[j]=new r.Animation(channel[j]);}if(channel[j] instanceof r.Animation){if(!channel[j].useSeconds){throw new Error("Timelined animations must be timed in seconds");}channel[j]._timelineOffset=channelDuration*1000;channelDuration+=channel[j].duration;channel[j]._channelIndex=i;}}this.duration=totalDuration=Math.max(channelDuration,totalDuration);}this._controlAnim=new r.Animation(totalDuration);events.addListener(this._controlAnim,"frame",this._processFrame,this);events.addListener(this._controlAnim,"complete",this._complete,this);addEventsFromOpts(this,opts,timelineEventConstructorNames);};r.Timeline.prototype={_advanceChannel:function(i){var currentAnim=this._channels[i][this._channelPos[i]],nextAnim=this._channels[i][++this._channelPos[i]];if(currentAnim&&currentAnim._playing){currentAnim._playing=false;events.fire(currentAnim,"complete");if(currentAnim._opts.destroyOnComplete){currentAnim.destroy();}}if((nextAnim)!==undefined){if(typeof nextAnim=="function"){nextAnim();this._advanceChannel(i);}else{nextAnim.position=0;nextAnim._channelIndex=i;events.fire(nextAnim,"start");nextAnim._playing=true;}}},_complete:function(){if(this.loop){this.start();return;}this._playing=false;events.fire(this,"complete");if(this._opts.destroyOnComplete){this.destroy();}},_processFrame:function(){var i,len,anim,controlAnim=this._controlAnim,msFromStart=(new Date().valueOf())-controlAnim._timeAnchor;for(i=0,len=this._channels.length;i<len;i++){if(!(anim=this._channels[i][this._channelPos[i]])){continue;}anim.position=(msFromStart-anim._timelineOffset)/1000;if(anim.position>anim.duration){anim.position=anim.duration;}anim.value=anim.tween(anim.position/anim.duration);events.fire(anim,"frame");if(anim.position==anim.duration){this._advanceChannel(i);}}},start:function(){var e=events.fire(this,"start");if(e.defaultPrevented()){return this;}var i,iLen,j,jLen,anim;this._playing=true;for(i=0,iLen=this._channels.length;i<iLen;i++){this._channelPos[i]=-1;this._advanceChannel(i);for(j=this._channels[i].length;j;j--){anim=this._channels[i][j];if(anim instanceof r.Animation){anim.goTo(0);}}}this._controlAnim.start();return this;},stop:function(){if(this._playing){var e=events.fire(this,"stop");if(e.defaultPrevented()){return this;}this._playing=false;var anim;for(var i=0,len=this._channels.length;i<len;i++){anim=this._channels[i][this._channelPos[i]];if(anim instanceof r.Animation&&anim._playing){events.fire(anim,"stop");anim._playing=false;}}this._controlAnim.stop();}return this;},destroy:function(){var i,j;this.stop();events.removeAllListeners(this);this._controlAnim.destroy();i=this._channels.length;while(i--){j=this._channels[i].length;while(j--){if(this._channels[i][j].destroy){this._channels[i][j].destroy();}}}return this;},resume:function(){if(!this._playing){var e=events.fire(this,"resume");if(e.defaultPrevented()){return this;}this._playing=true;var anim;for(var i=0,len=this._channels.length;i<len;i++){anim=this._channels[i][this._channelPos[i]];if(anim instanceof r.Animation&&!anim._playing){events.fire(anim,"resume");anim._playing=true;}}this._controlAnim.resume();}return this;},isPlaying:function(){return this._playing;},goTo:function(pos){var i,j,k,channelsLen=this._channels.length,channelLen,anim,runningDuration;if(typeof pos=="number"){if(pos>this.duration){if(this.loop){pos=pos%this.duration;}else{pos=this.duration;}}this._controlAnim.goTo(pos);for(i=0;i<channelsLen;i++){runningDuration=0;for(j=0,channelLen=this._channels[i].length;j<channelLen;j++){anim=this._channels[i][j];if(anim instanceof r.Animation){if((runningDuration+anim.duration)>pos){this._channelPos[i]=j;anim.goTo(pos-runningDuration);break;}anim.goTo(anim.duration);runningDuration+=anim.duration;}}for(k=channelLen;k>j;k--){anim.goTo(0);}}}else{for(i=0;i<channelsLen;i++){runningDuration=0;for(j=0,channelLen=this._channels[i].length;j<channelLen;j++){anim=this._channels[i][j];if(anim===pos){return this.goTo(runningDuration);}if(anim instanceof r.Animation){runningDuration+=anim.duration;}}}throw"Animation not found in animation channels";}return this;}};glow.anim=r;}});(window.gloader||glow).module({name:"glow.forms",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.dom","glow.events","glow.anim","glow.net","glow.i18n"]],builder:function(g){var b=g.i18n,f=g.lang.interpolate;b.addLocaleModule("GLOW_FORMS","en",{TEST_MESSAGE_REQUIRED:"Value is required",TEST_MESSAGE_IS_NUMBER:"Must be a number.",TEST_MESSAGE_MIN:"The value must be at least {arg}.",TEST_MESSAGE_MAX:"The value must be less than {arg}.",TEST_MESSAGE_RANGE:"The value must be {min} or greater, and less than {max}.",TEST_MESSAGE_MIN_COUNT:"Must be have at least {arg} values.",TEST_MESSAGE_MAX_COUNT:"Must be have at most {arg} values.",TEST_MESSAGE_COUNT:"Must have {arg} values.",TEST_MESSAGE_REGEX:"Must be in the correct format.",TEST_MESSAGE_MIN_LEN:"Must be at least {arg} characters.",TEST_MESSAGE_MAX_LEN:"Must be at most {arg} characters.",TEST_MESSAGE_IS_EMAIL:"Must be a valid email address.",TEST_MESSAGE_SAME_AS:"Must be the same as: {arg}",TEST_MESSAGE_AJAX:"server responded",TEST_MESSAGE_IS:"Must be {arg}",TEST_MESSAGE_IS_NOT:"Must not be {arg}"});g.forms={};g.forms.Form=function(k,j){this.formNode=g.dom.get(k);if(!this.formNode[0]){throw"Could not find form. Possibly run before DOM ready.";}this._fields=[];this._result=null;this.opts=j||{};g.events.addListener(this,"validate",this.opts.onValidate||a.defaultFeedback);this._idleTimer=null;this._localeModule=b.getLocaleModule("GLOW_FORMS");var h=this;g.events.addListener(this.formNode,"submit",function(){h.validate("submit");return false;});};g.forms.Form.prototype.validate=function(h,j){this.eventName=h||"submit";this._result=new g.forms.ValidateResult(this.eventName);this._result.form=this;this._fieldCur=0;this._testCur=-1;this._fieldName=j;c.call(this);};var c=function(){this._testCur++;if(this._testCur>=this._fields[this._fieldCur]._tests.length){if(!d.call(this)){return;}}var j=this._fields[this._fieldCur]._tests[this._testCur];var h;if(j.opts.field){h=this.formNode.val()[j.opts.field]||"";j.isConditional=true;}else{h=this.formNode.val()[this._fields[this._fieldCur].name]||"";}if(!h.join){h=[h];}var k=function(l){return function(){e.apply(l,arguments);};}(this);j.opts.on=j.opts.on||"submit";if(this._result.eventName&&(" "+j.opts.on+" ").indexOf(" "+this._result.eventName+" ")!=-1){if(this._fieldName&&this._fieldName!=j.name){c.call(this);return;}if(typeof g.forms.tests[j.type]!="function"){throw"Unimplemented test: no test exists of type '"+j.type+"'.";}j.opts._localeModule=this._localeModule;g.forms.tests[j.type](h,j.opts,k,this.formNode.val());}else{c.call(this);}};var d=function(){this._fieldCur++;this._testCur=0;if(this._fieldCur>=this._fields.length){this._fieldCur=0;g.events.fire(this,"validate",this._result);if(this.eventName=="submit"&&this._result&&!this._result.defaultPrevented()){try{this.formNode[0].submit();}catch(h){throw new Error("Glow can't submit the form because the submit function can't be called. Perhaps that form's submit was replaced by an input element named 'submit'?");}}return false;}return true;};var e=function(h,j){if(typeof h=="boolean"){h=(h)?g.forms.PASS:g.forms.FAIL;}if(this._fields[this._fieldCur]._tests[this._testCur].isConditional&&h===g.forms.FAIL){h=g.forms.SKIP;}this._result.fields.push({name:this._fields[this._fieldCur].name,result:h,message:j});if(h!==g.forms.PASS){if(h===g.forms.FAIL){this._result.errorCount++;}this._testCur=this._fields[this._fieldCur]._tests.length;}c.call(this);};g.forms.Form.prototype.addTests=function(r){var q={name:r,_tests:[]};var j=function(s){return function(){s.validate.apply(s,["change",r]);};}(this);var n=function(s){return function(){s.validate.apply(s,["click",r]);};}(this);var o=function(s){return function(){s.validate.apply(s,["idle",r]);};}(this);for(var l=1;l<arguments.length;l++){var k=arguments[l][0];var p=(arguments[l].length>1)?arguments[l][1]:{};q._tests.push({name:r,type:k,opts:p});if(!j.added&&(" "+p.on+" ").indexOf(" change ")!=-1){var m=this.formNode.get("*").each(function(s){if(this.name==r){g.events.addListener(this,"change",j);j.added=true;}});}if(!n.added&&(" "+p.on+" ").indexOf(" click ")!=-1){var m=this.formNode.get("*").each(function(s){if(this.name==r){g.events.addListener(this,"click",n);n.added=true;}});}if(!o.added&&(" "+p.on+" ").indexOf(" idle ")!=-1){var h=(typeof p.delay!="undefined")?parseInt(p.delay):1000;var m=this.formNode.get("*").each(function(s){if(this.name==r){g.events.addListener(this,"keyup",function(u){return function(){window.clearTimeout(this.idleTimeoutID);if(this.value){this.idleTimeoutID=window.setTimeout(o,u);}};}(h));g.events.addListener(this,"blur",function(){window.clearTimeout(this.idleTimeoutID);});o.added=true;}});}}this._fields.push(q);return this;};g.forms.ValidateResult=function(h){g.events.Event.apply(this);this.eventName=h;this.errorCount=0;this.value=undefined;this.fields=[];};g.lang.extend(g.forms.ValidateResult,g.events.Event);g.forms.PASS=1;g.forms.FAIL=0;g.forms.SKIP=-1;g.forms.tests={required:function(j,m,n){var l=m.message||m._localeModule.TEST_MESSAGE_REQUIRED;for(var k=0,h=j.length;k<h;k++){if(/^\s*$/.test(j[k])){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},isNumber:function(j,m,n){var l=m.message||m._localeModule.TEST_MESSAGE_IS_NUMBER;for(var k=0,h=j.length;k<h;k++){if(j[k]==""||isNaN(j[k])){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},min:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MIN,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(Number(j[k])<Number(m.arg)){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},max:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MAX,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(Number(j[k])>Number(m.arg)){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},range:function(k,o,p){var n=o.arg.split("..");if(typeof n[0]=="undefined"||typeof n[1]=="undefined"){throw"Range test requires a parameter like 0..10.";}var m=o.message||f(o._localeModule.TEST_MESSAGE_RANGE,{min:n[0],max:n[1]});n[0]*=1;n[1]*=1;if(n[0]>n[1]){var j=n[0];n[0]=n[1];n[1]=j;}for(var l=0,h=k.length;l<h;l++){if(k[l]<n[0]||k[l]>n[1]){p(g.forms.FAIL,m);return;}}p(g.forms.PASS,m);},minCount:function(h,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MIN_COUNT,{arg:m.arg});var k=0;for(var j=0;j<h.length;j++){if(h[j]!=""){k++;}}if(k<m.arg){n(g.forms.FAIL,l);return;}n(g.forms.PASS,l);},maxCount:function(h,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MAX_COUNT,{arg:m.arg});var k=0;for(var j=0;j<h.length;j++){if(h[j]!=""){k++;}}if(k>m.arg){n(g.forms.FAIL,l);return;}n(g.forms.PASS,l);},count:function(h,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_COUNT,{arg:m.arg});var k=0;for(var j=0;j<h.length;j++){if(h[j]!=""){k++;}}if(k!=m.arg){n(g.forms.FAIL,l);return;}n(g.forms.PASS,l);},regex:function(j,n,o){var m=n.message||n._localeModule.TEST_MESSAGE_REGEX;var l=(typeof n.arg=="string")?new RegExp(n.arg):n.arg;for(var k=0,h=j.length;k<h;k++){if(!l.test(j[k])){o(g.forms.FAIL,m);return;}}o(g.forms.PASS,m);},minLen:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MIN_LEN,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(j[k].length<m.arg){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},maxLen:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_MAX_LEN,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(j[k].length>m.arg){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},isEmail:function(j,m,n){var l=m.message||m._localeModule.TEST_MESSAGE_IS_EMAIL;for(var k=0,h=j.length;k<h;k++){if(!/^\s*[a-z0-9._%+-]+@[a-z0-9.-]+\.[a-z]{2,}\s*$/i.test(j[k])){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},sameAs:function(j,n,p,o){var m=n.message||f(n._localeModule.TEST_MESSAGE_SAME_AS,{arg:n.arg});var k=o[n.arg];for(var l=0,h=j.length;l<h;l++){if(j[l]!=k){p(g.forms.FAIL,m);return;}}p(g.forms.PASS,m);},ajax:function(o,h,q,n){var m={},r=(h.message||h._localeModule.TEST_MESSAGE_AJAX);for(var k in n){if(typeof n[k]=="string"){m[k]=escape(n[k]);}else{if(typeof n[k].push!="undefined"){m[k]=g.lang.map(n[k],function(p){return escape(p);}).join(",");}}}var j=g.lang.interpolate(h.url,m);var l=g.net.get(j,{onLoad:function(p){var s=h.arg(p);if(typeof s.push=="undefined"){s=[s,r];}q(s[0],s[1]);},onError:function(p){alert("Error getting file: "+j);}});},custom:function(h,j,k){j.arg.apply(this,arguments);},is:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_IS,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(j[k]!=m.arg){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);},isNot:function(j,m,n){var l=m.message||f(m._localeModule.TEST_MESSAGE_IS_NOT,{arg:m.arg});for(var k=0,h=j.length;k<h;k++){if(j[k]==m.arg){n(g.forms.FAIL,l);return;}}n(g.forms.PASS,l);}};var a=g.forms.feedback={};a.defaultFeedback=(function(){var h;function l(){if(!h){h=g.dom.create('<input type="hidden" value="0" name="1.7.3" id="1.7.3" />').appendTo(document.body);}h[0].value++;}function k(o){var n=o.fields,r,q,s,p,m;for(p=0,m=n.length;p<m;p++){r=g.dom.get(o.form.formNode[0].elements[n[p].name]);q=g.dom.get("."+n[p].name.replace(/(\W)/g,"\\$1")+"-msgContainer");if(!q[0]&&r.length==1){q=o.form.formNode.get("label").filter(function(){return this.htmlFor==r[0].id;});}s=q.get("span.glow-errorMsg");if(n[p].result){s.remove();r.removeClass("glow-invalid");}else{if(q.length){if(!s[0]){q.append((s=g.dom.create('<span class="glow-errorMsg"></span>')));}s.text(n[p].message);r.addClass("glow-invalid");}}}}function j(p){var s=p.fields,v,n,o,r,m,q,t;p.form.formNode.get("div.glow-errorSummary").remove();n=g.dom.create('<div class="glow-errorSummary" tabindex="-1"><ul></ul></div>');o=n.get("ul");for(q=0,t=s.length;q<t;q++){v=g.dom.get(p.form.formNode[0].elements[s[q].name]);r=g.dom.get("."+s[q].name.replace(/(\W)/g,"\\$1")+"-prompt");if(!r[0]&&v.length==1){r=p.form.formNode.get("label").filter(function(){return this.htmlFor==v[0].id;});}if(r[0]){r.get("span.glow-errorMsg").remove();m=g.lang.trim(r.text());if(m.slice(-1)==":"){m=m.slice(0,-1);}}else{m=s[q].name.replace(/^\w/,function(w){return w.toUpperCase();});}if(!s[q].result){o.append(g.dom.create("<li></li>").text(m+": "+s[q].message));}}p.form.formNode.prepend(n.css("opacity","0"));g.anim.css(n,"0.5",{opacity:{from:0,to:1}},{tween:g.tweens.easeOut()}).start();try{n[0].focus();}catch(u){}l();}return function(m){if(m.eventName=="submit"){if(!m.errorCount){m.form.formNode.get("div.glow-errorSummary").remove();return;}j(m);}setTimeout(function(){k(m);},0);return false;};}());}});(window.gloader||glow).module({name:"glow.embed",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.dom","glow.data","glow.i18n"]],builder:function(n){var o=n.i18n;o.addLocaleModule("GLOW_EMBED","en",{FLASH_MESSAGE:"This content requires Flash Player version {min} (installed version: {installed})",NO_PLAYER_MESSAGE:"No Flash Flayer installed, or version is pre 6.0.0"});function b(s){var r="";for(var t in s){if(t.toLowerCase()=="flashvars"&&typeof s[t]=="object"){r+=' FlashVars="'+n.data.encodeUrl(s[t])+'"';}else{r+=" "+t+'="'+s[t]+'"';}}return r;}function q(s){var u="",t,v;for(t in s){if(t.toLowerCase()=="flashvars"&&typeof s[t]=="object"){v=n.data.encodeUrl(s[t]);}else{v=s[t];}u+='<param name="'+t+'" value="'+v+'" />\n';}return u;}function h(s,r){s=s||{};for(var t in r){if(typeof s[t]=="undefined"){s[t]=r[t];}else{if(typeof r[t]=="object"){s[t]=h(s[t],r[t]);}}}return s;}function c(){var r=(navigator.platform||navigator.userAgent);return r.match(/win/i)?"win":r.match(/mac/i)?"mac":"other";}function l(r){var t=/^WIN (\d+),(\d+),(\d+),\d+$/;var s=r.GetVariable("$version");if($match=t.exec(s)){return{major:parseInt($match[1]),minor:parseInt($match[2]),release:parseInt($match[3]),actual:s};}else{}}function k(){var s,u,v={major:0,minor:0,release:0},r=v;if(n.env.ie){try{u=new ActiveXObject("ShockwaveFlash.ShockwaveFlash.7");r=l(u);}catch(w){try{u=new ActiveXObject("ShockwaveFlash.ShockwaveFlash.6");try{u.AllowScriptAccess="always";r=l(u);}catch(w){r={major:6,minor:0,release:29};}}catch(w){}}}else{var t=/^Shockwave Flash\s*(\d+)\.(\d+)\s*\w(\d+)$/;if((u=navigator.plugins["Shockwave Flash"])&&(s=t.exec(u.description))){r={major:parseInt(s[1]),minor:parseInt(s[2]),release:parseInt(s[3]),actual:u.description};}}r.toString=function(){return this.major?[this.major,this.minor,this.release].join("."):o.getLocaleModule("GLOW_EMBED").NO_PLAYER_MESSAGE;};return r;}var m=k();function f(t){if(typeof t!="object"){var s=String(t).match(/^(\d+)(?:\.(\d+)(?:\.(\d+))?)?$/);if(!s){throw new Error('glow.embed._meetsVersionRequirements: invalid format for version string, require "n.n.n" or "n.n" or simply "n" where n is a numeric value');}t={major:parseInt(s[1],10),minor:parseInt(s[2]||0,10),release:parseInt(s[3]||0,10)};}var r=m,u=t;return(r.major>u.major||(r.major==u.major&&r.minor>u.minor)||(r.major==u.major&&r.minor==u.minor&&r.release>=u.release));}var e=n.env.ie?j:g;function g(t,r,s){return'<embed type="application/x-shockwave-flash" src="'+t+'"'+b(r)+b(s)+"></embed>";}function j(t,r,s){return'<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" '+b(r)+'><param name="movie" value="'+t+'" />'+q(s)+"</object>";}var a={},p=0;function d(){return n.UID+"FlashEmbed"+(p++);}a.Flash=function(x,r,w,v){v=h(v,{width:"100%",height:"100%",params:{allowscriptaccess:"always",allowfullscreen:"true",quality:"high"},attributes:{},message:n.lang.interpolate(o.getLocaleModule("GLOW_EMBED").FLASH_MESSAGE,{min:w,installed:m}),id:(v&&v.attributes&&v.attributes.id)||d()});r=n.dom.get(r);if(!r.length){throw new Error("glow.embed.Flash unable to locate container");}this.container=r;this.movie=null;this._displayErrorMessage=typeof v.message=="function"?v.message:function(){return v.message;};this.isSupported;if(this.isSupported=f(w)){var s=v.attributes,u=["id","width","height"],t=u.length;while(t--){if(v[u[t]]){s[u[t]]=v[u[t]];}}if(v.className){s["class"]=v.className;}this._embed_tag=e(x,s,v.params);}};a.Flash.version=function(){return m;};a.Flash.prototype.embed=function(){var s=this.container[0];if(this.isSupported){s.innerHTML=this._embed_tag;this.movie=s.firstChild;}else{var r=this._displayErrorMessage();if(r){s.innerHTML=r;}}return this;};n.embed=a;}});(window.gloader||glow).module({name:"glow.dragdrop",library:["glow","1.7.3"],depends:[["glow","1.7.3","glow.tweens","glow.events","glow.dom","glow.anim"]],builder:function(j){var c=j.events,k=c.addListener,v=c.fire,g=c.removeListener,q=j.dom,h=q.get,m=q.create;var n={},w=1000,a=(document.compatMode=="CSS1Compat"&&j.env.ie>=5)?true:false,x=(document.compatMode!="CSS1Compat"&&j.env.ie>=5)?true:false,d=j.env.ie>=5,s=["top","right","bottom","left"];function b(r,z){var A=r.prototype[z];var y="cached_"+z;r.prototype[z]=function(){if(y in this){return this[y];}return this[y]=A.apply(this,arguments);};}function u(A,z){var r=s.length,y;while(r--){y="margin-"+s[r];A.css(y,z.css(y));}}function f(r,y){var A=r.prototype[y];var z="cached_"+y;r.prototype[y]=function(B){if(!this[z]){this[z]={};}if(B in this[z]){return this[z][B];}return this[z][B]=A.apply(this,arguments);};}function t(A,z){for(var y=0,r=z.length;y<r;y++){delete A["cached_"+z[y]];}}var o=function(r){this.el=r;};o.prototype={val:function(r){var y=parseInt(this.el.css(r));return y||0;},width:function(){return this.borderWidth()-this.val("border-left-width")-this.val("border-right-width");},height:function(){return this.borderHeight()-this.val("border-top-width")-this.val("border-bottom-width");},offsetParentPageTop:function(){var r=this.el[0],z,y;while(r=r.offsetParent){if(h(r).css("position")!="static"){break;}}return r?h(r).offset().top:0;},offsetTop:function(){return this.el.position().top;},offsetLeft:function(){return this.el.position().left;},borderWidth:function(){var r=this.el[0].offsetWidth;if(j.env.khtml){r-=this.val("margin-left")+this.val("margin-right")+this.val("border-left-width")+this.val("border-right-width");}return r;},borderHeight:function(){if(this._logicalBottom){return this._logicalBottom-this.offsetTop();}var r=this.el[0].offsetHeight;if(j.env.khtml){r-=this.val("margin-top")+this.val("margin-bottom")+this.val("border-top-width")+this.val("border-bottom-width");}return r;},outerWidth:function(){return this.borderWidth()+this.val("margin-left")+this.val("margin-right");},outerHeight:function(){return this.borderHeight()+this.val("margin-top")+this.val("margin-bottom");},innerLeftPos:function(){return this.offsetLeft()+this.val("margin-left")+this.val("border-left-width")+this.val("padding-left");},innerTopPos:function(){return this.offsetTop()+this.val("margin-top")+this.val("border-top-width")+this.val("padding-top");},surroundWidth:function(){return this.val("border-left-width")+this.val("padding-left")+this.val("padding-right")+this.val("border-right-width");},surroundHeight:function(){return this.val("border-top-width")+this.val("padding-top")+this.val("padding-bottom")+this.val("border-bottom-width");},verticalCenter:function(){return this.offsetTop()+(this.outerHeight()/2);},horizontalCenter:function(){return this.offsetTop()+(this.outerWidth()/2);}};for(var p in o.prototype){if(p=="val"){f(o,p);}else{b(o,p);}}j.lang.apply(o.prototype,{resetPosition:function(){t(this,["offsetTop","offsetLeft","borderTopPos","borderLeftPos","innerTopPos","innerLeftPos","verticalCenter","horizontalCenter"]);},setLogicalBottom:function(r){this._logicalBottom=r;},boundsFor:function(z){var y,r,A=this.el.css("position");if(A!="static"){y=r=0;}else{y=this.innerTopPos();r=this.innerLeftPos();}return[y,r+this.width()-z.outerWidth(),y+this.height()-z.outerHeight(),r];},outerBounds:function(){var z=this.el.offset(),y=z.left,r=z.top;return[r,y+this.borderWidth(),r+this.borderHeight(),y];},intersectSize:function(z,A){var y=this.outerBounds(),r=z.outerBounds();if(A){y[1]++;r[1]++;y[2]++;r[2]++;}return(y[2]<r[0]?0:r[2]<y[0]?0:y[0]<r[0]?(y[2]<r[2]?y[2]-r[0]:r[2]-r[0]):r[2]<y[2]?r[2]-y[0]:y[2]-y[0])*(y[1]<r[3]?0:r[1]<y[3]?0:y[3]<r[3]?(y[1]<r[1]?y[1]-r[3]:r[1]-r[3]):r[1]<y[1]?r[1]-y[3]:y[1]-y[3]);},sizePlaceholder:function(C,D,A,y){var B=new o(C),z=this.el,r=D||z.css("position");C.css("display","none");z.after(C);C.css("width",(z[0].offsetWidth-B.surroundWidth())+"px").css("height",(z[0].offsetHeight-B.surroundHeight())+"px");u(C,z);C.remove();C.css("display","block");if(r!="static"){C.css("left",A+"px");C.css("top",y+"px");}C.css("position",r);},contains:function(z){var y=this.boundsFor(z),r=z.el.position(),B=r.top,A=r.left;return B>=y[0]&&A<=y[1]&&B<=y[2]&&A>=y[3];},containsPoint:function(y){var r=this.el.offset();return y.x>=r.left&&y.y>=r.top&&y.x<=r.left+this.borderWidth()&&y.y<=r.top+this.borderHeight();},positionedAncestorBox:function(){var r=this.el.parent(),y;while(r[0]){y=r.css("position")||"static";if(y=="relative"||y=="absolute"||y=="fixed"){return new o(r);}r=r.parent();}return null;}});function e(y){var r=y[0].tagName.toLowerCase()=="li"?"li":"div";var z=m("<"+r+"></"+r+">");if(r=="li"){z.css("list-style-type","none");}return z;}n.Draggable=function(A,B){this.element=h(A);this._opts=B=j.lang.apply({dragPrevention:["input","textarea","button","select","option","a"],placeholder:"spacer",placeholderClass:"glow-dragdrop-placeholder",step:{x:1,y:1}},B||{});if(typeof B.step=="number"){B.step={x:B.step,y:B.step};}else{B.step.x=B.step.x||1;B.step.y=B.step.y||1;}this._preventDrag=[];for(var y=0,r=B.dragPrevention.length;y<r;y++){this._preventDrag[y]=B.dragPrevention[y].toLowerCase();}if(B.container){this.container=h(B.container);}this._handle=B.handle&&this.element.get(B.handle)||this.element;if(B.dropTargets){this.dropTargets=h(B.dropTargets);}var z=this._listeners=[],y=0;if(B.onDrag){z[y++]=k(this,"drag",this._opts.onDrag,this);}if(B.onEnter){z[y++]=k(this,"enter",this._opts.onEnter,this);}if(B.onLeave){z[y++]=k(this,"leave",this._opts.onLeave,this);}if(B.onDrop){z[y++]=k(this,"drop",this._opts.onDrop,this);}if(B.onAfterDrop){z[y++]=k(this,"afterDrop",this._opts.onAfterDrop,this);}this._dragListener=k(this._handle,"mousedown",this._startDragMouse,this);return;};n.Draggable.prototype={_createPlaceholder:function(){var r=this.element,z,y=this._box;if(this._opts.placeholder=="clone"){z=r.clone();}else{z=e(r);}if(this._opts.placeholderClass){z.addClass(this._opts.placeholderClass);}y.sizePlaceholder(z,null,this._startLeft,this._startTop);r.after(z);this._placeholder=z;},_removePlaceholder:function(){this._placeholder.remove();},_resetPosition:function(){var G=this._preDragPosition,r=this.element,z=this._box,B=this._startOffset,D=r.css("position"),F,C;z.resetPosition();var A=z.el.position(),y={x:A.left,y:A.top};if(this._placeholder||this._dropIndicator){r.remove();}if(G=="static"&&y.y==B.y&&y.x==B.x){r.css("position","static");r.css("left","");r.css("top","");}else{r.css("z-index",this._preDragZIndex);r.css("position",G=="static"?"relative":G);if(G=="static"){F=y.x-B.x;C=y.y-B.y;}else{if(G=="relative"&&D!="relative"){F=this._startLeft+(y.x-B.x);C=this._startTop+(y.y-B.y);}}if(D!=G){r.css("left",F?F+"px":"");r.css("top",C?C+"px":"");}}if(this._dropIndicator){var E=this._dropIndicator.parent()[0];if(E){E.replaceChild(r[0],this._dropIndicator[0]);}delete this._dropIndicator;if(this._placeholder){this._placeholder.remove();delete this._placeholder;}r.css("position",G);if(G=="relative"&&D!="relative"){r.css("left",this._startLeft);r.css("top",this._startTop);}}else{if(this._placeholder){var E=this._placeholder.parent()[0];if(E){E.replaceChild(r[0],this._placeholder[0]);}delete this._placeholder;}}},_startDragMouse:function(I){var D=this._preventDrag,y=I.source,M=y.tagName.toLowerCase();for(var F=0,E=D.length;F<E;F++){if(D[F]==M){return;}}if(v(this,"drag").defaultPrevented()){return;}if(this._dragging==1){return this.endDrag();}else{if(this._dragging){return;}}this._dragging=1;var B=this.element,A=this.container,r=this._opts,H=this._box=new o(B),C=r.step;this._preDragPosition=B.css("position");var G=H.el.position(),J=this._startOffset={x:G.left,y:G.top};if(A){this._containerBox=new o(A);this._bounds=this._containerBox.boundsFor(H);if(C.x!=1){this._bounds[3]-=(this._bounds[3]-J.x)%C.x;this._bounds[1]-=(this._bounds[1]-J.x)%C.x;}if(C.y!=1){this._bounds[0]-=(this._bounds[0]-J.y)%C.y;this._bounds[2]-=(this._bounds[2]-J.y)%C.y;}}else{delete this._bounds;}this._mouseStart={x:I.pageX,y:I.pageY};this._preDragStyle=B.attr("style");this._preDragZIndex=B.css("z-index");B.css("z-index",w++);this._startLeft=B[0].style.left?parseInt(B[0].style.left):0;this._startTop=B[0].style.top?parseInt(B[0].style.top):0;if(r.placeholder&&r.placeholder!="none"){this._createPlaceholder();}B.css("position","absolute");B.css("left",J.x+"px");B.css("top",J.y+"px");if(a){this._scrollY=document.documentElement.scrollTop;this._innerHeight=document.documentElement.clientHeight;}else{if(x){this._scrollY=document.body.scrollTop;this._innerHeight=document.body.clientHeight;}else{this._scrollY=window.scrollY;this._innerHeight=window.innerHeight;}}var K=function(){return false;},L=document.documentElement;if(this.dropTargets){var z=new c.Event();z.draggable=this;for(var F=0,E=this.dropTargets.length;F<E;F++){v(this.dropTargets[F],"active",z);}this._mousePos={x:I.pageX,y:I.pageY};this._testForDropTargets();}this._dragListeners=[k(L,"selectstart",K),k(L,"dragstart",K),k(L,"mousedown",K),k(L,"mousemove",this._dragMouse,this),k(L,"mouseup",this._releaseElement,this)];return false;},_dragMouse:function(C){var r=this.element,y=this._opts.axis,D=y=="y"?this._startOffset.x:(this._startOffset.x+C.pageX-this._mouseStart.x),B=y=="x"?this._startOffset.y:(this._startOffset.y+C.pageY-this._mouseStart.y),A=this._bounds,z=this._opts.step;if(z.x!=1){D=Math.round((D-this._startOffset.x)/z.x)*z.x+this._startOffset.x;}if(z.y!=1){B=Math.round((B-this._startOffset.y)/z.y)*z.y+this._startOffset.y;}if(A){if(y!="y"){D=D<A[3]?A[3]:D>A[1]?A[1]:D;}if(y!="x"){B=B<A[0]?A[0]:B>A[2]?A[2]:B;}}r[0].style.left=D+"px";r[0].style.top=B+"px";if(this.dropTargets){this._mousePos={x:C.pageX,y:C.pageY};}if(d&&C.nativeEvent.button==0){this._releaseElement(C);return false;}return false;},_testForDropTargets:function(H){if(!this._lock){this._lock=0;}if(H){this._lock--;}else{if(this.lock){return;}}if(this._dragging!=1){return;}var z=this.activeTarget,y,R=this.dropTargets,S,C,F=this._box,K=this._mousePos;F.resetPosition();var D=0;for(var L=0,J=R.length;L<J;L++){S=R[L];C=S._box;if(S._opts.tolerance=="contained"){if(C.contains(F)){y=S;break;}}else{if(S._opts.tolerance=="cursor"){if(C.containsPoint(K)){y=S;break;}}else{var M=C.intersectSize(F,true);if(M>D){D=M;y=S;}}}}this.activeTarget=y;if(y!==z){if(y){var I=new c.Event();I.draggable=this;v(y,"enter",I);var r=new c.Event();r.dropTarget=y;v(this,"enter",r);}if(z){var E=new c.Event();E.draggable=this;v(z,"leave",E);var G=new c.Event();G.dropTarget=z;v(this,"leave",G);}}if(y&&y._opts.dropIndicator!="none"){var P,N=y._childBoxes,A=y._children;F.resetPosition();var Q=y._box.innerTopPos();var T=K.y-F.offsetParentPageTop();var O=0;for(var L=0,J=N.length;L<J;L++){if(A[L]==this.element[0]){continue;}P=N[L];Q+=P.outerHeight();if(T<=Q){if(y._dropIndicatorAt!=L){h(P.el).before(y._dropIndicator);y._dropIndicatorAt=L;}O=1;break;}}if(!O){if(P){h(P.el).after(y._dropIndicator);y._dropIndicatorAt=L+1;}else{y.element.append(y._dropIndicator);y._dropIndicatorAt=0;}}}this._lock++;var B=this;setTimeout(function(){B._testForDropTargets(1);},100);},_releaseElement:function(){if(this._dragging!=1){return;}this._dragging=2;var z,r;var D=this.dropTargets,C=this.activeTarget;if(D){for(z=0,r=D.length;z<r;z++){var B=new c.Event();B.draggable=this;B.droppedOnThis=C&&C==D[z];v(D[z],"inactive",B);}}if(C){var B=new c.Event();B.draggable=this;v(C,"drop",B);}var y=this._dragListeners;for(z=0,r=y.length;z<r;z++){c.removeListener(y[z]);}var A=v(this,"drop");if(!A.defaultPrevented()&&this.dropTargets){this.returnHome();}else{this.endDrag();}},endDrag:function(){if(this._dragging!=2){return;}this._dragging=0;if(this._reset){this._reset();delete this._reset;}if(this.placeholder){this.placeholder.remove();}this._resetPosition();delete this.activeTarget;v(this,"afterDrop");},returnHome:function(G){var z=(G)?G:j.tweens.linear(),C,r,A=this.element,E=this._box.el.position(),y=Math.pow(Math.pow(this._startOffset.x-E.left,2)+Math.pow(this._startOffset.y-E.top,2),0.5),B=0.3+(y/1000);var D=[[j.anim.css(A,B,{left:this._startOffset.x,top:this._startOffset.y},{tween:z})]];if(this._dropIndicator){D.push([j.anim.css(this._dropIndicator,B-0.1,{opacity:{to:0}})]);}var F=new j.anim.Timeline(D);k(F,"complete",function(){this.endDrag();},this);F.start();return;}};var l=0;n.DropTarget=function(r,y){r=this.element=h(r);if(!r.length){throw"no element passed into DropTarget constuctor";}if(r.length>1){throw"more than one element passed into DropTarget constructor";}this._id=++l;this._opts=y=j.lang.apply({dropIndicator:"none",dropIndicatorClass:"glow-dragdrop-dropindicator",tolerance:"intersect"},y||{});if(y.onActive){k(this,"active",y.onActive);}if(y.onInactive){k(this,"inactive",y.onInactive);}if(y.onEnter){k(this,"enter",y.onEnter);}if(y.onLeave){k(this,"leave",y.onLeave);}if(y.onDrop){k(this,"drop",y.onDrop);}k(this,"active",this._onActive);k(this,"inactive",this._onInactive);return this;};n.DropTarget.prototype={setLogicalBottom:function(r){this._logicalBottom=r;},_onActive:function(A){var y=A.draggable;this._box=new o(this.element);if(this._logicalBottom){this._box.setLogicalBottom(this._logicalBottom);}if(this._opts.dropIndicator=="none"){return;}this._onEnterListener=k(this,"enter",this._onEnter);this._onLeaveListener=k(this,"leave",this._onLeave);this._dropIndicator=e(y.element);if(this._opts.dropIndicatorClass){this._dropIndicator.addClass(this._opts.dropIndicatorClass);}y._box.sizePlaceholder(this._dropIndicator,"relative",0,0);var z=this._children=h(this.element.children()).filter(function(){var B=h(this);return(!A.draggable._placeholder||!B.eq(A.draggable._placeholder))&&(!this._dropIndicator||!B.eq(this._dropIndicator));});var r=this._childBoxes=[];z.each(function(B){r[B]=new o(h(z[B]));});},_onInactive:function(r){g(this._onEnterListener);g(this._onLeaveListener);delete this._box;if(this._opts.dropIndicator=="none"){return;}if(!r.droppedOnThis&&this._dropIndicator){this._dropIndicator.remove();delete this._dropIndicator;}delete this._childBoxes;delete this._children;},_onEnter:function(){this._dropIndicatorAt=-1;},_onLeave:function(){this._dropIndicator.remove();},moveToPosition:function(y){var C=this._dropIndicator,z=new o(C);var B=parseInt(C.css("margin-left"))||0,A=parseInt(C.css("margin-top"))||0,r=z.el.position();y._startOffset={x:r.left,y:r.top};y._dropIndicator=C;delete this._dropIndicator;}};j.dragdrop=n;}});
-/*@end @*/
diff --git a/ricoClient/js/baselibs/prototype.js b/ricoClient/js/baselibs/prototype.js
deleted file mode 100644 (file)
index 474b223..0000000
+++ /dev/null
@@ -1,6082 +0,0 @@
-/*  Prototype JavaScript framework, version 1.7
- *  (c) 2005-2010 Sam Stephenson
- *
- *  Prototype is freely distributable under the terms of an MIT-style license.
- *  For details, see the Prototype web site: http://www.prototypejs.org/
- *
- *--------------------------------------------------------------------------*/
-
-var Prototype = {
-
-  Version: '1.7',
-
-  Browser: (function(){
-    var ua = navigator.userAgent;
-    var isOpera = Object.prototype.toString.call(window.opera) == '[object Opera]';
-    return {
-      IE:             !!window.attachEvent && !isOpera,
-      Opera:          isOpera,
-      WebKit:         ua.indexOf('AppleWebKit/') > -1,
-      Gecko:          ua.indexOf('Gecko') > -1 && ua.indexOf('KHTML') === -1,
-      MobileSafari:   /Apple.*Mobile/.test(ua)
-    }
-  })(),
-
-  BrowserFeatures: {
-    XPath: !!document.evaluate,
-
-    SelectorsAPI: !!document.querySelector,
-
-    ElementExtensions: (function() {
-      var constructor = window.Element || window.HTMLElement;
-      return !!(constructor && constructor.prototype);
-    })(),
-    SpecificElementExtensions: (function() {
-      if (typeof window.HTMLDivElement !== 'undefined')
-        return true;
-
-      var div = document.createElement('div'),
-          form = document.createElement('form'),
-          isSupported = false;
-
-      if (div['__proto__'] && (div['__proto__'] !== form['__proto__'])) {
-        isSupported = true;
-      }
-
-      div = form = null;
-
-      return isSupported;
-    })()
-  },
-
-  ScriptFragment: '<script[^>]*>([\\S\\s]*?)<\/script>',
-  JSONFilter: /^\/\*-secure-([\s\S]*)\*\/\s*$/,
-
-  emptyFunction: function() { },
-
-  K: function(x) { return x }
-};
-
-if (Prototype.Browser.MobileSafari)
-  Prototype.BrowserFeatures.SpecificElementExtensions = false;
-
-
-var Abstract = { };
-
-
-var Try = {
-  these: function() {
-    var returnValue;
-
-    for (var i = 0, length = arguments.length; i < length; i++) {
-      var lambda = arguments[i];
-      try {
-        returnValue = lambda();
-        break;
-      } catch (e) { }
-    }
-
-    return returnValue;
-  }
-};
-
-/* Based on Alex Arnell's inheritance implementation. */
-
-var Class = (function() {
-
-  var IS_DONTENUM_BUGGY = (function(){
-    for (var p in { toString: 1 }) {
-      if (p === 'toString') return false;
-    }
-    return true;
-  })();
-
-  function subclass() {};
-  function create() {
-    var parent = null, properties = $A(arguments);
-    if (Object.isFunction(properties[0]))
-      parent = properties.shift();
-
-    function klass() {
-      this.initialize.apply(this, arguments);
-    }
-
-    Object.extend(klass, Class.Methods);
-    klass.superclass = parent;
-    klass.subclasses = [];
-
-    if (parent) {
-      subclass.prototype = parent.prototype;
-      klass.prototype = new subclass;
-      parent.subclasses.push(klass);
-    }
-
-    for (var i = 0, length = properties.length; i < length; i++)
-      klass.addMethods(properties[i]);
-
-    if (!klass.prototype.initialize)
-      klass.prototype.initialize = Prototype.emptyFunction;
-
-    klass.prototype.constructor = klass;
-    return klass;
-  }
-
-  function addMethods(source) {
-    var ancestor   = this.superclass && this.superclass.prototype,
-        properties = Object.keys(source);
-
-    if (IS_DONTENUM_BUGGY) {
-      if (source.toString != Object.prototype.toString)
-        properties.push("toString");
-      if (source.valueOf != Object.prototype.valueOf)
-        properties.push("valueOf");
-    }
-
-    for (var i = 0, length = properties.length; i < length; i++) {
-      var property = properties[i], value = source[property];
-      if (ancestor && Object.isFunction(value) &&
-          value.argumentNames()[0] == "$super") {
-        var method = value;
-        value = (function(m) {
-          return function() { return ancestor[m].apply(this, arguments); };
-        })(property).wrap(method);
-
-        value.valueOf = method.valueOf.bind(method);
-        value.toString = method.toString.bind(method);
-      }
-      this.prototype[property] = value;
-    }
-
-    return this;
-  }
-
-  return {
-    create: create,
-    Methods: {
-      addMethods: addMethods
-    }
-  };
-})();
-(function() {
-
-  var _toString = Object.prototype.toString,
-      NULL_TYPE = 'Null',
-      UNDEFINED_TYPE = 'Undefined',
-      BOOLEAN_TYPE = 'Boolean',
-      NUMBER_TYPE = 'Number',
-      STRING_TYPE = 'String',
-      OBJECT_TYPE = 'Object',
-      FUNCTION_CLASS = '[object Function]',
-      BOOLEAN_CLASS = '[object Boolean]',
-      NUMBER_CLASS = '[object Number]',
-      STRING_CLASS = '[object String]',
-      ARRAY_CLASS = '[object Array]',
-      DATE_CLASS = '[object Date]',
-      NATIVE_JSON_STRINGIFY_SUPPORT = window.JSON &&
-        typeof JSON.stringify === 'function' &&
-        JSON.stringify(0) === '0' &&
-        typeof JSON.stringify(Prototype.K) === 'undefined';
-
-  function Type(o) {
-    switch(o) {
-      case null: return NULL_TYPE;
-      case (void 0): return UNDEFINED_TYPE;
-    }
-    var type = typeof o;
-    switch(type) {
-      case 'boolean': return BOOLEAN_TYPE;
-      case 'number':  return NUMBER_TYPE;
-      case 'string':  return STRING_TYPE;
-    }
-    return OBJECT_TYPE;
-  }
-
-  function extend(destination, source) {
-    for (var property in source)
-      destination[property] = source[property];
-    return destination;
-  }
-
-  function inspect(object) {
-    try {
-      if (isUndefined(object)) return 'undefined';
-      if (object === null) return 'null';
-      return object.inspect ? object.inspect() : String(object);
-    } catch (e) {
-      if (e instanceof RangeError) return '...';
-      throw e;
-    }
-  }
-
-  function toJSON(value) {
-    return Str('', { '': value }, []);
-  }
-
-  function Str(key, holder, stack) {
-    var value = holder[key],
-        type = typeof value;
-
-    if (Type(value) === OBJECT_TYPE && typeof value.toJSON === 'function') {
-      value = value.toJSON(key);
-    }
-
-    var _class = _toString.call(value);
-
-    switch (_class) {
-      case NUMBER_CLASS:
-      case BOOLEAN_CLASS:
-      case STRING_CLASS:
-        value = value.valueOf();
-    }
-
-    switch (value) {
-      case null: return 'null';
-      case true: return 'true';
-      case false: return 'false';
-    }
-
-    type = typeof value;
-    switch (type) {
-      case 'string':
-        return value.inspect(true);
-      case 'number':
-        return isFinite(value) ? String(value) : 'null';
-      case 'object':
-
-        for (var i = 0, length = stack.length; i < length; i++) {
-          if (stack[i] === value) { throw new TypeError(); }
-        }
-        stack.push(value);
-
-        var partial = [];
-        if (_class === ARRAY_CLASS) {
-          for (var i = 0, length = value.length; i < length; i++) {
-            var str = Str(i, value, stack);
-            partial.push(typeof str === 'undefined' ? 'null' : str);
-          }
-          partial = '[' + partial.join(',') + ']';
-        } else {
-          var keys = Object.keys(value);
-          for (var i = 0, length = keys.length; i < length; i++) {
-            var key = keys[i], str = Str(key, value, stack);
-            if (typeof str !== "undefined") {
-               partial.push(key.inspect(true)+ ':' + str);
-             }
-          }
-          partial = '{' + partial.join(',') + '}';
-        }
-        stack.pop();
-        return partial;
-    }
-  }
-
-  function stringify(object) {
-    return JSON.stringify(object);
-  }
-
-  function toQueryString(object) {
-    return $H(object).toQueryString();
-  }
-
-  function toHTML(object) {
-    return object && object.toHTML ? object.toHTML() : String.interpret(object);
-  }
-
-  function keys(object) {
-    if (Type(object) !== OBJECT_TYPE) { throw new TypeError(); }
-    var results = [];
-    for (var property in object) {
-      if (object.hasOwnProperty(property)) {
-        results.push(property);
-      }
-    }
-    return results;
-  }
-
-  function values(object) {
-    var results = [];
-    for (var property in object)
-      results.push(object[property]);
-    return results;
-  }
-
-  function clone(object) {
-    return extend({ }, object);
-  }
-
-  function isElement(object) {
-    return !!(object && object.nodeType == 1);
-  }
-
-  function isArray(object) {
-    return _toString.call(object) === ARRAY_CLASS;
-  }
-
-  var hasNativeIsArray = (typeof Array.isArray == 'function')
-    && Array.isArray([]) && !Array.isArray({});
-
-  if (hasNativeIsArray) {
-    isArray = Array.isArray;
-  }
-
-  function isHash(object) {
-    return object instanceof Hash;
-  }
-
-  function isFunction(object) {
-    return _toString.call(object) === FUNCTION_CLASS;
-  }
-
-  function isString(object) {
-    return _toString.call(object) === STRING_CLASS;
-  }
-
-  function isNumber(object) {
-    return _toString.call(object) === NUMBER_CLASS;
-  }
-
-  function isDate(object) {
-    return _toString.call(object) === DATE_CLASS;
-  }
-
-  function isUndefined(object) {
-    return typeof object === "undefined";
-  }
-
-  extend(Object, {
-    extend:        extend,
-    inspect:       inspect,
-    toJSON:        NATIVE_JSON_STRINGIFY_SUPPORT ? stringify : toJSON,
-    toQueryString: toQueryString,
-    toHTML:        toHTML,
-    keys:          Object.keys || keys,
-    values:        values,
-    clone:         clone,
-    isElement:     isElement,
-    isArray:       isArray,
-    isHash:        isHash,
-    isFunction:    isFunction,
-    isString:      isString,
-    isNumber:      isNumber,
-    isDate:        isDate,
-    isUndefined:   isUndefined
-  });
-})();
-Object.extend(Function.prototype, (function() {
-  var slice = Array.prototype.slice;
-
-  function update(array, args) {
-    var arrayLength = array.length, length = args.length;
-    while (length--) array[arrayLength + length] = args[length];
-    return array;
-  }
-
-  function merge(array, args) {
-    array = slice.call(array, 0);
-    return update(array, args);
-  }
-
-  function argumentNames() {
-    var names = this.toString().match(/^[\s\(]*function[^(]*\(([^)]*)\)/)[1]
-      .replace(/\/\/.*?[\r\n]|\/\*(?:.|[\r\n])*?\*\//g, '')
-      .replace(/\s+/g, '').split(',');
-    return names.length == 1 && !names[0] ? [] : names;
-  }
-
-  function bind(context) {
-    if (arguments.length < 2 && Object.isUndefined(arguments[0])) return this;
-    var __method = this, args = slice.call(arguments, 1);
-    return function() {
-      var a = merge(args, arguments);
-      return __method.apply(context, a);
-    }
-  }
-
-  function bindAsEventListener(context) {
-    var __method = this, args = slice.call(arguments, 1);
-    return function(event) {
-      var a = update([event || window.event], args);
-      return __method.apply(context, a);
-    }
-  }
-
-  function curry() {
-    if (!arguments.length) return this;
-    var __method = this, args = slice.call(arguments, 0);
-    return function() {
-      var a = merge(args, arguments);
-      return __method.apply(this, a);
-    }
-  }
-
-  function delay(timeout) {
-    var __method = this, args = slice.call(arguments, 1);
-    timeout = timeout * 1000;
-    return window.setTimeout(function() {
-      return __method.apply(__method, args);
-    }, timeout);
-  }
-
-  function defer() {
-    var args = update([0.01], arguments);
-    return this.delay.apply(this, args);
-  }
-
-  function wrap(wrapper) {
-    var __method = this;
-    return function() {
-      var a = update([__method.bind(this)], arguments);
-      return wrapper.apply(this, a);
-    }
-  }
-
-  function methodize() {
-    if (this._methodized) return this._methodized;
-    var __method = this;
-    return this._methodized = function() {
-      var a = update([this], arguments);
-      return __method.apply(null, a);
-    };
-  }
-
-  return {
-    argumentNames:       argumentNames,
-    bind:                bind,
-    bindAsEventListener: bindAsEventListener,
-    curry:               curry,
-    delay:               delay,
-    defer:               defer,
-    wrap:                wrap,
-    methodize:           methodize
-  }
-})());
-
-
-
-(function(proto) {
-
-
-  function toISOString() {
-    return this.getUTCFullYear() + '-' +
-      (this.getUTCMonth() + 1).toPaddedString(2) + '-' +
-      this.getUTCDate().toPaddedString(2) + 'T' +
-      this.getUTCHours().toPaddedString(2) + ':' +
-      this.getUTCMinutes().toPaddedString(2) + ':' +
-      this.getUTCSeconds().toPaddedString(2) + 'Z';
-  }
-
-
-  function toJSON() {
-    return this.toISOString();
-  }
-
-  if (!proto.toISOString) proto.toISOString = toISOString;
-  if (!proto.toJSON) proto.toJSON = toJSON;
-
-})(Date.prototype);
-
-
-RegExp.prototype.match = RegExp.prototype.test;
-
-RegExp.escape = function(str) {
-  return String(str).replace(/([.*+?^=!:${}()|[\]\/\\])/g, '\\$1');
-};
-var PeriodicalExecuter = Class.create({
-  initialize: function(callback, frequency) {
-    this.callback = callback;
-    this.frequency = frequency;
-    this.currentlyExecuting = false;
-
-    this.registerCallback();
-  },
-
-  registerCallback: function() {
-    this.timer = setInterval(this.onTimerEvent.bind(this), this.frequency * 1000);
-  },
-
-  execute: function() {
-    this.callback(this);
-  },
-
-  stop: function() {
-    if (!this.timer) return;
-    clearInterval(this.timer);
-    this.timer = null;
-  },
-
-  onTimerEvent: function() {
-    if (!this.currentlyExecuting) {
-      try {
-        this.currentlyExecuting = true;
-        this.execute();
-        this.currentlyExecuting = false;
-      } catch(e) {
-        this.currentlyExecuting = false;
-        throw e;
-      }
-    }
-  }
-});
-Object.extend(String, {
-  interpret: function(value) {
-    return value == null ? '' : String(value);
-  },
-  specialChar: {
-    '\b': '\\b',
-    '\t': '\\t',
-    '\n': '\\n',
-    '\f': '\\f',
-    '\r': '\\r',
-    '\\': '\\\\'
-  }
-});
-
-Object.extend(String.prototype, (function() {
-  var NATIVE_JSON_PARSE_SUPPORT = window.JSON &&
-    typeof JSON.parse === 'function' &&
-    JSON.parse('{"test": true}').test;
-
-  function prepareReplacement(replacement) {
-    if (Object.isFunction(replacement)) return replacement;
-    var template = new Template(replacement);
-    return function(match) { return template.evaluate(match) };
-  }
-
-  function gsub(pattern, replacement) {
-    var result = '', source = this, match;
-    replacement = prepareReplacement(replacement);
-
-    if (Object.isString(pattern))
-      pattern = RegExp.escape(pattern);
-
-    if (!(pattern.length || pattern.source)) {
-      replacement = replacement('');
-      return replacement + source.split('').join(replacement) + replacement;
-    }
-
-    while (source.length > 0) {
-      if (match = source.match(pattern)) {
-        result += source.slice(0, match.index);
-        result += String.interpret(replacement(match));
-        source  = source.slice(match.index + match[0].length);
-      } else {
-        result += source, source = '';
-      }
-    }
-    return result;
-  }
-
-  function sub(pattern, replacement, count) {
-    replacement = prepareReplacement(replacement);
-    count = Object.isUndefined(count) ? 1 : count;
-
-    return this.gsub(pattern, function(match) {
-      if (--count < 0) return match[0];
-      return replacement(match);
-    });
-  }
-
-  function scan(pattern, iterator) {
-    this.gsub(pattern, iterator);
-    return String(this);
-  }
-
-  function truncate(length, truncation) {
-    length = length || 30;
-    truncation = Object.isUndefined(truncation) ? '...' : truncation;
-    return this.length > length ?
-      this.slice(0, length - truncation.length) + truncation : String(this);
-  }
-
-  function strip() {
-    return this.replace(/^\s+/, '').replace(/\s+$/, '');
-  }
-
-  function stripTags() {
-    return this.replace(/<\w+(\s+("[^"]*"|'[^']*'|[^>])+)?>|<\/\w+>/gi, '');
-  }
-
-  function stripScripts() {
-    return this.replace(new RegExp(Prototype.ScriptFragment, 'img'), '');
-  }
-
-  function extractScripts() {
-    var matchAll = new RegExp(Prototype.ScriptFragment, 'img'),
-        matchOne = new RegExp(Prototype.ScriptFragment, 'im');
-    return (this.match(matchAll) || []).map(function(scriptTag) {
-      return (scriptTag.match(matchOne) || ['', ''])[1];
-    });
-  }
-
-  function evalScripts() {
-    return this.extractScripts().map(function(script) { return eval(script) });
-  }
-
-  function escapeHTML() {
-    return this.replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
-  }
-
-  function unescapeHTML() {
-    return this.stripTags().replace(/&lt;/g,'<').replace(/&gt;/g,'>').replace(/&amp;/g,'&');
-  }
-
-
-  function toQueryParams(separator) {
-    var match = this.strip().match(/([^?#]*)(#.*)?$/);
-    if (!match) return { };
-
-    return match[1].split(separator || '&').inject({ }, function(hash, pair) {
-      if ((pair = pair.split('='))[0]) {
-        var key = decodeURIComponent(pair.shift()),
-            value = pair.length > 1 ? pair.join('=') : pair[0];
-
-        if (value != undefined) value = decodeURIComponent(value);
-
-        if (key in hash) {
-          if (!Object.isArray(hash[key])) hash[key] = [hash[key]];
-          hash[key].push(value);
-        }
-        else hash[key] = value;
-      }
-      return hash;
-    });
-  }
-
-  function toArray() {
-    return this.split('');
-  }
-
-  function succ() {
-    return this.slice(0, this.length - 1) +
-      String.fromCharCode(this.charCodeAt(this.length - 1) + 1);
-  }
-
-  function times(count) {
-    return count < 1 ? '' : new Array(count + 1).join(this);
-  }
-
-  function camelize() {
-    return this.replace(/-+(.)?/g, function(match, chr) {
-      return chr ? chr.toUpperCase() : '';
-    });
-  }
-
-  function capitalize() {
-    return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase();
-  }
-
-  function underscore() {
-    return this.replace(/::/g, '/')
-               .replace(/([A-Z]+)([A-Z][a-z])/g, '$1_$2')
-               .replace(/([a-z\d])([A-Z])/g, '$1_$2')
-               .replace(/-/g, '_')
-               .toLowerCase();
-  }
-
-  function dasherize() {
-    return this.replace(/_/g, '-');
-  }
-
-  function inspect(useDoubleQuotes) {
-    var escapedString = this.replace(/[\x00-\x1f\\]/g, function(character) {
-      if (character in String.specialChar) {
-        return String.specialChar[character];
-      }
-      return '\\u00' + character.charCodeAt().toPaddedString(2, 16);
-    });
-    if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"';
-    return "'" + escapedString.replace(/'/g, '\\\'') + "'";
-  }
-
-  function unfilterJSON(filter) {
-    return this.replace(filter || Prototype.JSONFilter, '$1');
-  }
-
-  function isJSON() {
-    var str = this;
-    if (str.blank()) return false;
-    str = str.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@');
-    str = str.replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']');
-    str = str.replace(/(?:^|:|,)(?:\s*\[)+/g, '');
-    return (/^[\],:{}\s]*$/).test(str);
-  }
-
-  function evalJSON(sanitize) {
-    var json = this.unfilterJSON(),
-        cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
-    if (cx.test(json)) {
-      json = json.replace(cx, function (a) {
-        return '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
-      });
-    }
-    try {
-      if (!sanitize || json.isJSON()) return eval('(' + json + ')');
-    } catch (e) { }
-    throw new SyntaxError('Badly formed JSON string: ' + this.inspect());
-  }
-
-  function parseJSON() {
-    var json = this.unfilterJSON();
-    return JSON.parse(json);
-  }
-
-  function include(pattern) {
-    return this.indexOf(pattern) > -1;
-  }
-
-  function startsWith(pattern) {
-    return this.lastIndexOf(pattern, 0) === 0;
-  }
-
-  function endsWith(pattern) {
-    var d = this.length - pattern.length;
-    return d >= 0 && this.indexOf(pattern, d) === d;
-  }
-
-  function empty() {
-    return this == '';
-  }
-
-  function blank() {
-    return /^\s*$/.test(this);
-  }
-
-  function interpolate(object, pattern) {
-    return new Template(this, pattern).evaluate(object);
-  }
-
-  return {
-    gsub:           gsub,
-    sub:            sub,
-    scan:           scan,
-    truncate:       truncate,
-    strip:          String.prototype.trim || strip,
-    stripTags:      stripTags,
-    stripScripts:   stripScripts,
-    extractScripts: extractScripts,
-    evalScripts:    evalScripts,
-    escapeHTML:     escapeHTML,
-    unescapeHTML:   unescapeHTML,
-    toQueryParams:  toQueryParams,
-    parseQuery:     toQueryParams,
-    toArray:        toArray,
-    succ:           succ,
-    times:          times,
-    camelize:       camelize,
-    capitalize:     capitalize,
-    underscore:     underscore,
-    dasherize:      dasherize,
-    inspect:        inspect,
-    unfilterJSON:   unfilterJSON,
-    isJSON:         isJSON,
-    evalJSON:       NATIVE_JSON_PARSE_SUPPORT ? parseJSON : evalJSON,
-    include:        include,
-    startsWith:     startsWith,
-    endsWith:       endsWith,
-    empty:          empty,
-    blank:          blank,
-    interpolate:    interpolate
-  };
-})());
-
-var Template = Class.create({
-  initialize: function(template, pattern) {
-    this.template = template.toString();
-    this.pattern = pattern || Template.Pattern;
-  },
-
-  evaluate: function(object) {
-    if (object && Object.isFunction(object.toTemplateReplacements))
-      object = object.toTemplateReplacements();
-
-    return this.template.gsub(this.pattern, function(match) {
-      if (object == null) return (match[1] + '');
-
-      var before = match[1] || '';
-      if (before == '\\') return match[2];
-
-      var ctx = object, expr = match[3],
-          pattern = /^([^.[]+|\[((?:.*?[^\\])?)\])(\.|\[|$)/;
-
-      match = pattern.exec(expr);
-      if (match == null) return before;
-
-      while (match != null) {
-        var comp = match[1].startsWith('[') ? match[2].replace(/\\\\]/g, ']') : match[1];
-        ctx = ctx[comp];
-        if (null == ctx || '' == match[3]) break;
-        expr = expr.substring('[' == match[3] ? match[1].length : match[0].length);
-        match = pattern.exec(expr);
-      }
-
-      return before + String.interpret(ctx);
-    });
-  }
-});
-Template.Pattern = /(^|.|\r|\n)(#\{(.*?)\})/;
-
-var $break = { };
-
-var Enumerable = (function() {
-  function each(iterator, context) {
-    var index = 0;
-    try {
-      this._each(function(value) {
-        iterator.call(context, value, index++);
-      });
-    } catch (e) {
-      if (e != $break) throw e;
-    }
-    return this;
-  }
-
-  function eachSlice(number, iterator, context) {
-    var index = -number, slices = [], array = this.toArray();
-    if (number < 1) return array;
-    while ((index += number) < array.length)
-      slices.push(array.slice(index, index+number));
-    return slices.collect(iterator, context);
-  }
-
-  function all(iterator, context) {
-    iterator = iterator || Prototype.K;
-    var result = true;
-    this.each(function(value, index) {
-      result = result && !!iterator.call(context, value, index);
-      if (!result) throw $break;
-    });
-    return result;
-  }
-
-  function any(iterator, context) {
-    iterator = iterator || Prototype.K;
-    var result = false;
-    this.each(function(value, index) {
-      if (result = !!iterator.call(context, value, index))
-        throw $break;
-    });
-    return result;
-  }
-
-  function collect(iterator, context) {
-    iterator = iterator || Prototype.K;
-    var results = [];
-    this.each(function(value, index) {
-      results.push(iterator.call(context, value, index));
-    });
-    return results;
-  }
-
-  function detect(iterator, context) {
-    var result;
-    this.each(function(value, index) {
-      if (iterator.call(context, value, index)) {
-        result = value;
-        throw $break;
-      }
-    });
-    return result;
-  }
-
-  function findAll(iterator, context) {
-    var results = [];
-    this.each(function(value, index) {
-      if (iterator.call(context, value, index))
-        results.push(value);
-    });
-    return results;
-  }
-
-  function grep(filter, iterator, context) {
-    iterator = iterator || Prototype.K;
-    var results = [];
-
-    if (Object.isString(filter))
-      filter = new RegExp(RegExp.escape(filter));
-
-    this.each(function(value, index) {
-      if (filter.match(value))
-        results.push(iterator.call(context, value, index));
-    });
-    return results;
-  }
-
-  function include(object) {
-    if (Object.isFunction(this.indexOf))
-      if (this.indexOf(object) != -1) return true;
-
-    var found = false;
-    this.each(function(value) {
-      if (value == object) {
-        found = true;
-        throw $break;
-      }
-    });
-    return found;
-  }
-
-  function inGroupsOf(number, fillWith) {
-    fillWith = Object.isUndefined(fillWith) ? null : fillWith;
-    return this.eachSlice(number, function(slice) {
-      while(slice.length < number) slice.push(fillWith);
-      return slice;
-    });
-  }
-
-  function inject(memo, iterator, context) {
-    this.each(function(value, index) {
-      memo = iterator.call(context, memo, value, index);
-    });
-    return memo;
-  }
-
-  function invoke(method) {
-    var args = $A(arguments).slice(1);
-    return this.map(function(value) {
-      return value[method].apply(value, args);
-    });
-  }
-
-  function max(iterator, context) {
-    iterator = iterator || Prototype.K;
-    var result;
-    this.each(function(value, index) {
-      value = iterator.call(context, value, index);
-      if (result == null || value >= result)
-        result = value;
-    });
-    return result;
-  }
-
-  function min(iterator, context) {
-    iterator = iterator || Prototype.K;
-    var result;
-    this.each(function(value, index) {
-      value = iterator.call(context, value, index);
-      if (result == null || value < result)
-        result = value;
-    });
-    return result;
-  }
-
-  function partition(iterator, context) {
-    iterator = iterator || Prototype.K;
-    var trues = [], falses = [];
-    this.each(function(value, index) {
-      (iterator.call(context, value, index) ?
-        trues : falses).push(value);
-    });
-    return [trues, falses];
-  }
-
-  function pluck(property) {
-    var results = [];
-    this.each(function(value) {
-      results.push(value[property]);
-    });
-    return results;
-  }
-
-  function reject(iterator, context) {
-    var results = [];
-    this.each(function(value, index) {
-      if (!iterator.call(context, value, index))
-        results.push(value);
-    });
-    return results;
-  }
-
-  function sortBy(iterator, context) {
-    return this.map(function(value, index) {
-      return {
-        value: value,
-        criteria: iterator.call(context, value, index)
-      };
-    }).sort(function(left, right) {
-      var a = left.criteria, b = right.criteria;
-      return a < b ? -1 : a > b ? 1 : 0;
-    }).pluck('value');
-  }
-
-  function toArray() {
-    return this.map();
-  }
-
-  function zip() {
-    var iterator = Prototype.K, args = $A(arguments);
-    if (Object.isFunction(args.last()))
-      iterator = args.pop();
-
-    var collections = [this].concat(args).map($A);
-    return this.map(function(value, index) {
-      return iterator(collections.pluck(index));
-    });
-  }
-
-  function size() {
-    return this.toArray().length;
-  }
-
-  function inspect() {
-    return '#<Enumerable:' + this.toArray().inspect() + '>';
-  }
-
-
-
-
-
-
-
-
-
-  return {
-    each:       each,
-    eachSlice:  eachSlice,
-    all:        all,
-    every:      all,
-    any:        any,
-    some:       any,
-    collect:    collect,
-    map:        collect,
-    detect:     detect,
-    findAll:    findAll,
-    select:     findAll,
-    filter:     findAll,
-    grep:       grep,
-    include:    include,
-    member:     include,
-    inGroupsOf: inGroupsOf,
-    inject:     inject,
-    invoke:     invoke,
-    max:        max,
-    min:        min,
-    partition:  partition,
-    pluck:      pluck,
-    reject:     reject,
-    sortBy:     sortBy,
-    toArray:    toArray,
-    entries:    toArray,
-    zip:        zip,
-    size:       size,
-    inspect:    inspect,
-    find:       detect
-  };
-})();
-
-function $A(iterable) {
-  if (!iterable) return [];
-  if ('toArray' in Object(iterable)) return iterable.toArray();
-  var length = iterable.length || 0, results = new Array(length);
-  while (length--) results[length] = iterable[length];
-  return results;
-}
-
-
-function $w(string) {
-  if (!Object.isString(string)) return [];
-  string = string.strip();
-  return string ? string.split(/\s+/) : [];
-}
-
-Array.from = $A;
-
-
-(function() {
-  var arrayProto = Array.prototype,
-      slice = arrayProto.slice,
-      _each = arrayProto.forEach; // use native browser JS 1.6 implementation if available
-
-  function each(iterator, context) {
-    for (var i = 0, length = this.length >>> 0; i < length; i++) {
-      if (i in this) iterator.call(context, this[i], i, this);
-    }
-  }
-  if (!_each) _each = each;
-
-  function clear() {
-    this.length = 0;
-    return this;
-  }
-
-  function first() {
-    return this[0];
-  }
-
-  function last() {
-    return this[this.length - 1];
-  }
-
-  function compact() {
-    return this.select(function(value) {
-      return value != null;
-    });
-  }
-
-  function flatten() {
-    return this.inject([], function(array, value) {
-      if (Object.isArray(value))
-        return array.concat(value.flatten());
-      array.push(value);
-      return array;
-    });
-  }
-
-  function without() {
-    var values = slice.call(arguments, 0);
-    return this.select(function(value) {
-      return !values.include(value);
-    });
-  }
-
-  function reverse(inline) {
-    return (inline === false ? this.toArray() : this)._reverse();
-  }
-
-  function uniq(sorted) {
-    return this.inject([], function(array, value, index) {
-      if (0 == index || (sorted ? array.last() != value : !array.include(value)))
-        array.push(value);
-      return array;
-    });
-  }
-
-  function intersect(array) {
-    return this.uniq().findAll(function(item) {
-      return array.detect(function(value) { return item === value });
-    });
-  }
-
-
-  function clone() {
-    return slice.call(this, 0);
-  }
-
-  function size() {
-    return this.length;
-  }
-
-  function inspect() {
-    return '[' + this.map(Object.inspect).join(', ') + ']';
-  }
-
-  function indexOf(item, i) {
-    i || (i = 0);
-    var length = this.length;
-    if (i < 0) i = length + i;
-    for (; i < length; i++)
-      if (this[i] === item) return i;
-    return -1;
-  }
-
-  function lastIndexOf(item, i) {
-    i = isNaN(i) ? this.length : (i < 0 ? this.length + i : i) + 1;
-    var n = this.slice(0, i).reverse().indexOf(item);
-    return (n < 0) ? n : i - n - 1;
-  }
-
-  function concat() {
-    var array = slice.call(this, 0), item;
-    for (var i = 0, length = arguments.length; i < length; i++) {
-      item = arguments[i];
-      if (Object.isArray(item) && !('callee' in item)) {
-        for (var j = 0, arrayLength = item.length; j < arrayLength; j++)
-          array.push(item[j]);
-      } else {
-        array.push(item);
-      }
-    }
-    return array;
-  }
-
-  Object.extend(arrayProto, Enumerable);
-
-  if (!arrayProto._reverse)
-    arrayProto._reverse = arrayProto.reverse;
-
-  Object.extend(arrayProto, {
-    _each:     _each,
-    clear:     clear,
-    first:     first,
-    last:      last,
-    compact:   compact,
-    flatten:   flatten,
-    without:   without,
-    reverse:   reverse,
-    uniq:      uniq,
-    intersect: intersect,
-    clone:     clone,
-    toArray:   clone,
-    size:      size,
-    inspect:   inspect
-  });
-
-  var CONCAT_ARGUMENTS_BUGGY = (function() {
-    return [].concat(arguments)[0][0] !== 1;
-  })(1,2)
-
-  if (CONCAT_ARGUMENTS_BUGGY) arrayProto.concat = concat;
-
-  if (!arrayProto.indexOf) arrayProto.indexOf = indexOf;
-  if (!arrayProto.lastIndexOf) arrayProto.lastIndexOf = lastIndexOf;
-})();
-function $H(object) {
-  return new Hash(object);
-};
-
-var Hash = Class.create(Enumerable, (function() {
-  function initialize(object) {
-    this._object = Object.isHash(object) ? object.toObject() : Object.clone(object);
-  }
-
-
-  function _each(iterator) {
-    for (var key in this._object) {
-      var value = this._object[key], pair = [key, value];
-      pair.key = key;
-      pair.value = value;
-      iterator(pair);
-    }
-  }
-
-  function set(key, value) {
-    return this._object[key] = value;
-  }
-
-  function get(key) {
-    if (this._object[key] !== Object.prototype[key])
-      return this._object[key];
-  }
-
-  function unset(key) {
-    var value = this._object[key];
-    delete this._object[key];
-    return value;
-  }
-
-  function toObject() {
-    return Object.clone(this._object);
-  }
-
-
-
-  function keys() {
-    return this.pluck('key');
-  }
-
-  function values() {
-    return this.pluck('value');
-  }
-
-  function index(value) {
-    var match = this.detect(function(pair) {
-      return pair.value === value;
-    });
-    return match && match.key;
-  }
-
-  function merge(object) {
-    return this.clone().update(object);
-  }
-
-  function update(object) {
-    return new Hash(object).inject(this, function(result, pair) {
-      result.set(pair.key, pair.value);
-      return result;
-    });
-  }
-
-  function toQueryPair(key, value) {
-    if (Object.isUndefined(value)) return key;
-    return key + '=' + encodeURIComponent(String.interpret(value));
-  }
-
-  function toQueryString() {
-    return this.inject([], function(results, pair) {
-      var key = encodeURIComponent(pair.key), values = pair.value;
-
-      if (values && typeof values == 'object') {
-        if (Object.isArray(values)) {
-          var queryValues = [];
-          for (var i = 0, len = values.length, value; i < len; i++) {
-            value = values[i];
-            queryValues.push(toQueryPair(key, value));
-          }
-          return results.concat(queryValues);
-        }
-      } else results.push(toQueryPair(key, values));
-      return results;
-    }).join('&');
-  }
-
-  function inspect() {
-    return '#<Hash:{' + this.map(function(pair) {
-      return pair.map(Object.inspect).join(': ');
-    }).join(', ') + '}>';
-  }
-
-  function clone() {
-    return new Hash(this);
-  }
-
-  return {
-    initialize:             initialize,
-    _each:                  _each,
-    set:                    set,
-    get:                    get,
-    unset:                  unset,
-    toObject:               toObject,
-    toTemplateReplacements: toObject,
-    keys:                   keys,
-    values:                 values,
-    index:                  index,
-    merge:                  merge,
-    update:                 update,
-    toQueryString:          toQueryString,
-    inspect:                inspect,
-    toJSON:                 toObject,
-    clone:                  clone
-  };
-})());
-
-Hash.from = $H;
-Object.extend(Number.prototype, (function() {
-  function toColorPart() {
-    return this.toPaddedString(2, 16);
-  }
-
-  function succ() {
-    return this + 1;
-  }
-
-  function times(iterator, context) {
-    $R(0, this, true).each(iterator, context);
-    return this;
-  }
-
-  function toPaddedString(length, radix) {
-    var string = this.toString(radix || 10);
-    return '0'.times(length - string.length) + string;
-  }
-
-  function abs() {
-    return Math.abs(this);
-  }
-
-  function round() {
-    return Math.round(this);
-  }
-
-  function ceil() {
-    return Math.ceil(this);
-  }
-
-  function floor() {
-    return Math.floor(this);
-  }
-
-  return {
-    toColorPart:    toColorPart,
-    succ:           succ,
-    times:          times,
-    toPaddedString: toPaddedString,
-    abs:            abs,
-    round:          round,
-    ceil:           ceil,
-    floor:          floor
-  };
-})());
-
-function $R(start, end, exclusive) {
-  return new ObjectRange(start, end, exclusive);
-}
-
-var ObjectRange = Class.create(Enumerable, (function() {
-  function initialize(start, end, exclusive) {
-    this.start = start;
-    this.end = end;
-    this.exclusive = exclusive;
-  }
-
-  function _each(iterator) {
-    var value = this.start;
-    while (this.include(value)) {
-      iterator(value);
-      value = value.succ();
-    }
-  }
-
-  function include(value) {
-    if (value < this.start)
-      return false;
-    if (this.exclusive)
-      return value < this.end;
-    return value <= this.end;
-  }
-
-  return {
-    initialize: initialize,
-    _each:      _each,
-    include:    include
-  };
-})());
-
-
-
-var Ajax = {
-  getTransport: function() {
-    return Try.these(
-      function() {return new XMLHttpRequest()},
-      function() {return new ActiveXObject('Msxml2.XMLHTTP')},
-      function() {return new ActiveXObject('Microsoft.XMLHTTP')}
-    ) || false;
-  },
-
-  activeRequestCount: 0
-};
-
-Ajax.Responders = {
-  responders: [],
-
-  _each: function(iterator) {
-    this.responders._each(iterator);
-  },
-
-  register: function(responder) {
-    if (!this.include(responder))
-      this.responders.push(responder);
-  },
-
-  unregister: function(responder) {
-    this.responders = this.responders.without(responder);
-  },
-
-  dispatch: function(callback, request, transport, json) {
-    this.each(function(responder) {
-      if (Object.isFunction(responder[callback])) {
-        try {
-          responder[callback].apply(responder, [request, transport, json]);
-        } catch (e) { }
-      }
-    });
-  }
-};
-
-Object.extend(Ajax.Responders, Enumerable);
-
-Ajax.Responders.register({
-  onCreate:   function() { Ajax.activeRequestCount++ },
-  onComplete: function() { Ajax.activeRequestCount-- }
-});
-Ajax.Base = Class.create({
-  initialize: function(options) {
-    this.options = {
-      method:       'post',
-      asynchronous: true,
-      contentType:  'application/x-www-form-urlencoded',
-      encoding:     'UTF-8',
-      parameters:   '',
-      evalJSON:     true,
-      evalJS:       true
-    };
-    Object.extend(this.options, options || { });
-
-    this.options.method = this.options.method.toLowerCase();
-
-    if (Object.isHash(this.options.parameters))
-      this.options.parameters = this.options.parameters.toObject();
-  }
-});
-Ajax.Request = Class.create(Ajax.Base, {
-  _complete: false,
-
-  initialize: function($super, url, options) {
-    $super(options);
-    this.transport = Ajax.getTransport();
-    this.request(url);
-  },
-
-  request: function(url) {
-    this.url = url;
-    this.method = this.options.method;
-    var params = Object.isString(this.options.parameters) ?
-          this.options.parameters :
-          Object.toQueryString(this.options.parameters);
-
-    if (!['get', 'post'].include(this.method)) {
-      params += (params ? '&' : '') + "_method=" + this.method;
-      this.method = 'post';
-    }
-
-    if (params && this.method === 'get') {
-      this.url += (this.url.include('?') ? '&' : '?') + params;
-    }
-
-    this.parameters = params.toQueryParams();
-
-    try {
-      var response = new Ajax.Response(this);
-      if (this.options.onCreate) this.options.onCreate(response);
-      Ajax.Responders.dispatch('onCreate', this, response);
-
-      this.transport.open(this.method.toUpperCase(), this.url,
-        this.options.asynchronous);
-
-      if (this.options.asynchronous) this.respondToReadyState.bind(this).defer(1);
-
-      this.transport.onreadystatechange = this.onStateChange.bind(this);
-      this.setRequestHeaders();
-
-      this.body = this.method == 'post' ? (this.options.postBody || params) : null;
-      this.transport.send(this.body);
-
-      /* Force Firefox to handle ready state 4 for synchronous requests */
-      if (!this.options.asynchronous && this.transport.overrideMimeType)
-        this.onStateChange();
-
-    }
-    catch (e) {
-      this.dispatchException(e);
-    }
-  },
-
-  onStateChange: function() {
-    var readyState = this.transport.readyState;
-    if (readyState > 1 && !((readyState == 4) && this._complete))
-      this.respondToReadyState(this.transport.readyState);
-  },
-
-  setRequestHeaders: function() {
-    var headers = {
-      'X-Requested-With': 'XMLHttpRequest',
-      'X-Prototype-Version': Prototype.Version,
-      'Accept': 'text/javascript, text/html, application/xml, text/xml, */*'
-    };
-
-    if (this.method == 'post') {
-      headers['Content-type'] = this.options.contentType +
-        (this.options.encoding ? '; charset=' + this.options.encoding : '');
-
-      /* Force "Connection: close" for older Mozilla browsers to work
-       * around a bug where XMLHttpRequest sends an incorrect
-       * Content-length header. See Mozilla Bugzilla #246651.
-       */
-      if (this.transport.overrideMimeType &&
-          (navigator.userAgent.match(/Gecko\/(\d{4})/) || [0,2005])[1] < 2005)
-            headers['Connection'] = 'close';
-    }
-
-    if (typeof this.options.requestHeaders == 'object') {
-      var extras = this.options.requestHeaders;
-
-      if (Object.isFunction(extras.push))
-        for (var i = 0, length = extras.length; i < length; i += 2)
-          headers[extras[i]] = extras[i+1];
-      else
-        $H(extras).each(function(pair) { headers[pair.key] = pair.value });
-    }
-
-    for (var name in headers)
-      this.transport.setRequestHeader(name, headers[name]);
-  },
-
-  success: function() {
-    var status = this.getStatus();
-    return !status || (status >= 200 && status < 300) || status == 304;
-  },
-
-  getStatus: function() {
-    try {
-      if (this.transport.status === 1223) return 204;
-      return this.transport.status || 0;
-    } catch (e) { return 0 }
-  },
-
-  respondToReadyState: function(readyState) {
-    var state = Ajax.Request.Events[readyState], response = new Ajax.Response(this);
-
-    if (state == 'Complete') {
-      try {
-        this._complete = true;
-        (this.options['on' + response.status]
-         || this.options['on' + (this.success() ? 'Success' : 'Failure')]
-         || Prototype.emptyFunction)(response, response.headerJSON);
-      } catch (e) {
-        this.dispatchException(e);
-      }
-
-      var contentType = response.getHeader('Content-type');
-      if (this.options.evalJS == 'force'
-          || (this.options.evalJS && this.isSameOrigin() && contentType
-          && contentType.match(/^\s*(text|application)\/(x-)?(java|ecma)script(;.*)?\s*$/i)))
-        this.evalResponse();
-    }
-
-    try {
-      (this.options['on' + state] || Prototype.emptyFunction)(response, response.headerJSON);
-      Ajax.Responders.dispatch('on' + state, this, response, response.headerJSON);
-    } catch (e) {
-      this.dispatchException(e);
-    }
-
-    if (state == 'Complete') {
-      this.transport.onreadystatechange = Prototype.emptyFunction;
-    }
-  },
-
-  isSameOrigin: function() {
-    var m = this.url.match(/^\s*https?:\/\/[^\/]*/);
-    return !m || (m[0] == '#{protocol}//#{domain}#{port}'.interpolate({
-      protocol: location.protocol,
-      domain: document.domain,
-      port: location.port ? ':' + location.port : ''
-    }));
-  },
-
-  getHeader: function(name) {
-    try {
-      return this.transport.getResponseHeader(name) || null;
-    } catch (e) { return null; }
-  },
-
-  evalResponse: function() {
-    try {
-      return eval((this.transport.responseText || '').unfilterJSON());
-    } catch (e) {
-      this.dispatchException(e);
-    }
-  },
-
-  dispatchException: function(exception) {
-    (this.options.onException || Prototype.emptyFunction)(this, exception);
-    Ajax.Responders.dispatch('onException', this, exception);
-  }
-});
-
-Ajax.Request.Events =
-  ['Uninitialized', 'Loading', 'Loaded', 'Interactive', 'Complete'];
-
-
-
-
-
-
-
-
-Ajax.Response = Class.create({
-  initialize: function(request){
-    this.request = request;
-    var transport  = this.transport  = request.transport,
-        readyState = this.readyState = transport.readyState;
-
-    if ((readyState > 2 && !Prototype.Browser.IE) || readyState == 4) {
-      this.status       = this.getStatus();
-      this.statusText   = this.getStatusText();
-      this.responseText = String.interpret(transport.responseText);
-      this.headerJSON   = this._getHeaderJSON();
-    }
-
-    if (readyState == 4) {
-      var xml = transport.responseXML;
-      this.responseXML  = Object.isUndefined(xml) ? null : xml;
-      this.responseJSON = this._getResponseJSON();
-    }
-  },
-
-  status:      0,
-
-  statusText: '',
-
-  getStatus: Ajax.Request.prototype.getStatus,
-
-  getStatusText: function() {
-    try {
-      return this.transport.statusText || '';
-    } catch (e) { return '' }
-  },
-
-  getHeader: Ajax.Request.prototype.getHeader,
-
-  getAllHeaders: function() {
-    try {
-      return this.getAllResponseHeaders();
-    } catch (e) { return null }
-  },
-
-  getResponseHeader: function(name) {
-    return this.transport.getResponseHeader(name);
-  },
-
-  getAllResponseHeaders: function() {
-    return this.transport.getAllResponseHeaders();
-  },
-
-  _getHeaderJSON: function() {
-    var json = this.getHeader('X-JSON');
-    if (!json) return null;
-    json = decodeURIComponent(escape(json));
-    try {
-      return json.evalJSON(this.request.options.sanitizeJSON ||
-        !this.request.isSameOrigin());
-    } catch (e) {
-      this.request.dispatchException(e);
-    }
-  },
-
-  _getResponseJSON: function() {
-    var options = this.request.options;
-    if (!options.evalJSON || (options.evalJSON != 'force' &&
-      !(this.getHeader('Content-type') || '').include('application/json')) ||
-        this.responseText.blank())
-          return null;
-    try {
-      return this.responseText.evalJSON(options.sanitizeJSON ||
-        !this.request.isSameOrigin());
-    } catch (e) {
-      this.request.dispatchException(e);
-    }
-  }
-});
-
-Ajax.Updater = Class.create(Ajax.Request, {
-  initialize: function($super, container, url, options) {
-    this.container = {
-      success: (container.success || container),
-      failure: (container.failure || (container.success ? null : container))
-    };
-
-    options = Object.clone(options);
-    var onComplete = options.onComplete;
-    options.onComplete = (function(response, json) {
-      this.updateContent(response.responseText);
-      if (Object.isFunction(onComplete)) onComplete(response, json);
-    }).bind(this);
-
-    $super(url, options);
-  },
-
-  updateContent: function(responseText) {
-    var receiver = this.container[this.success() ? 'success' : 'failure'],
-        options = this.options;
-
-    if (!options.evalScripts) responseText = responseText.stripScripts();
-
-    if (receiver = $(receiver)) {
-      if (options.insertion) {
-        if (Object.isString(options.insertion)) {
-          var insertion = { }; insertion[options.insertion] = responseText;
-          receiver.insert(insertion);
-        }
-        else options.insertion(receiver, responseText);
-      }
-      else receiver.update(responseText);
-    }
-  }
-});
-
-Ajax.PeriodicalUpdater = Class.create(Ajax.Base, {
-  initialize: function($super, container, url, options) {
-    $super(options);
-    this.onComplete = this.options.onComplete;
-
-    this.frequency = (this.options.frequency || 2);
-    this.decay = (this.options.decay || 1);
-
-    this.updater = { };
-    this.container = container;
-    this.url = url;
-
-    this.start();
-  },
-
-  start: function() {
-    this.options.onComplete = this.updateComplete.bind(this);
-    this.onTimerEvent();
-  },
-
-  stop: function() {
-    this.updater.options.onComplete = undefined;
-    clearTimeout(this.timer);
-    (this.onComplete || Prototype.emptyFunction).apply(this, arguments);
-  },
-
-  updateComplete: function(response) {
-    if (this.options.decay) {
-      this.decay = (response.responseText == this.lastText ?
-        this.decay * this.options.decay : 1);
-
-      this.lastText = response.responseText;
-    }
-    this.timer = this.onTimerEvent.bind(this).delay(this.decay * this.frequency);
-  },
-
-  onTimerEvent: function() {
-    this.updater = new Ajax.Updater(this.container, this.url, this.options);
-  }
-});
-
-
-function $(element) {
-  if (arguments.length > 1) {
-    for (var i = 0, elements = [], length = arguments.length; i < length; i++)
-      elements.push($(arguments[i]));
-    return elements;
-  }
-  if (Object.isString(element))
-    element = document.getElementById(element);
-  return Element.extend(element);
-}
-
-if (Prototype.BrowserFeatures.XPath) {
-  document._getElementsByXPath = function(expression, parentElement) {
-    var results = [];
-    var query = document.evaluate(expression, $(parentElement) || document,
-      null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
-    for (var i = 0, length = query.snapshotLength; i < length; i++)
-      results.push(Element.extend(query.snapshotItem(i)));
-    return results;
-  };
-}
-
-/*--------------------------------------------------------------------------*/
-
-if (!Node) var Node = { };
-
-if (!Node.ELEMENT_NODE) {
-  Object.extend(Node, {
-    ELEMENT_NODE: 1,
-    ATTRIBUTE_NODE: 2,
-    TEXT_NODE: 3,
-    CDATA_SECTION_NODE: 4,
-    ENTITY_REFERENCE_NODE: 5,
-    ENTITY_NODE: 6,
-    PROCESSING_INSTRUCTION_NODE: 7,
-    COMMENT_NODE: 8,
-    DOCUMENT_NODE: 9,
-    DOCUMENT_TYPE_NODE: 10,
-    DOCUMENT_FRAGMENT_NODE: 11,
-    NOTATION_NODE: 12
-  });
-}
-
-
-
-(function(global) {
-  function shouldUseCache(tagName, attributes) {
-    if (tagName === 'select') return false;
-    if ('type' in attributes) return false;
-    return true;
-  }
-
-  var HAS_EXTENDED_CREATE_ELEMENT_SYNTAX = (function(){
-    try {
-      var el = document.createElement('<input name="x">');
-      return el.tagName.toLowerCase() === 'input' && el.name === 'x';
-    }
-    catch(err) {
-      return false;
-    }
-  })();
-
-  var element = global.Element;
-
-  global.Element = function(tagName, attributes) {
-    attributes = attributes || { };
-    tagName = tagName.toLowerCase();
-    var cache = Element.cache;
-
-    if (HAS_EXTENDED_CREATE_ELEMENT_SYNTAX && attributes.name) {
-      tagName = '<' + tagName + ' name="' + attributes.name + '">';
-      delete attributes.name;
-      return Element.writeAttribute(document.createElement(tagName), attributes);
-    }
-
-    if (!cache[tagName]) cache[tagName] = Element.extend(document.createElement(tagName));
-
-    var node = shouldUseCache(tagName, attributes) ?
-     cache[tagName].cloneNode(false) : document.createElement(tagName);
-
-    return Element.writeAttribute(node, attributes);
-  };
-
-  Object.extend(global.Element, element || { });
-  if (element) global.Element.prototype = element.prototype;
-
-})(this);
-
-Element.idCounter = 1;
-Element.cache = { };
-
-Element._purgeElement = function(element) {
-  var uid = element._prototypeUID;
-  if (uid) {
-    Element.stopObserving(element);
-    element._prototypeUID = void 0;
-    delete Element.Storage[uid];
-  }
-}
-
-Element.Methods = {
-  visible: function(element) {
-    return $(element).style.display != 'none';
-  },
-
-  toggle: function(element) {
-    element = $(element);
-    Element[Element.visible(element) ? 'hide' : 'show'](element);
-    return element;
-  },
-
-  hide: function(element) {
-    element = $(element);
-    element.style.display = 'none';
-    return element;
-  },
-
-  show: function(element) {
-    element = $(element);
-    element.style.display = '';
-    return element;
-  },
-
-  remove: function(element) {
-    element = $(element);
-    element.parentNode.removeChild(element);
-    return element;
-  },
-
-  update: (function(){
-
-    var SELECT_ELEMENT_INNERHTML_BUGGY = (function(){
-      var el = document.createElement("select"),
-          isBuggy = true;
-      el.innerHTML = "<option value=\"test\">test</option>";
-      if (el.options && el.options[0]) {
-        isBuggy = el.options[0].nodeName.toUpperCase() !== "OPTION";
-      }
-      el = null;
-      return isBuggy;
-    })();
-
-    var TABLE_ELEMENT_INNERHTML_BUGGY = (function(){
-      try {
-        var el = document.createElement("table");
-        if (el && el.tBodies) {
-          el.innerHTML = "<tbody><tr><td>test</td></tr></tbody>";
-          var isBuggy = typeof el.tBodies[0] == "undefined";
-          el = null;
-          return isBuggy;
-        }
-      } catch (e) {
-        return true;
-      }
-    })();
-
-    var LINK_ELEMENT_INNERHTML_BUGGY = (function() {
-      try {
-        var el = document.createElement('div');
-        el.innerHTML = "<link>";
-        var isBuggy = (el.childNodes.length === 0);
-        el = null;
-        return isBuggy;
-      } catch(e) {
-        return true;
-      }
-    })();
-
-    var ANY_INNERHTML_BUGGY = SELECT_ELEMENT_INNERHTML_BUGGY ||
-     TABLE_ELEMENT_INNERHTML_BUGGY || LINK_ELEMENT_INNERHTML_BUGGY;
-
-    var SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING = (function () {
-      var s = document.createElement("script"),
-          isBuggy = false;
-      try {
-        s.appendChild(document.createTextNode(""));
-        isBuggy = !s.firstChild ||
-          s.firstChild && s.firstChild.nodeType !== 3;
-      } catch (e) {
-        isBuggy = true;
-      }
-      s = null;
-      return isBuggy;
-    })();
-
-
-    function update(element, content) {
-      element = $(element);
-      var purgeElement = Element._purgeElement;
-
-      var descendants = element.getElementsByTagName('*'),
-       i = descendants.length;
-      while (i--) purgeElement(descendants[i]);
-
-      if (content && content.toElement)
-        content = content.toElement();
-
-      if (Object.isElement(content))
-        return element.update().insert(content);
-
-      content = Object.toHTML(content);
-
-      var tagName = element.tagName.toUpperCase();
-
-      if (tagName === 'SCRIPT' && SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING) {
-        element.text = content;
-        return element;
-      }
-
-      if (ANY_INNERHTML_BUGGY) {
-        if (tagName in Element._insertionTranslations.tags) {
-          while (element.firstChild) {
-            element.removeChild(element.firstChild);
-          }
-          Element._getContentFromAnonymousElement(tagName, content.stripScripts())
-            .each(function(node) {
-              element.appendChild(node)
-            });
-        } else if (LINK_ELEMENT_INNERHTML_BUGGY && Object.isString(content) && content.indexOf('<link') > -1) {
-          while (element.firstChild) {
-            element.removeChild(element.firstChild);
-          }
-          var nodes = Element._getContentFromAnonymousElement(tagName, content.stripScripts(), true);
-          nodes.each(function(node) { element.appendChild(node) });
-        }
-        else {
-          element.innerHTML = content.stripScripts();
-        }
-      }
-      else {
-        element.innerHTML = content.stripScripts();
-      }
-
-      content.evalScripts.bind(content).defer();
-      return element;
-    }
-
-    return update;
-  })(),
-
-  replace: function(element, content) {
-    element = $(element);
-    if (content && content.toElement) content = content.toElement();
-    else if (!Object.isElement(content)) {
-      content = Object.toHTML(content);
-      var range = element.ownerDocument.createRange();
-      range.selectNode(element);
-      content.evalScripts.bind(content).defer();
-      content = range.createContextualFragment(content.stripScripts());
-    }
-    element.parentNode.replaceChild(content, element);
-    return element;
-  },
-
-  insert: function(element, insertions) {
-    element = $(element);
-
-    if (Object.isString(insertions) || Object.isNumber(insertions) ||
-        Object.isElement(insertions) || (insertions && (insertions.toElement || insertions.toHTML)))
-          insertions = {bottom:insertions};
-
-    var content, insert, tagName, childNodes;
-
-    for (var position in insertions) {
-      content  = insertions[position];
-      position = position.toLowerCase();
-      insert = Element._insertionTranslations[position];
-
-      if (content && content.toElement) content = content.toElement();
-      if (Object.isElement(content)) {
-        insert(element, content);
-        continue;
-      }
-
-      content = Object.toHTML(content);
-
-      tagName = ((position == 'before' || position == 'after')
-        ? element.parentNode : element).tagName.toUpperCase();
-
-      childNodes = Element._getContentFromAnonymousElement(tagName, content.stripScripts());
-
-      if (position == 'top' || position == 'after') childNodes.reverse();
-      childNodes.each(insert.curry(element));
-
-      content.evalScripts.bind(content).defer();
-    }
-
-    return element;
-  },
-
-  wrap: function(element, wrapper, attributes) {
-    element = $(element);
-    if (Object.isElement(wrapper))
-      $(wrapper).writeAttribute(attributes || { });
-    else if (Object.isString(wrapper)) wrapper = new Element(wrapper, attributes);
-    else wrapper = new Element('div', wrapper);
-    if (element.parentNode)
-      element.parentNode.replaceChild(wrapper, element);
-    wrapper.appendChild(element);
-    return wrapper;
-  },
-
-  inspect: function(element) {
-    element = $(element);
-    var result = '<' + element.tagName.toLowerCase();
-    $H({'id': 'id', 'className': 'class'}).each(function(pair) {
-      var property = pair.first(),
-          attribute = pair.last(),
-          value = (element[property] || '').toString();
-      if (value) result += ' ' + attribute + '=' + value.inspect(true);
-    });
-    return result + '>';
-  },
-
-  recursivelyCollect: function(element, property, maximumLength) {
-    element = $(element);
-    maximumLength = maximumLength || -1;
-    var elements = [];
-
-    while (element = element[property]) {
-      if (element.nodeType == 1)
-        elements.push(Element.extend(element));
-      if (elements.length == maximumLength)
-        break;
-    }
-
-    return elements;
-  },
-
-  ancestors: function(element) {
-    return Element.recursivelyCollect(element, 'parentNode');
-  },
-
-  descendants: function(element) {
-    return Element.select(element, "*");
-  },
-
-  firstDescendant: function(element) {
-    element = $(element).firstChild;
-    while (element && element.nodeType != 1) element = element.nextSibling;
-    return $(element);
-  },
-
-  immediateDescendants: function(element) {
-    var results = [], child = $(element).firstChild;
-    while (child) {
-      if (child.nodeType === 1) {
-        results.push(Element.extend(child));
-      }
-      child = child.nextSibling;
-    }
-    return results;
-  },
-
-  previousSiblings: function(element, maximumLength) {
-    return Element.recursivelyCollect(element, 'previousSibling');
-  },
-
-  nextSiblings: function(element) {
-    return Element.recursivelyCollect(element, 'nextSibling');
-  },
-
-  siblings: function(element) {
-    element = $(element);
-    return Element.previousSiblings(element).reverse()
-      .concat(Element.nextSiblings(element));
-  },
-
-  match: function(element, selector) {
-    element = $(element);
-    if (Object.isString(selector))
-      return Prototype.Selector.match(element, selector);
-    return selector.match(element);
-  },
-
-  up: function(element, expression, index) {
-    element = $(element);
-    if (arguments.length == 1) return $(element.parentNode);
-    var ancestors = Element.ancestors(element);
-    return Object.isNumber(expression) ? ancestors[expression] :
-      Prototype.Selector.find(ancestors, expression, index);
-  },
-
-  down: function(element, expression, index) {
-    element = $(element);
-    if (arguments.length == 1) return Element.firstDescendant(element);
-    return Object.isNumber(expression) ? Element.descendants(element)[expression] :
-      Element.select(element, expression)[index || 0];
-  },
-
-  previous: function(element, expression, index) {
-    element = $(element);
-    if (Object.isNumber(expression)) index = expression, expression = false;
-    if (!Object.isNumber(index)) index = 0;
-
-    if (expression) {
-      return Prototype.Selector.find(element.previousSiblings(), expression, index);
-    } else {
-      return element.recursivelyCollect("previousSibling", index + 1)[index];
-    }
-  },
-
-  next: function(element, expression, index) {
-    element = $(element);
-    if (Object.isNumber(expression)) index = expression, expression = false;
-    if (!Object.isNumber(index)) index = 0;
-
-    if (expression) {
-      return Prototype.Selector.find(element.nextSiblings(), expression, index);
-    } else {
-      var maximumLength = Object.isNumber(index) ? index + 1 : 1;
-      return element.recursivelyCollect("nextSibling", index + 1)[index];
-    }
-  },
-
-
-  select: function(element) {
-    element = $(element);
-    var expressions = Array.prototype.slice.call(arguments, 1).join(', ');
-    return Prototype.Selector.select(expressions, element);
-  },
-
-  adjacent: function(element) {
-    element = $(element);
-    var expressions = Array.prototype.slice.call(arguments, 1).join(', ');
-    return Prototype.Selector.select(expressions, element.parentNode).without(element);
-  },
-
-  identify: function(element) {
-    element = $(element);
-    var id = Element.readAttribute(element, 'id');
-    if (id) return id;
-    do { id = 'anonymous_element_' + Element.idCounter++ } while ($(id));
-    Element.writeAttribute(element, 'id', id);
-    return id;
-  },
-
-  readAttribute: function(element, name) {
-    element = $(element);
-    if (Prototype.Browser.IE) {
-      var t = Element._attributeTranslations.read;
-      if (t.values[name]) return t.values[name](element, name);
-      if (t.names[name]) name = t.names[name];
-      if (name.include(':')) {
-        return (!element.attributes || !element.attributes[name]) ? null :
-         element.attributes[name].value;
-      }
-    }
-    return element.getAttribute(name);
-  },
-
-  writeAttribute: function(element, name, value) {
-    element = $(element);
-    var attributes = { }, t = Element._attributeTranslations.write;
-
-    if (typeof name == 'object') attributes = name;
-    else attributes[name] = Object.isUndefined(value) ? true : value;
-
-    for (var attr in attributes) {
-      name = t.names[attr] || attr;
-      value = attributes[attr];
-      if (t.values[attr]) name = t.values[attr](element, value);
-      if (value === false || value === null)
-        element.removeAttribute(name);
-      else if (value === true)
-        element.setAttribute(name, name);
-      else element.setAttribute(name, value);
-    }
-    return element;
-  },
-
-  getHeight: function(element) {
-    return Element.getDimensions(element).height;
-  },
-
-  getWidth: function(element) {
-    return Element.getDimensions(element).width;
-  },
-
-  classNames: function(element) {
-    return new Element.ClassNames(element);
-  },
-
-  hasClassName: function(element, className) {
-    if (!(element = $(element))) return;
-    var elementClassName = element.className;
-    return (elementClassName.length > 0 && (elementClassName == className ||
-      new RegExp("(^|\\s)" + className + "(\\s|$)").test(elementClassName)));
-  },
-
-  addClassName: function(element, className) {
-    if (!(element = $(element))) return;
-    if (!Element.hasClassName(element, className))
-      element.className += (element.className ? ' ' : '') + className;
-    return element;
-  },
-
-  removeClassName: function(element, className) {
-    if (!(element = $(element))) return;
-    element.className = element.className.replace(
-      new RegExp("(^|\\s+)" + className + "(\\s+|$)"), ' ').strip();
-    return element;
-  },
-
-  toggleClassName: function(element, className) {
-    if (!(element = $(element))) return;
-    return Element[Element.hasClassName(element, className) ?
-      'removeClassName' : 'addClassName'](element, className);
-  },
-
-  cleanWhitespace: function(element) {
-    element = $(element);
-    var node = element.firstChild;
-    while (node) {
-      var nextNode = node.nextSibling;
-      if (node.nodeType == 3 && !/\S/.test(node.nodeValue))
-        element.removeChild(node);
-      node = nextNode;
-    }
-    return element;
-  },
-
-  empty: function(element) {
-    return $(element).innerHTML.blank();
-  },
-
-  descendantOf: function(element, ancestor) {
-    element = $(element), ancestor = $(ancestor);
-
-    if (element.compareDocumentPosition)
-      return (element.compareDocumentPosition(ancestor) & 8) === 8;
-
-    if (ancestor.contains)
-      return ancestor.contains(element) && ancestor !== element;
-
-    while (element = element.parentNode)
-      if (element == ancestor) return true;
-
-    return false;
-  },
-
-  scrollTo: function(element) {
-    element = $(element);
-    var pos = Element.cumulativeOffset(element);
-    window.scrollTo(pos[0], pos[1]);
-    return element;
-  },
-
-  getStyle: function(element, style) {
-    element = $(element);
-    style = style == 'float' ? 'cssFloat' : style.camelize();
-    var value = element.style[style];
-    if (!value || value == 'auto') {
-      var css = document.defaultView.getComputedStyle(element, null);
-      value = css ? css[style] : null;
-    }
-    if (style == 'opacity') return value ? parseFloat(value) : 1.0;
-    return value == 'auto' ? null : value;
-  },
-
-  getOpacity: function(element) {
-    return $(element).getStyle('opacity');
-  },
-
-  setStyle: function(element, styles) {
-    element = $(element);
-    var elementStyle = element.style, match;
-    if (Object.isString(styles)) {
-      element.style.cssText += ';' + styles;
-      return styles.include('opacity') ?
-        element.setOpacity(styles.match(/opacity:\s*(\d?\.?\d*)/)[1]) : element;
-    }
-    for (var property in styles)
-      if (property == 'opacity') element.setOpacity(styles[property]);
-      else
-        elementStyle[(property == 'float' || property == 'cssFloat') ?
-          (Object.isUndefined(elementStyle.styleFloat) ? 'cssFloat' : 'styleFloat') :
-            property] = styles[property];
-
-    return element;
-  },
-
-  setOpacity: function(element, value) {
-    element = $(element);
-    element.style.opacity = (value == 1 || value === '') ? '' :
-      (value < 0.00001) ? 0 : value;
-    return element;
-  },
-
-  makePositioned: function(element) {
-    element = $(element);
-    var pos = Element.getStyle(element, 'position');
-    if (pos == 'static' || !pos) {
-      element._madePositioned = true;
-      element.style.position = 'relative';
-      if (Prototype.Browser.Opera) {
-        element.style.top = 0;
-        element.style.left = 0;
-      }
-    }
-    return element;
-  },
-
-  undoPositioned: function(element) {
-    element = $(element);
-    if (element._madePositioned) {
-      element._madePositioned = undefined;
-      element.style.position =
-        element.style.top =
-        element.style.left =
-        element.style.bottom =
-        element.style.right = '';
-    }
-    return element;
-  },
-
-  makeClipping: function(element) {
-    element = $(element);
-    if (element._overflow) return element;
-    element._overflow = Element.getStyle(element, 'overflow') || 'auto';
-    if (element._overflow !== 'hidden')
-      element.style.overflow = 'hidden';
-    return element;
-  },
-
-  undoClipping: function(element) {
-    element = $(element);
-    if (!element._overflow) return element;
-    element.style.overflow = element._overflow == 'auto' ? '' : element._overflow;
-    element._overflow = null;
-    return element;
-  },
-
-  clonePosition: function(element, source) {
-    var options = Object.extend({
-      setLeft:    true,
-      setTop:     true,
-      setWidth:   true,
-      setHeight:  true,
-      offsetTop:  0,
-      offsetLeft: 0
-    }, arguments[2] || { });
-
-    source = $(source);
-    var p = Element.viewportOffset(source), delta = [0, 0], parent = null;
-
-    element = $(element);
-
-    if (Element.getStyle(element, 'position') == 'absolute') {
-      parent = Element.getOffsetParent(element);
-      delta = Element.viewportOffset(parent);
-    }
-
-    if (parent == document.body) {
-      delta[0] -= document.body.offsetLeft;
-      delta[1] -= document.body.offsetTop;
-    }
-
-    if (options.setLeft)   element.style.left  = (p[0] - delta[0] + options.offsetLeft) + 'px';
-    if (options.setTop)    element.style.top   = (p[1] - delta[1] + options.offsetTop) + 'px';
-    if (options.setWidth)  element.style.width = source.offsetWidth + 'px';
-    if (options.setHeight) element.style.height = source.offsetHeight + 'px';
-    return element;
-  }
-};
-
-Object.extend(Element.Methods, {
-  getElementsBySelector: Element.Methods.select,
-
-  childElements: Element.Methods.immediateDescendants
-});
-
-Element._attributeTranslations = {
-  write: {
-    names: {
-      className: 'class',
-      htmlFor:   'for'
-    },
-    values: { }
-  }
-};
-
-if (Prototype.Browser.Opera) {
-  Element.Methods.getStyle = Element.Methods.getStyle.wrap(
-    function(proceed, element, style) {
-      switch (style) {
-        case 'height': case 'width':
-          if (!Element.visible(element)) return null;
-
-          var dim = parseInt(proceed(element, style), 10);
-
-          if (dim !== element['offset' + style.capitalize()])
-            return dim + 'px';
-
-          var properties;
-          if (style === 'height') {
-            properties = ['border-top-width', 'padding-top',
-             'padding-bottom', 'border-bottom-width'];
-          }
-          else {
-            properties = ['border-left-width', 'padding-left',
-             'padding-right', 'border-right-width'];
-          }
-          return properties.inject(dim, function(memo, property) {
-            var val = proceed(element, property);
-            return val === null ? memo : memo - parseInt(val, 10);
-          }) + 'px';
-        default: return proceed(element, style);
-      }
-    }
-  );
-
-  Element.Methods.readAttribute = Element.Methods.readAttribute.wrap(
-    function(proceed, element, attribute) {
-      if (attribute === 'title') return element.title;
-      return proceed(element, attribute);
-    }
-  );
-}
-
-else if (Prototype.Browser.IE) {
-  Element.Methods.getStyle = function(element, style) {
-    element = $(element);
-    style = (style == 'float' || style == 'cssFloat') ? 'styleFloat' : style.camelize();
-    var value = element.style[style];
-    if (!value && element.currentStyle) value = element.currentStyle[style];
-
-    if (style == 'opacity') {
-      if (value = (element.getStyle('filter') || '').match(/alpha\(opacity=(.*)\)/))
-        if (value[1]) return parseFloat(value[1]) / 100;
-      return 1.0;
-    }
-
-    if (value == 'auto') {
-      if ((style == 'width' || style == 'height') && (element.getStyle('display') != 'none'))
-        return element['offset' + style.capitalize()] + 'px';
-      return null;
-    }
-    return value;
-  };
-
-  Element.Methods.setOpacity = function(element, value) {
-    function stripAlpha(filter){
-      return filter.replace(/alpha\([^\)]*\)/gi,'');
-    }
-    element = $(element);
-    var currentStyle = element.currentStyle;
-    if ((currentStyle && !currentStyle.hasLayout) ||
-      (!currentStyle && element.style.zoom == 'normal'))
-        element.style.zoom = 1;
-
-    var filter = element.getStyle('filter'), style = element.style;
-    if (value == 1 || value === '') {
-      (filter = stripAlpha(filter)) ?
-        style.filter = filter : style.removeAttribute('filter');
-      return element;
-    } else if (value < 0.00001) value = 0;
-    style.filter = stripAlpha(filter) +
-      'alpha(opacity=' + (value * 100) + ')';
-    return element;
-  };
-
-  Element._attributeTranslations = (function(){
-
-    var classProp = 'className',
-        forProp = 'for',
-        el = document.createElement('div');
-
-    el.setAttribute(classProp, 'x');
-
-    if (el.className !== 'x') {
-      el.setAttribute('class', 'x');
-      if (el.className === 'x') {
-        classProp = 'class';
-      }
-    }
-    el = null;
-
-    el = document.createElement('label');
-    el.setAttribute(forProp, 'x');
-    if (el.htmlFor !== 'x') {
-      el.setAttribute('htmlFor', 'x');
-      if (el.htmlFor === 'x') {
-        forProp = 'htmlFor';
-      }
-    }
-    el = null;
-
-    return {
-      read: {
-        names: {
-          'class':      classProp,
-          'className':  classProp,
-          'for':        forProp,
-          'htmlFor':    forProp
-        },
-        values: {
-          _getAttr: function(element, attribute) {
-            return element.getAttribute(attribute);
-          },
-          _getAttr2: function(element, attribute) {
-            return element.getAttribute(attribute, 2);
-          },
-          _getAttrNode: function(element, attribute) {
-            var node = element.getAttributeNode(attribute);
-            return node ? node.value : "";
-          },
-          _getEv: (function(){
-
-            var el = document.createElement('div'), f;
-            el.onclick = Prototype.emptyFunction;
-            var value = el.getAttribute('onclick');
-
-            if (String(value).indexOf('{') > -1) {
-              f = function(element, attribute) {
-                attribute = element.getAttribute(attribute);
-                if (!attribute) return null;
-                attribute = attribute.toString();
-                attribute = attribute.split('{')[1];
-                attribute = attribute.split('}')[0];
-                return attribute.strip();
-              };
-            }
-            else if (value === '') {
-              f = function(element, attribute) {
-                attribute = element.getAttribute(attribute);
-                if (!attribute) return null;
-                return attribute.strip();
-              };
-            }
-            el = null;
-            return f;
-          })(),
-          _flag: function(element, attribute) {
-            return $(element).hasAttribute(attribute) ? attribute : null;
-          },
-          style: function(element) {
-            return element.style.cssText.toLowerCase();
-          },
-          title: function(element) {
-            return element.title;
-          }
-        }
-      }
-    }
-  })();
-
-  Element._attributeTranslations.write = {
-    names: Object.extend({
-      cellpadding: 'cellPadding',
-      cellspacing: 'cellSpacing'
-    }, Element._attributeTranslations.read.names),
-    values: {
-      checked: function(element, value) {
-        element.checked = !!value;
-      },
-
-      style: function(element, value) {
-        element.style.cssText = value ? value : '';
-      }
-    }
-  };
-
-  Element._attributeTranslations.has = {};
-
-  $w('colSpan rowSpan vAlign dateTime accessKey tabIndex ' +
-      'encType maxLength readOnly longDesc frameBorder').each(function(attr) {
-    Element._attributeTranslations.write.names[attr.toLowerCase()] = attr;
-    Element._attributeTranslations.has[attr.toLowerCase()] = attr;
-  });
-
-  (function(v) {
-    Object.extend(v, {
-      href:        v._getAttr2,
-      src:         v._getAttr2,
-      type:        v._getAttr,
-      action:      v._getAttrNode,
-      disabled:    v._flag,
-      checked:     v._flag,
-      readonly:    v._flag,
-      multiple:    v._flag,
-      onload:      v._getEv,
-      onunload:    v._getEv,
-      onclick:     v._getEv,
-      ondblclick:  v._getEv,
-      onmousedown: v._getEv,
-      onmouseup:   v._getEv,
-      onmouseover: v._getEv,
-      onmousemove: v._getEv,
-      onmouseout:  v._getEv,
-      onfocus:     v._getEv,
-      onblur:      v._getEv,
-      onkeypress:  v._getEv,
-      onkeydown:   v._getEv,
-      onkeyup:     v._getEv,
-      onsubmit:    v._getEv,
-      onreset:     v._getEv,
-      onselect:    v._getEv,
-      onchange:    v._getEv
-    });
-  })(Element._attributeTranslations.read.values);
-
-  if (Prototype.BrowserFeatures.ElementExtensions) {
-    (function() {
-      function _descendants(element) {
-        var nodes = element.getElementsByTagName('*'), results = [];
-        for (var i = 0, node; node = nodes[i]; i++)
-          if (node.tagName !== "!") // Filter out comment nodes.
-            results.push(node);
-        return results;
-      }
-
-      Element.Methods.down = function(element, expression, index) {
-        element = $(element);
-        if (arguments.length == 1) return element.firstDescendant();
-        return Object.isNumber(expression) ? _descendants(element)[expression] :
-          Element.select(element, expression)[index || 0];
-      }
-    })();
-  }
-
-}
-
-else if (Prototype.Browser.Gecko && /rv:1\.8\.0/.test(navigator.userAgent)) {
-  Element.Methods.setOpacity = function(element, value) {
-    element = $(element);
-    element.style.opacity = (value == 1) ? 0.999999 :
-      (value === '') ? '' : (value < 0.00001) ? 0 : value;
-    return element;
-  };
-}
-
-else if (Prototype.Browser.WebKit) {
-  Element.Methods.setOpacity = function(element, value) {
-    element = $(element);
-    element.style.opacity = (value == 1 || value === '') ? '' :
-      (value < 0.00001) ? 0 : value;
-
-    if (value == 1)
-      if (element.tagName.toUpperCase() == 'IMG' && element.width) {
-        element.width++; element.width--;
-      } else try {
-        var n = document.createTextNode(' ');
-        element.appendChild(n);
-        element.removeChild(n);
-      } catch (e) { }
-
-    return element;
-  };
-}
-
-if ('outerHTML' in document.documentElement) {
-  Element.Methods.replace = function(element, content) {
-    element = $(element);
-
-    if (content && content.toElement) content = content.toElement();
-    if (Object.isElement(content)) {
-      element.parentNode.replaceChild(content, element);
-      return element;
-    }
-
-    content = Object.toHTML(content);
-    var parent = element.parentNode, tagName = parent.tagName.toUpperCase();
-
-    if (Element._insertionTranslations.tags[tagName]) {
-      var nextSibling = element.next(),
-          fragments = Element._getContentFromAnonymousElement(tagName, content.stripScripts());
-      parent.removeChild(element);
-      if (nextSibling)
-        fragments.each(function(node) { parent.insertBefore(node, nextSibling) });
-      else
-        fragments.each(function(node) { parent.appendChild(node) });
-    }
-    else element.outerHTML = content.stripScripts();
-
-    content.evalScripts.bind(content).defer();
-    return element;
-  };
-}
-
-Element._returnOffset = function(l, t) {
-  var result = [l, t];
-  result.left = l;
-  result.top = t;
-  return result;
-};
-
-Element._getContentFromAnonymousElement = function(tagName, html, force) {
-  var div = new Element('div'),
-      t = Element._insertionTranslations.tags[tagName];
-
-  var workaround = false;
-  if (t) workaround = true;
-  else if (force) {
-    workaround = true;
-    t = ['', '', 0];
-  }
-
-  if (workaround) {
-    div.innerHTML = '&nbsp;' + t[0] + html + t[1];
-    div.removeChild(div.firstChild);
-    for (var i = t[2]; i--; ) {
-      div = div.firstChild;
-    }
-  }
-  else {
-    div.innerHTML = html;
-  }
-  return $A(div.childNodes);
-};
-
-Element._insertionTranslations = {
-  before: function(element, node) {
-    element.parentNode.insertBefore(node, element);
-  },
-  top: function(element, node) {
-    element.insertBefore(node, element.firstChild);
-  },
-  bottom: function(element, node) {
-    element.appendChild(node);
-  },
-  after: function(element, node) {
-    element.parentNode.insertBefore(node, element.nextSibling);
-  },
-  tags: {
-    TABLE:  ['<table>',                '</table>',                   1],
-    TBODY:  ['<table><tbody>',         '</tbody></table>',           2],
-    TR:     ['<table><tbody><tr>',     '</tr></tbody></table>',      3],
-    TD:     ['<table><tbody><tr><td>', '</td></tr></tbody></table>', 4],
-    SELECT: ['<select>',               '</select>',                  1]
-  }
-};
-
-(function() {
-  var tags = Element._insertionTranslations.tags;
-  Object.extend(tags, {
-    THEAD: tags.TBODY,
-    TFOOT: tags.TBODY,
-    TH:    tags.TD
-  });
-})();
-
-Element.Methods.Simulated = {
-  hasAttribute: function(element, attribute) {
-    attribute = Element._attributeTranslations.has[attribute] || attribute;
-    var node = $(element).getAttributeNode(attribute);
-    return !!(node && node.specified);
-  }
-};
-
-Element.Methods.ByTag = { };
-
-Object.extend(Element, Element.Methods);
-
-(function(div) {
-
-  if (!Prototype.BrowserFeatures.ElementExtensions && div['__proto__']) {
-    window.HTMLElement = { };
-    window.HTMLElement.prototype = div['__proto__'];
-    Prototype.BrowserFeatures.ElementExtensions = true;
-  }
-
-  div = null;
-
-})(document.createElement('div'));
-
-Element.extend = (function() {
-
-  function checkDeficiency(tagName) {
-    if (typeof window.Element != 'undefined') {
-      var proto = window.Element.prototype;
-      if (proto) {
-        var id = '_' + (Math.random()+'').slice(2),
-            el = document.createElement(tagName);
-        proto[id] = 'x';
-        var isBuggy = (el[id] !== 'x');
-        delete proto[id];
-        el = null;
-        return isBuggy;
-      }
-    }
-    return false;
-  }
-
-  function extendElementWith(element, methods) {
-    for (var property in methods) {
-      var value = methods[property];
-      if (Object.isFunction(value) && !(property in element))
-        element[property] = value.methodize();
-    }
-  }
-
-  var HTMLOBJECTELEMENT_PROTOTYPE_BUGGY = checkDeficiency('object');
-
-  if (Prototype.BrowserFeatures.SpecificElementExtensions) {
-    if (HTMLOBJECTELEMENT_PROTOTYPE_BUGGY) {
-      return function(element) {
-        if (element && typeof element._extendedByPrototype == 'undefined') {
-          var t = element.tagName;
-          if (t && (/^(?:object|applet|embed)$/i.test(t))) {
-            extendElementWith(element, Element.Methods);
-            extendElementWith(element, Element.Methods.Simulated);
-            extendElementWith(element, Element.Methods.ByTag[t.toUpperCase()]);
-          }
-        }
-        return element;
-      }
-    }
-    return Prototype.K;
-  }
-
-  var Methods = { }, ByTag = Element.Methods.ByTag;
-
-  var extend = Object.extend(function(element) {
-    if (!element || typeof element._extendedByPrototype != 'undefined' ||
-        element.nodeType != 1 || element == window) return element;
-
-    var methods = Object.clone(Methods),
-        tagName = element.tagName.toUpperCase();
-
-    if (ByTag[tagName]) Object.extend(methods, ByTag[tagName]);
-
-    extendElementWith(element, methods);
-
-    element._extendedByPrototype = Prototype.emptyFunction;
-    return element;
-
-  }, {
-    refresh: function() {
-      if (!Prototype.BrowserFeatures.ElementExtensions) {
-        Object.extend(Methods, Element.Methods);
-        Object.extend(Methods, Element.Methods.Simulated);
-      }
-    }
-  });
-
-  extend.refresh();
-  return extend;
-})();
-
-if (document.documentElement.hasAttribute) {
-  Element.hasAttribute = function(element, attribute) {
-    return element.hasAttribute(attribute);
-  };
-}
-else {
-  Element.hasAttribute = Element.Methods.Simulated.hasAttribute;
-}
-
-Element.addMethods = function(methods) {
-  var F = Prototype.BrowserFeatures, T = Element.Methods.ByTag;
-
-  if (!methods) {
-    Object.extend(Form, Form.Methods);
-    Object.extend(Form.Element, Form.Element.Methods);
-    Object.extend(Element.Methods.ByTag, {
-      "FORM":     Object.clone(Form.Methods),
-      "INPUT":    Object.clone(Form.Element.Methods),
-      "SELECT":   Object.clone(Form.Element.Methods),
-      "TEXTAREA": Object.clone(Form.Element.Methods),
-      "BUTTON":   Object.clone(Form.Element.Methods)
-    });
-  }
-
-  if (arguments.length == 2) {
-    var tagName = methods;
-    methods = arguments[1];
-  }
-
-  if (!tagName) Object.extend(Element.Methods, methods || { });
-  else {
-    if (Object.isArray(tagName)) tagName.each(extend);
-    else extend(tagName);
-  }
-
-  function extend(tagName) {
-    tagName = tagName.toUpperCase();
-    if (!Element.Methods.ByTag[tagName])
-      Element.Methods.ByTag[tagName] = { };
-    Object.extend(Element.Methods.ByTag[tagName], methods);
-  }
-
-  function copy(methods, destination, onlyIfAbsent) {
-    onlyIfAbsent = onlyIfAbsent || false;
-    for (var property in methods) {
-      var value = methods[property];
-      if (!Object.isFunction(value)) continue;
-      if (!onlyIfAbsent || !(property in destination))
-        destination[property] = value.methodize();
-    }
-  }
-
-  function findDOMClass(tagName) {
-    var klass;
-    var trans = {
-      "OPTGROUP": "OptGroup", "TEXTAREA": "TextArea", "P": "Paragraph",
-      "FIELDSET": "FieldSet", "UL": "UList", "OL": "OList", "DL": "DList",
-      "DIR": "Directory", "H1": "Heading", "H2": "Heading", "H3": "Heading",
-      "H4": "Heading", "H5": "Heading", "H6": "Heading", "Q": "Quote",
-      "INS": "Mod", "DEL": "Mod", "A": "Anchor", "IMG": "Image", "CAPTION":
-      "TableCaption", "COL": "TableCol", "COLGROUP": "TableCol", "THEAD":
-      "TableSection", "TFOOT": "TableSection", "TBODY": "TableSection", "TR":
-      "TableRow", "TH": "TableCell", "TD": "TableCell", "FRAMESET":
-      "FrameSet", "IFRAME": "IFrame"
-    };
-    if (trans[tagName]) klass = 'HTML' + trans[tagName] + 'Element';
-    if (window[klass]) return window[klass];
-    klass = 'HTML' + tagName + 'Element';
-    if (window[klass]) return window[klass];
-    klass = 'HTML' + tagName.capitalize() + 'Element';
-    if (window[klass]) return window[klass];
-
-    var element = document.createElement(tagName),
-        proto = element['__proto__'] || element.constructor.prototype;
-
-    element = null;
-    return proto;
-  }
-
-  var elementPrototype = window.HTMLElement ? HTMLElement.prototype :
-   Element.prototype;
-
-  if (F.ElementExtensions) {
-    copy(Element.Methods, elementPrototype);
-    copy(Element.Methods.Simulated, elementPrototype, true);
-  }
-
-  if (F.SpecificElementExtensions) {
-    for (var tag in Element.Methods.ByTag) {
-      var klass = findDOMClass(tag);
-      if (Object.isUndefined(klass)) continue;
-      copy(T[tag], klass.prototype);
-    }
-  }
-
-  Object.extend(Element, Element.Methods);
-  delete Element.ByTag;
-
-  if (Element.extend.refresh) Element.extend.refresh();
-  Element.cache = { };
-};
-
-
-document.viewport = {
-
-  getDimensions: function() {
-    return { width: this.getWidth(), height: this.getHeight() };
-  },
-
-  getScrollOffsets: function() {
-    return Element._returnOffset(
-      window.pageXOffset || document.documentElement.scrollLeft || document.body.scrollLeft,
-      window.pageYOffset || document.documentElement.scrollTop  || document.body.scrollTop);
-  }
-};
-
-(function(viewport) {
-  var B = Prototype.Browser, doc = document, element, property = {};
-
-  function getRootElement() {
-    if (B.WebKit && !doc.evaluate)
-      return document;
-
-    if (B.Opera && window.parseFloat(window.opera.version()) < 9.5)
-      return document.body;
-
-    return document.documentElement;
-  }
-
-  function define(D) {
-    if (!element) element = getRootElement();
-
-    property[D] = 'client' + D;
-
-    viewport['get' + D] = function() { return element[property[D]] };
-    return viewport['get' + D]();
-  }
-
-  viewport.getWidth  = define.curry('Width');
-
-  viewport.getHeight = define.curry('Height');
-})(document.viewport);
-
-
-Element.Storage = {
-  UID: 1
-};
-
-Element.addMethods({
-  getStorage: function(element) {
-    if (!(element = $(element))) return;
-
-    var uid;
-    if (element === window) {
-      uid = 0;
-    } else {
-      if (typeof element._prototypeUID === "undefined")
-        element._prototypeUID = Element.Storage.UID++;
-      uid = element._prototypeUID;
-    }
-
-    if (!Element.Storage[uid])
-      Element.Storage[uid] = $H();
-
-    return Element.Storage[uid];
-  },
-
-  store: function(element, key, value) {
-    if (!(element = $(element))) return;
-
-    if (arguments.length === 2) {
-      Element.getStorage(element).update(key);
-    } else {
-      Element.getStorage(element).set(key, value);
-    }
-
-    return element;
-  },
-
-  retrieve: function(element, key, defaultValue) {
-    if (!(element = $(element))) return;
-    var hash = Element.getStorage(element), value = hash.get(key);
-
-    if (Object.isUndefined(value)) {
-      hash.set(key, defaultValue);
-      value = defaultValue;
-    }
-
-    return value;
-  },
-
-  clone: function(element, deep) {
-    if (!(element = $(element))) return;
-    var clone = element.cloneNode(deep);
-    clone._prototypeUID = void 0;
-    if (deep) {
-      var descendants = Element.select(clone, '*'),
-          i = descendants.length;
-      while (i--) {
-        descendants[i]._prototypeUID = void 0;
-      }
-    }
-    return Element.extend(clone);
-  },
-
-  purge: function(element) {
-    if (!(element = $(element))) return;
-    var purgeElement = Element._purgeElement;
-
-    purgeElement(element);
-
-    var descendants = element.getElementsByTagName('*'),
-     i = descendants.length;
-
-    while (i--) purgeElement(descendants[i]);
-
-    return null;
-  }
-});
-
-(function() {
-
-  function toDecimal(pctString) {
-    var match = pctString.match(/^(\d+)%?$/i);
-    if (!match) return null;
-    return (Number(match[1]) / 100);
-  }
-
-  function getPixelValue(value, property, context) {
-    var element = null;
-    if (Object.isElement(value)) {
-      element = value;
-      value = element.getStyle(property);
-    }
-
-    if (value === null) {
-      return null;
-    }
-
-    if ((/^(?:-)?\d+(\.\d+)?(px)?$/i).test(value)) {
-      return window.parseFloat(value);
-    }
-
-    var isPercentage = value.include('%'), isViewport = (context === document.viewport);
-
-    if (/\d/.test(value) && element && element.runtimeStyle && !(isPercentage && isViewport)) {
-      var style = element.style.left, rStyle = element.runtimeStyle.left;
-      element.runtimeStyle.left = element.currentStyle.left;
-      element.style.left = value || 0;
-      value = element.style.pixelLeft;
-      element.style.left = style;
-      element.runtimeStyle.left = rStyle;
-
-      return value;
-    }
-
-    if (element && isPercentage) {
-      context = context || element.parentNode;
-      var decimal = toDecimal(value);
-      var whole = null;
-      var position = element.getStyle('position');
-
-      var isHorizontal = property.include('left') || property.include('right') ||
-       property.include('width');
-
-      var isVertical =  property.include('top') || property.include('bottom') ||
-        property.include('height');
-
-      if (context === document.viewport) {
-        if (isHorizontal) {
-          whole = document.viewport.getWidth();
-        } else if (isVertical) {
-          whole = document.viewport.getHeight();
-        }
-      } else {
-        if (isHorizontal) {
-          whole = $(context).measure('width');
-        } else if (isVertical) {
-          whole = $(context).measure('height');
-        }
-      }
-
-      return (whole === null) ? 0 : whole * decimal;
-    }
-
-    return 0;
-  }
-
-  function toCSSPixels(number) {
-    if (Object.isString(number) && number.endsWith('px')) {
-      return number;
-    }
-    return number + 'px';
-  }
-
-  function isDisplayed(element) {
-    var originalElement = element;
-    while (element && element.parentNode) {
-      var display = element.getStyle('display');
-      if (display === 'none') {
-        return false;
-      }
-      element = $(element.parentNode);
-    }
-    return true;
-  }
-
-  var hasLayout = Prototype.K;
-  if ('currentStyle' in document.documentElement) {
-    hasLayout = function(element) {
-      if (!element.currentStyle.hasLayout) {
-        element.style.zoom = 1;
-      }
-      return element;
-    };
-  }
-
-  function cssNameFor(key) {
-    if (key.include('border')) key = key + '-width';
-    return key.camelize();
-  }
-
-  Element.Layout = Class.create(Hash, {
-    initialize: function($super, element, preCompute) {
-      $super();
-      this.element = $(element);
-
-      Element.Layout.PROPERTIES.each( function(property) {
-        this._set(property, null);
-      }, this);
-
-      if (preCompute) {
-        this._preComputing = true;
-        this._begin();
-        Element.Layout.PROPERTIES.each( this._compute, this );
-        this._end();
-        this._preComputing = false;
-      }
-    },
-
-    _set: function(property, value) {
-      return Hash.prototype.set.call(this, property, value);
-    },
-
-    set: function(property, value) {
-      throw "Properties of Element.Layout are read-only.";
-    },
-
-    get: function($super, property) {
-      var value = $super(property);
-      return value === null ? this._compute(property) : value;
-    },
-
-    _begin: function() {
-      if (this._prepared) return;
-
-      var element = this.element;
-      if (isDisplayed(element)) {
-        this._prepared = true;
-        return;
-      }
-
-      var originalStyles = {
-        position:   element.style.position   || '',
-        width:      element.style.width      || '',
-        visibility: element.style.visibility || '',
-        display:    element.style.display    || ''
-      };
-
-      element.store('prototype_original_styles', originalStyles);
-
-      var position = element.getStyle('position'),
-       width = element.getStyle('width');
-
-      if (width === "0px" || width === null) {
-        element.style.display = 'block';
-        width = element.getStyle('width');
-      }
-
-      var context = (position === 'fixed') ? document.viewport :
-       element.parentNode;
-
-      element.setStyle({
-        position:   'absolute',
-        visibility: 'hidden',
-        display:    'block'
-      });
-
-      var positionedWidth = element.getStyle('width');
-
-      var newWidth;
-      if (width && (positionedWidth === width)) {
-        newWidth = getPixelValue(element, 'width', context);
-      } else if (position === 'absolute' || position === 'fixed') {
-        newWidth = getPixelValue(element, 'width', context);
-      } else {
-        var parent = element.parentNode, pLayout = $(parent).getLayout();
-
-        newWidth = pLayout.get('width') -
-         this.get('margin-left') -
-         this.get('border-left') -
-         this.get('padding-left') -
-         this.get('padding-right') -
-         this.get('border-right') -
-         this.get('margin-right');
-      }
-
-      element.setStyle({ width: newWidth + 'px' });
-
-      this._prepared = true;
-    },
-
-    _end: function() {
-      var element = this.element;
-      var originalStyles = element.retrieve('prototype_original_styles');
-      element.store('prototype_original_styles', null);
-      element.setStyle(originalStyles);
-      this._prepared = false;
-    },
-
-    _compute: function(property) {
-      var COMPUTATIONS = Element.Layout.COMPUTATIONS;
-      if (!(property in COMPUTATIONS)) {
-        throw "Property not found.";
-      }
-
-      return this._set(property, COMPUTATIONS[property].call(this, this.element));
-    },
-
-    toObject: function() {
-      var args = $A(arguments);
-      var keys = (args.length === 0) ? Element.Layout.PROPERTIES :
-       args.join(' ').split(' ');
-      var obj = {};
-      keys.each( function(key) {
-        if (!Element.Layout.PROPERTIES.include(key)) return;
-        var value = this.get(key);
-        if (value != null) obj[key] = value;
-      }, this);
-      return obj;
-    },
-
-    toHash: function() {
-      var obj = this.toObject.apply(this, arguments);
-      return new Hash(obj);
-    },
-
-    toCSS: function() {
-      var args = $A(arguments);
-      var keys = (args.length === 0) ? Element.Layout.PROPERTIES :
-       args.join(' ').split(' ');
-      var css = {};
-
-      keys.each( function(key) {
-        if (!Element.Layout.PROPERTIES.include(key)) return;
-        if (Element.Layout.COMPOSITE_PROPERTIES.include(key)) return;
-
-        var value = this.get(key);
-        if (value != null) css[cssNameFor(key)] = value + 'px';
-      }, this);
-      return css;
-    },
-
-    inspect: function() {
-      return "#<Element.Layout>";
-    }
-  });
-
-  Object.extend(Element.Layout, {
-    PROPERTIES: $w('height width top left right bottom border-left border-right border-top border-bottom padding-left padding-right padding-top padding-bottom margin-top margin-bottom margin-left margin-right padding-box-width padding-box-height border-box-width border-box-height margin-box-width margin-box-height'),
-
-    COMPOSITE_PROPERTIES: $w('padding-box-width padding-box-height margin-box-width margin-box-height border-box-width border-box-height'),
-
-    COMPUTATIONS: {
-      'height': function(element) {
-        if (!this._preComputing) this._begin();
-
-        var bHeight = this.get('border-box-height');
-        if (bHeight <= 0) {
-          if (!this._preComputing) this._end();
-          return 0;
-        }
-
-        var bTop = this.get('border-top'),
-         bBottom = this.get('border-bottom');
-
-        var pTop = this.get('padding-top'),
-         pBottom = this.get('padding-bottom');
-
-        if (!this._preComputing) this._end();
-
-        return bHeight - bTop - bBottom - pTop - pBottom;
-      },
-
-      'width': function(element) {
-        if (!this._preComputing) this._begin();
-
-        var bWidth = this.get('border-box-width');
-        if (bWidth <= 0) {
-          if (!this._preComputing) this._end();
-          return 0;
-        }
-
-        var bLeft = this.get('border-left'),
-         bRight = this.get('border-right');
-
-        var pLeft = this.get('padding-left'),
-         pRight = this.get('padding-right');
-
-        if (!this._preComputing) this._end();
-
-        return bWidth - bLeft - bRight - pLeft - pRight;
-      },
-
-      'padding-box-height': function(element) {
-        var height = this.get('height'),
-         pTop = this.get('padding-top'),
-         pBottom = this.get('padding-bottom');
-
-        return height + pTop + pBottom;
-      },
-
-      'padding-box-width': function(element) {
-        var width = this.get('width'),
-         pLeft = this.get('padding-left'),
-         pRight = this.get('padding-right');
-
-        return width + pLeft + pRight;
-      },
-
-      'border-box-height': function(element) {
-        if (!this._preComputing) this._begin();
-        var height = element.offsetHeight;
-        if (!this._preComputing) this._end();
-        return height;
-      },
-
-      'border-box-width': function(element) {
-        if (!this._preComputing) this._begin();
-        var width = element.offsetWidth;
-        if (!this._preComputing) this._end();
-        return width;
-      },
-
-      'margin-box-height': function(element) {
-        var bHeight = this.get('border-box-height'),
-         mTop = this.get('margin-top'),
-         mBottom = this.get('margin-bottom');
-
-        if (bHeight <= 0) return 0;
-
-        return bHeight + mTop + mBottom;
-      },
-
-      'margin-box-width': function(element) {
-        var bWidth = this.get('border-box-width'),
-         mLeft = this.get('margin-left'),
-         mRight = this.get('margin-right');
-
-        if (bWidth <= 0) return 0;
-
-        return bWidth + mLeft + mRight;
-      },
-
-      'top': function(element) {
-        var offset = element.positionedOffset();
-        return offset.top;
-      },
-
-      'bottom': function(element) {
-        var offset = element.positionedOffset(),
-         parent = element.getOffsetParent(),
-         pHeight = parent.measure('height');
-
-        var mHeight = this.get('border-box-height');
-
-        return pHeight - mHeight - offset.top;
-      },
-
-      'left': function(element) {
-        var offset = element.positionedOffset();
-        return offset.left;
-      },
-
-      'right': function(element) {
-        var offset = element.positionedOffset(),
-         parent = element.getOffsetParent(),
-         pWidth = parent.measure('width');
-
-        var mWidth = this.get('border-box-width');
-
-        return pWidth - mWidth - offset.left;
-      },
-
-      'padding-top': function(element) {
-        return getPixelValue(element, 'paddingTop');
-      },
-
-      'padding-bottom': function(element) {
-        return getPixelValue(element, 'paddingBottom');
-      },
-
-      'padding-left': function(element) {
-        return getPixelValue(element, 'paddingLeft');
-      },
-
-      'padding-right': function(element) {
-        return getPixelValue(element, 'paddingRight');
-      },
-
-      'border-top': function(element) {
-        return getPixelValue(element, 'borderTopWidth');
-      },
-
-      'border-bottom': function(element) {
-        return getPixelValue(element, 'borderBottomWidth');
-      },
-
-      'border-left': function(element) {
-        return getPixelValue(element, 'borderLeftWidth');
-      },
-
-      'border-right': function(element) {
-        return getPixelValue(element, 'borderRightWidth');
-      },
-
-      'margin-top': function(element) {
-        return getPixelValue(element, 'marginTop');
-      },
-
-      'margin-bottom': function(element) {
-        return getPixelValue(element, 'marginBottom');
-      },
-
-      'margin-left': function(element) {
-        return getPixelValue(element, 'marginLeft');
-      },
-
-      'margin-right': function(element) {
-        return getPixelValue(element, 'marginRight');
-      }
-    }
-  });
-
-  if ('getBoundingClientRect' in document.documentElement) {
-    Object.extend(Element.Layout.COMPUTATIONS, {
-      'right': function(element) {
-        var parent = hasLayout(element.getOffsetParent());
-        var rect = element.getBoundingClientRect(),
-         pRect = parent.getBoundingClientRect();
-
-        return (pRect.right - rect.right).round();
-      },
-
-      'bottom': function(element) {
-        var parent = hasLayout(element.getOffsetParent());
-        var rect = element.getBoundingClientRect(),
-         pRect = parent.getBoundingClientRect();
-
-        return (pRect.bottom - rect.bottom).round();
-      }
-    });
-  }
-
-  Element.Offset = Class.create({
-    initialize: function(left, top) {
-      this.left = left.round();
-      this.top  = top.round();
-
-      this[0] = this.left;
-      this[1] = this.top;
-    },
-
-    relativeTo: function(offset) {
-      return new Element.Offset(
-        this.left - offset.left,
-        this.top  - offset.top
-      );
-    },
-
-    inspect: function() {
-      return "#<Element.Offset left: #{left} top: #{top}>".interpolate(this);
-    },
-
-    toString: function() {
-      return "[#{left}, #{top}]".interpolate(this);
-    },
-
-    toArray: function() {
-      return [this.left, this.top];
-    }
-  });
-
-  function getLayout(element, preCompute) {
-    return new Element.Layout(element, preCompute);
-  }
-
-  function measure(element, property) {
-    return $(element).getLayout().get(property);
-  }
-
-  function getDimensions(element) {
-    element = $(element);
-    var display = Element.getStyle(element, 'display');
-
-    if (display && display !== 'none') {
-      return { width: element.offsetWidth, height: element.offsetHeight };
-    }
-
-    var style = element.style;
-    var originalStyles = {
-      visibility: style.visibility,
-      position:   style.position,
-      display:    style.display
-    };
-
-    var newStyles = {
-      visibility: 'hidden',
-      display:    'block'
-    };
-
-    if (originalStyles.position !== 'fixed')
-      newStyles.position = 'absolute';
-
-    Element.setStyle(element, newStyles);
-
-    var dimensions = {
-      width:  element.offsetWidth,
-      height: element.offsetHeight
-    };
-
-    Element.setStyle(element, originalStyles);
-
-    return dimensions;
-  }
-
-  function getOffsetParent(element) {
-    element = $(element);
-
-    if (isDocument(element) || isDetached(element) || isBody(element) || isHtml(element))
-      return $(document.body);
-
-    var isInline = (Element.getStyle(element, 'display') === 'inline');
-    if (!isInline && element.offsetParent) return $(element.offsetParent);
-
-    while ((element = element.parentNode) && element !== document.body) {
-      if (Element.getStyle(element, 'position') !== 'static') {
-        return isHtml(element) ? $(document.body) : $(element);
-      }
-    }
-
-    return $(document.body);
-  }
-
-
-  function cumulativeOffset(element) {
-    element = $(element);
-    var valueT = 0, valueL = 0;
-    if (element.parentNode) {
-      do {
-        valueT += element.offsetTop  || 0;
-        valueL += element.offsetLeft || 0;
-        element = element.offsetParent;
-      } while (element);
-    }
-    return new Element.Offset(valueL, valueT);
-  }
-
-  function positionedOffset(element) {
-    element = $(element);
-
-    var layout = element.getLayout();
-
-    var valueT = 0, valueL = 0;
-    do {
-      valueT += element.offsetTop  || 0;
-      valueL += element.offsetLeft || 0;
-      element = element.offsetParent;
-      if (element) {
-        if (isBody(element)) break;
-        var p = Element.getStyle(element, 'position');
-        if (p !== 'static') break;
-      }
-    } while (element);
-
-    valueL -= layout.get('margin-top');
-    valueT -= layout.get('margin-left');
-
-    return new Element.Offset(valueL, valueT);
-  }
-
-  function cumulativeScrollOffset(element) {
-    var valueT = 0, valueL = 0;
-    do {
-      valueT += element.scrollTop  || 0;
-      valueL += element.scrollLeft || 0;
-      element = element.parentNode;
-    } while (element);
-    return new Element.Offset(valueL, valueT);
-  }
-
-  function viewportOffset(forElement) {
-    element = $(element);
-    var valueT = 0, valueL = 0, docBody = document.body;
-
-    var element = forElement;
-    do {
-      valueT += element.offsetTop  || 0;
-      valueL += element.offsetLeft || 0;
-      if (element.offsetParent == docBody &&
-        Element.getStyle(element, 'position') == 'absolute') break;
-    } while (element = element.offsetParent);
-
-    element = forElement;
-    do {
-      if (element != docBody) {
-        valueT -= element.scrollTop  || 0;
-        valueL -= element.scrollLeft || 0;
-      }
-    } while (element = element.parentNode);
-    return new Element.Offset(valueL, valueT);
-  }
-
-  function absolutize(element) {
-    element = $(element);
-
-    if (Element.getStyle(element, 'position') === 'absolute') {
-      return element;
-    }
-
-    var offsetParent = getOffsetParent(element);
-    var eOffset = element.viewportOffset(),
-     pOffset = offsetParent.viewportOffset();
-
-    var offset = eOffset.relativeTo(pOffset);
-    var layout = element.getLayout();
-
-    element.store('prototype_absolutize_original_styles', {
-      left:   element.getStyle('left'),
-      top:    element.getStyle('top'),
-      width:  element.getStyle('width'),
-      height: element.getStyle('height')
-    });
-
-    element.setStyle({
-      position: 'absolute',
-      top:    offset.top + 'px',
-      left:   offset.left + 'px',
-      width:  layout.get('width') + 'px',
-      height: layout.get('height') + 'px'
-    });
-
-    return element;
-  }
-
-  function relativize(element) {
-    element = $(element);
-    if (Element.getStyle(element, 'position') === 'relative') {
-      return element;
-    }
-
-    var originalStyles =
-     element.retrieve('prototype_absolutize_original_styles');
-
-    if (originalStyles) element.setStyle(originalStyles);
-    return element;
-  }
-
-  if (Prototype.Browser.IE) {
-    getOffsetParent = getOffsetParent.wrap(
-      function(proceed, element) {
-        element = $(element);
-
-        if (isDocument(element) || isDetached(element) || isBody(element) || isHtml(element))
-          return $(document.body);
-
-        var position = element.getStyle('position');
-        if (position !== 'static') return proceed(element);
-
-        element.setStyle({ position: 'relative' });
-        var value = proceed(element);
-        element.setStyle({ position: position });
-        return value;
-      }
-    );
-
-    positionedOffset = positionedOffset.wrap(function(proceed, element) {
-      element = $(element);
-      if (!element.parentNode) return new Element.Offset(0, 0);
-      var position = element.getStyle('position');
-      if (position !== 'static') return proceed(element);
-
-      var offsetParent = element.getOffsetParent();
-      if (offsetParent && offsetParent.getStyle('position') === 'fixed')
-        hasLayout(offsetParent);
-
-      element.setStyle({ position: 'relative' });
-      var value = proceed(element);
-      element.setStyle({ position: position });
-      return value;
-    });
-  } else if (Prototype.Browser.Webkit) {
-    cumulativeOffset = function(element) {
-      element = $(element);
-      var valueT = 0, valueL = 0;
-      do {
-        valueT += element.offsetTop  || 0;
-        valueL += element.offsetLeft || 0;
-        if (element.offsetParent == document.body)
-          if (Element.getStyle(element, 'position') == 'absolute') break;
-
-        element = element.offsetParent;
-      } while (element);
-
-      return new Element.Offset(valueL, valueT);
-    };
-  }
-
-
-  Element.addMethods({
-    getLayout:              getLayout,
-    measure:                measure,
-    getDimensions:          getDimensions,
-    getOffsetParent:        getOffsetParent,
-    cumulativeOffset:       cumulativeOffset,
-    positionedOffset:       positionedOffset,
-    cumulativeScrollOffset: cumulativeScrollOffset,
-    viewportOffset:         viewportOffset,
-    absolutize:             absolutize,
-    relativize:             relativize
-  });
-
-  function isBody(element) {
-    return element.nodeName.toUpperCase() === 'BODY';
-  }
-
-  function isHtml(element) {
-    return element.nodeName.toUpperCase() === 'HTML';
-  }
-
-  function isDocument(element) {
-    return element.nodeType === Node.DOCUMENT_NODE;
-  }
-
-  function isDetached(element) {
-    return element !== document.body &&
-     !Element.descendantOf(element, document.body);
-  }
-
-  if ('getBoundingClientRect' in document.documentElement) {
-    Element.addMethods({
-      viewportOffset: function(element) {
-        element = $(element);
-        if (isDetached(element)) return new Element.Offset(0, 0);
-
-        var rect = element.getBoundingClientRect(),
-         docEl = document.documentElement;
-        return new Element.Offset(rect.left - docEl.clientLeft,
-         rect.top - docEl.clientTop);
-      }
-    });
-  }
-})();
-window.$$ = function() {
-  var expression = $A(arguments).join(', ');
-  return Prototype.Selector.select(expression, document);
-};
-
-Prototype.Selector = (function() {
-
-  function select() {
-    throw new Error('Method "Prototype.Selector.select" must be defined.');
-  }
-
-  function match() {
-    throw new Error('Method "Prototype.Selector.match" must be defined.');
-  }
-
-  function find(elements, expression, index) {
-    index = index || 0;
-    var match = Prototype.Selector.match, length = elements.length, matchIndex = 0, i;
-
-    for (i = 0; i < length; i++) {
-      if (match(elements[i], expression) && index == matchIndex++) {
-        return Element.extend(elements[i]);
-      }
-    }
-  }
-
-  function extendElements(elements) {
-    for (var i = 0, length = elements.length; i < length; i++) {
-      Element.extend(elements[i]);
-    }
-    return elements;
-  }
-
-
-  var K = Prototype.K;
-
-  return {
-    select: select,
-    match: match,
-    find: find,
-    extendElements: (Element.extend === K) ? K : extendElements,
-    extendElement: Element.extend
-  };
-})();
-Prototype._original_property = window.Sizzle;
-/*!
- * Sizzle CSS Selector Engine - v1.0
- *  Copyright 2009, The Dojo Foundation
- *  Released under the MIT, BSD, and GPL Licenses.
- *  More information: http://sizzlejs.com/
- */
-(function(){
-
-var chunker = /((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^[\]]*\]|['"][^'"]*['"]|[^[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,
-       done = 0,
-       toString = Object.prototype.toString,
-       hasDuplicate = false,
-       baseHasDuplicate = true;
-
-[0, 0].sort(function(){
-       baseHasDuplicate = false;
-       return 0;
-});
-
-var Sizzle = function(selector, context, results, seed) {
-       results = results || [];
-       var origContext = context = context || document;
-
-       if ( context.nodeType !== 1 && context.nodeType !== 9 ) {
-               return [];
-       }
-
-       if ( !selector || typeof selector !== "string" ) {
-               return results;
-       }
-
-       var parts = [], m, set, checkSet, check, mode, extra, prune = true, contextXML = isXML(context),
-               soFar = selector;
-
-       while ( (chunker.exec(""), m = chunker.exec(soFar)) !== null ) {
-               soFar = m[3];
-
-               parts.push( m[1] );
-
-               if ( m[2] ) {
-                       extra = m[3];
-                       break;
-               }
-       }
-
-       if ( parts.length > 1 && origPOS.exec( selector ) ) {
-               if ( parts.length === 2 && Expr.relative[ parts[0] ] ) {
-                       set = posProcess( parts[0] + parts[1], context );
-               } else {
-                       set = Expr.relative[ parts[0] ] ?
-                               [ context ] :
-                               Sizzle( parts.shift(), context );
-
-                       while ( parts.length ) {
-                               selector = parts.shift();
-
-                               if ( Expr.relative[ selector ] )
-                                       selector += parts.shift();
-
-                               set = posProcess( selector, set );
-                       }
-               }
-       } else {
-               if ( !seed && parts.length > 1 && context.nodeType === 9 && !contextXML &&
-                               Expr.match.ID.test(parts[0]) && !Expr.match.ID.test(parts[parts.length - 1]) ) {
-                       var ret = Sizzle.find( parts.shift(), context, contextXML );
-                       context = ret.expr ? Sizzle.filter( ret.expr, ret.set )[0] : ret.set[0];
-               }
-
-               if ( context ) {
-                       var ret = seed ?
-                               { expr: parts.pop(), set: makeArray(seed) } :
-                               Sizzle.find( parts.pop(), parts.length === 1 && (parts[0] === "~" || parts[0] === "+") && context.parentNode ? context.parentNode : context, contextXML );
-                       set = ret.expr ? Sizzle.filter( ret.expr, ret.set ) : ret.set;
-
-                       if ( parts.length > 0 ) {
-                               checkSet = makeArray(set);
-                       } else {
-                               prune = false;
-                       }
-
-                       while ( parts.length ) {
-                               var cur = parts.pop(), pop = cur;
-
-                               if ( !Expr.relative[ cur ] ) {
-                                       cur = "";
-                               } else {
-                                       pop = parts.pop();
-                               }
-
-                               if ( pop == null ) {
-                                       pop = context;
-                               }
-
-                               Expr.relative[ cur ]( checkSet, pop, contextXML );
-                       }
-               } else {
-                       checkSet = parts = [];
-               }
-       }
-
-       if ( !checkSet ) {
-               checkSet = set;
-       }
-
-       if ( !checkSet ) {
-               throw "Syntax error, unrecognized expression: " + (cur || selector);
-       }
-
-       if ( toString.call(checkSet) === "[object Array]" ) {
-               if ( !prune ) {
-                       results.push.apply( results, checkSet );
-               } else if ( context && context.nodeType === 1 ) {
-                       for ( var i = 0; checkSet[i] != null; i++ ) {
-                               if ( checkSet[i] && (checkSet[i] === true || checkSet[i].nodeType === 1 && contains(context, checkSet[i])) ) {
-                                       results.push( set[i] );
-                               }
-                       }
-               } else {
-                       for ( var i = 0; checkSet[i] != null; i++ ) {
-                               if ( checkSet[i] && checkSet[i].nodeType === 1 ) {
-                                       results.push( set[i] );
-                               }
-                       }
-               }
-       } else {
-               makeArray( checkSet, results );
-       }
-
-       if ( extra ) {
-               Sizzle( extra, origContext, results, seed );
-               Sizzle.uniqueSort( results );
-       }
-
-       return results;
-};
-
-Sizzle.uniqueSort = function(results){
-       if ( sortOrder ) {
-               hasDuplicate = baseHasDuplicate;
-               results.sort(sortOrder);
-
-               if ( hasDuplicate ) {
-                       for ( var i = 1; i < results.length; i++ ) {
-                               if ( results[i] === results[i-1] ) {
-                                       results.splice(i--, 1);
-                               }
-                       }
-               }
-       }
-
-       return results;
-};
-
-Sizzle.matches = function(expr, set){
-       return Sizzle(expr, null, null, set);
-};
-
-Sizzle.find = function(expr, context, isXML){
-       var set, match;
-
-       if ( !expr ) {
-               return [];
-       }
-
-       for ( var i = 0, l = Expr.order.length; i < l; i++ ) {
-               var type = Expr.order[i], match;
-
-               if ( (match = Expr.leftMatch[ type ].exec( expr )) ) {
-                       var left = match[1];
-                       match.splice(1,1);
-
-                       if ( left.substr( left.length - 1 ) !== "\\" ) {
-                               match[1] = (match[1] || "").replace(/\\/g, "");
-                               set = Expr.find[ type ]( match, context, isXML );
-                               if ( set != null ) {
-                                       expr = expr.replace( Expr.match[ type ], "" );
-                                       break;
-                               }
-                       }
-               }
-       }
-
-       if ( !set ) {
-               set = context.getElementsByTagName("*");
-       }
-
-       return {set: set, expr: expr};
-};
-
-Sizzle.filter = function(expr, set, inplace, not){
-       var old = expr, result = [], curLoop = set, match, anyFound,
-               isXMLFilter = set && set[0] && isXML(set[0]);
-
-       while ( expr && set.length ) {
-               for ( var type in Expr.filter ) {
-                       if ( (match = Expr.match[ type ].exec( expr )) != null ) {
-                               var filter = Expr.filter[ type ], found, item;
-                               anyFound = false;
-
-                               if ( curLoop == result ) {
-                                       result = [];
-                               }
-
-                               if ( Expr.preFilter[ type ] ) {
-                                       match = Expr.preFilter[ type ]( match, curLoop, inplace, result, not, isXMLFilter );
-
-                                       if ( !match ) {
-                                               anyFound = found = true;
-                                       } else if ( match === true ) {
-                                               continue;
-                                       }
-                               }
-
-                               if ( match ) {
-                                       for ( var i = 0; (item = curLoop[i]) != null; i++ ) {
-                                               if ( item ) {
-                                                       found = filter( item, match, i, curLoop );
-                                                       var pass = not ^ !!found;
-
-                                                       if ( inplace && found != null ) {
-                                                               if ( pass ) {
-                                                                       anyFound = true;
-                                                               } else {
-                                                                       curLoop[i] = false;
-                                                               }
-                                                       } else if ( pass ) {
-                                                               result.push( item );
-                                                               anyFound = true;
-                                                       }
-                                               }
-                                       }
-                               }
-
-                               if ( found !== undefined ) {
-                                       if ( !inplace ) {
-                                               curLoop = result;
-                                       }
-
-                                       expr = expr.replace( Expr.match[ type ], "" );
-
-                                       if ( !anyFound ) {
-                                               return [];
-                                       }
-
-                                       break;
-                               }
-                       }
-               }
-
-               if ( expr == old ) {
-                       if ( anyFound == null ) {
-                               throw "Syntax error, unrecognized expression: " + expr;
-                       } else {
-                               break;
-                       }
-               }
-
-               old = expr;
-       }
-
-       return curLoop;
-};
-
-var Expr = Sizzle.selectors = {
-       order: [ "ID", "NAME", "TAG" ],
-       match: {
-               ID: /#((?:[\w\u00c0-\uFFFF-]|\\.)+)/,
-               CLASS: /\.((?:[\w\u00c0-\uFFFF-]|\\.)+)/,
-               NAME: /\[name=['"]*((?:[\w\u00c0-\uFFFF-]|\\.)+)['"]*\]/,
-               ATTR: /\[\s*((?:[\w\u00c0-\uFFFF-]|\\.)+)\s*(?:(\S?=)\s*(['"]*)(.*?)\3|)\s*\]/,
-               TAG: /^((?:[\w\u00c0-\uFFFF\*-]|\\.)+)/,
-               CHILD: /:(only|nth|last|first)-child(?:\((even|odd|[\dn+-]*)\))?/,
-               POS: /:(nth|eq|gt|lt|first|last|even|odd)(?:\((\d*)\))?(?=[^-]|$)/,
-               PSEUDO: /:((?:[\w\u00c0-\uFFFF-]|\\.)+)(?:\((['"]*)((?:\([^\)]+\)|[^\2\(\)]*)+)\2\))?/
-       },
-       leftMatch: {},
-       attrMap: {
-               "class": "className",
-               "for": "htmlFor"
-       },
-       attrHandle: {
-               href: function(elem){
-                       return elem.getAttribute("href");
-               }
-       },
-       relative: {
-               "+": function(checkSet, part, isXML){
-                       var isPartStr = typeof part === "string",
-                               isTag = isPartStr && !/\W/.test(part),
-                               isPartStrNotTag = isPartStr && !isTag;
-
-                       if ( isTag && !isXML ) {
-                               part = part.toUpperCase();
-                       }
-
-                       for ( var i = 0, l = checkSet.length, elem; i < l; i++ ) {
-                               if ( (elem = checkSet[i]) ) {
-                                       while ( (elem = elem.previousSibling) && elem.nodeType !== 1 ) {}
-
-                                       checkSet[i] = isPartStrNotTag || elem && elem.nodeName === part ?
-                                               elem || false :
-                                               elem === part;
-                               }
-                       }
-
-                       if ( isPartStrNotTag ) {
-                               Sizzle.filter( part, checkSet, true );
-                       }
-               },
-               ">": function(checkSet, part, isXML){
-                       var isPartStr = typeof part === "string";
-
-                       if ( isPartStr && !/\W/.test(part) ) {
-                               part = isXML ? part : part.toUpperCase();
-
-                               for ( var i = 0, l = checkSet.length; i < l; i++ ) {
-                                       var elem = checkSet[i];
-                                       if ( elem ) {
-                                               var parent = elem.parentNode;
-                                               checkSet[i] = parent.nodeName === part ? parent : false;
-                                       }
-                               }
-                       } else {
-                               for ( var i = 0, l = checkSet.length; i < l; i++ ) {
-                                       var elem = checkSet[i];
-                                       if ( elem ) {
-                                               checkSet[i] = isPartStr ?
-                                                       elem.parentNode :
-                                                       elem.parentNode === part;
-                                       }
-                               }
-
-                               if ( isPartStr ) {
-                                       Sizzle.filter( part, checkSet, true );
-                               }
-                       }
-               },
-               "": function(checkSet, part, isXML){
-                       var doneName = done++, checkFn = dirCheck;
-
-                       if ( !/\W/.test(part) ) {
-                               var nodeCheck = part = isXML ? part : part.toUpperCase();
-                               checkFn = dirNodeCheck;
-                       }
-
-                       checkFn("parentNode", part, doneName, checkSet, nodeCheck, isXML);
-               },
-               "~": function(checkSet, part, isXML){
-                       var doneName = done++, checkFn = dirCheck;
-
-                       if ( typeof part === "string" && !/\W/.test(part) ) {
-                               var nodeCheck = part = isXML ? part : part.toUpperCase();
-                               checkFn = dirNodeCheck;
-                       }
-
-                       checkFn("previousSibling", part, doneName, checkSet, nodeCheck, isXML);
-               }
-       },
-       find: {
-               ID: function(match, context, isXML){
-                       if ( typeof context.getElementById !== "undefined" && !isXML ) {
-                               var m = context.getElementById(match[1]);
-                               return m ? [m] : [];
-                       }
-               },
-               NAME: function(match, context, isXML){
-                       if ( typeof context.getElementsByName !== "undefined" ) {
-                               var ret = [], results = context.getElementsByName(match[1]);
-
-                               for ( var i = 0, l = results.length; i < l; i++ ) {
-                                       if ( results[i].getAttribute("name") === match[1] ) {
-                                               ret.push( results[i] );
-                                       }
-                               }
-
-                               return ret.length === 0 ? null : ret;
-                       }
-               },
-               TAG: function(match, context){
-                       return context.getElementsByTagName(match[1]);
-               }
-       },
-       preFilter: {
-               CLASS: function(match, curLoop, inplace, result, not, isXML){
-                       match = " " + match[1].replace(/\\/g, "") + " ";
-
-                       if ( isXML ) {
-                               return match;
-                       }
-
-                       for ( var i = 0, elem; (elem = curLoop[i]) != null; i++ ) {
-                               if ( elem ) {
-                                       if ( not ^ (elem.className && (" " + elem.className + " ").indexOf(match) >= 0) ) {
-                                               if ( !inplace )
-                                                       result.push( elem );
-                                       } else if ( inplace ) {
-                                               curLoop[i] = false;
-                                       }
-                               }
-                       }
-
-                       return false;
-               },
-               ID: function(match){
-                       return match[1].replace(/\\/g, "");
-               },
-               TAG: function(match, curLoop){
-                       for ( var i = 0; curLoop[i] === false; i++ ){}
-                       return curLoop[i] && isXML(curLoop[i]) ? match[1] : match[1].toUpperCase();
-               },
-               CHILD: function(match){
-                       if ( match[1] == "nth" ) {
-                               var test = /(-?)(\d*)n((?:\+|-)?\d*)/.exec(
-                                       match[2] == "even" && "2n" || match[2] == "odd" && "2n+1" ||
-                                       !/\D/.test( match[2] ) && "0n+" + match[2] || match[2]);
-
-                               match[2] = (test[1] + (test[2] || 1)) - 0;
-                               match[3] = test[3] - 0;
-                       }
-
-                       match[0] = done++;
-
-                       return match;
-               },
-               ATTR: function(match, curLoop, inplace, result, not, isXML){
-                       var name = match[1].replace(/\\/g, "");
-
-                       if ( !isXML && Expr.attrMap[name] ) {
-                               match[1] = Expr.attrMap[name];
-                       }
-
-                       if ( match[2] === "~=" ) {
-                               match[4] = " " + match[4] + " ";
-                       }
-
-                       return match;
-               },
-               PSEUDO: function(match, curLoop, inplace, result, not){
-                       if ( match[1] === "not" ) {
-                               if ( ( chunker.exec(match[3]) || "" ).length > 1 || /^\w/.test(match[3]) ) {
-                                       match[3] = Sizzle(match[3], null, null, curLoop);
-                               } else {
-                                       var ret = Sizzle.filter(match[3], curLoop, inplace, true ^ not);
-                                       if ( !inplace ) {
-                                               result.push.apply( result, ret );
-                                       }
-                                       return false;
-                               }
-                       } else if ( Expr.match.POS.test( match[0] ) || Expr.match.CHILD.test( match[0] ) ) {
-                               return true;
-                       }
-
-                       return match;
-               },
-               POS: function(match){
-                       match.unshift( true );
-                       return match;
-               }
-       },
-       filters: {
-               enabled: function(elem){
-                       return elem.disabled === false && elem.type !== "hidden";
-               },
-               disabled: function(elem){
-                       return elem.disabled === true;
-               },
-               checked: function(elem){
-                       return elem.checked === true;
-               },
-               selected: function(elem){
-                       elem.parentNode.selectedIndex;
-                       return elem.selected === true;
-               },
-               parent: function(elem){
-                       return !!elem.firstChild;
-               },
-               empty: function(elem){
-                       return !elem.firstChild;
-               },
-               has: function(elem, i, match){
-                       return !!Sizzle( match[3], elem ).length;
-               },
-               header: function(elem){
-                       return /h\d/i.test( elem.nodeName );
-               },
-               text: function(elem){
-                       return "text" === elem.type;
-               },
-               radio: function(elem){
-                       return "radio" === elem.type;
-               },
-               checkbox: function(elem){
-                       return "checkbox" === elem.type;
-               },
-               file: function(elem){
-                       return "file" === elem.type;
-               },
-               password: function(elem){
-                       return "password" === elem.type;
-               },
-               submit: function(elem){
-                       return "submit" === elem.type;
-               },
-               image: function(elem){
-                       return "image" === elem.type;
-               },
-               reset: function(elem){
-                       return "reset" === elem.type;
-               },
-               button: function(elem){
-                       return "button" === elem.type || elem.nodeName.toUpperCase() === "BUTTON";
-               },
-               input: function(elem){
-                       return /input|select|textarea|button/i.test(elem.nodeName);
-               }
-       },
-       setFilters: {
-               first: function(elem, i){
-                       return i === 0;
-               },
-               last: function(elem, i, match, array){
-                       return i === array.length - 1;
-               },
-               even: function(elem, i){
-                       return i % 2 === 0;
-               },
-               odd: function(elem, i){
-                       return i % 2 === 1;
-               },
-               lt: function(elem, i, match){
-                       return i < match[3] - 0;
-               },
-               gt: function(elem, i, match){
-                       return i > match[3] - 0;
-               },
-               nth: function(elem, i, match){
-                       return match[3] - 0 == i;
-               },
-               eq: function(elem, i, match){
-                       return match[3] - 0 == i;
-               }
-       },
-       filter: {
-               PSEUDO: function(elem, match, i, array){
-                       var name = match[1], filter = Expr.filters[ name ];
-
-                       if ( filter ) {
-                               return filter( elem, i, match, array );
-                       } else if ( name === "contains" ) {
-                               return (elem.textContent || elem.innerText || "").indexOf(match[3]) >= 0;
-                       } else if ( name === "not" ) {
-                               var not = match[3];
-
-                               for ( var i = 0, l = not.length; i < l; i++ ) {
-                                       if ( not[i] === elem ) {
-                                               return false;
-                                       }
-                               }
-
-                               return true;
-                       }
-               },
-               CHILD: function(elem, match){
-                       var type = match[1], node = elem;
-                       switch (type) {
-                               case 'only':
-                               case 'first':
-                                       while ( (node = node.previousSibling) )  {
-                                               if ( node.nodeType === 1 ) return false;
-                                       }
-                                       if ( type == 'first') return true;
-                                       node = elem;
-                               case 'last':
-                                       while ( (node = node.nextSibling) )  {
-                                               if ( node.nodeType === 1 ) return false;
-                                       }
-                                       return true;
-                               case 'nth':
-                                       var first = match[2], last = match[3];
-
-                                       if ( first == 1 && last == 0 ) {
-                                               return true;
-                                       }
-
-                                       var doneName = match[0],
-                                               parent = elem.parentNode;
-
-                                       if ( parent && (parent.sizcache !== doneName || !elem.nodeIndex) ) {
-                                               var count = 0;
-                                               for ( node = parent.firstChild; node; node = node.nextSibling ) {
-                                                       if ( node.nodeType === 1 ) {
-                                                               node.nodeIndex = ++count;
-                                                       }
-                                               }
-                                               parent.sizcache = doneName;
-                                       }
-
-                                       var diff = elem.nodeIndex - last;
-                                       if ( first == 0 ) {
-                                               return diff == 0;
-                                       } else {
-                                               return ( diff % first == 0 && diff / first >= 0 );
-                                       }
-                       }
-               },
-               ID: function(elem, match){
-                       return elem.nodeType === 1 && elem.getAttribute("id") === match;
-               },
-               TAG: function(elem, match){
-                       return (match === "*" && elem.nodeType === 1) || elem.nodeName === match;
-               },
-               CLASS: function(elem, match){
-                       return (" " + (elem.className || elem.getAttribute("class")) + " ")
-                               .indexOf( match ) > -1;
-               },
-               ATTR: function(elem, match){
-                       var name = match[1],
-                               result = Expr.attrHandle[ name ] ?
-                                       Expr.attrHandle[ name ]( elem ) :
-                                       elem[ name ] != null ?
-                                               elem[ name ] :
-                                               elem.getAttribute( name ),
-                               value = result + "",
-                               type = match[2],
-                               check = match[4];
-
-                       return result == null ?
-                               type === "!=" :
-                               type === "=" ?
-                               value === check :
-                               type === "*=" ?
-                               value.indexOf(check) >= 0 :
-                               type === "~=" ?
-                               (" " + value + " ").indexOf(check) >= 0 :
-                               !check ?
-                               value && result !== false :
-                               type === "!=" ?
-                               value != check :
-                               type === "^=" ?
-                               value.indexOf(check) === 0 :
-                               type === "$=" ?
-                               value.substr(value.length - check.length) === check :
-                               type === "|=" ?
-                               value === check || value.substr(0, check.length + 1) === check + "-" :
-                               false;
-               },
-               POS: function(elem, match, i, array){
-                       var name = match[2], filter = Expr.setFilters[ name ];
-
-                       if ( filter ) {
-                               return filter( elem, i, match, array );
-                       }
-               }
-       }
-};
-
-var origPOS = Expr.match.POS;
-
-for ( var type in Expr.match ) {
-       Expr.match[ type ] = new RegExp( Expr.match[ type ].source + /(?![^\[]*\])(?![^\(]*\))/.source );
-       Expr.leftMatch[ type ] = new RegExp( /(^(?:.|\r|\n)*?)/.source + Expr.match[ type ].source );
-}
-
-var makeArray = function(array, results) {
-       array = Array.prototype.slice.call( array, 0 );
-
-       if ( results ) {
-               results.push.apply( results, array );
-               return results;
-       }
-
-       return array;
-};
-
-try {
-       Array.prototype.slice.call( document.documentElement.childNodes, 0 );
-
-} catch(e){
-       makeArray = function(array, results) {
-               var ret = results || [];
-
-               if ( toString.call(array) === "[object Array]" ) {
-                       Array.prototype.push.apply( ret, array );
-               } else {
-                       if ( typeof array.length === "number" ) {
-                               for ( var i = 0, l = array.length; i < l; i++ ) {
-                                       ret.push( array[i] );
-                               }
-                       } else {
-                               for ( var i = 0; array[i]; i++ ) {
-                                       ret.push( array[i] );
-                               }
-                       }
-               }
-
-               return ret;
-       };
-}
-
-var sortOrder;
-
-if ( document.documentElement.compareDocumentPosition ) {
-       sortOrder = function( a, b ) {
-               if ( !a.compareDocumentPosition || !b.compareDocumentPosition ) {
-                       if ( a == b ) {
-                               hasDuplicate = true;
-                       }
-                       return 0;
-               }
-
-               var ret = a.compareDocumentPosition(b) & 4 ? -1 : a === b ? 0 : 1;
-               if ( ret === 0 ) {
-                       hasDuplicate = true;
-               }
-               return ret;
-       };
-} else if ( "sourceIndex" in document.documentElement ) {
-       sortOrder = function( a, b ) {
-               if ( !a.sourceIndex || !b.sourceIndex ) {
-                       if ( a == b ) {
-                               hasDuplicate = true;
-                       }
-                       return 0;
-               }
-
-               var ret = a.sourceIndex - b.sourceIndex;
-               if ( ret === 0 ) {
-                       hasDuplicate = true;
-               }
-               return ret;
-       };
-} else if ( document.createRange ) {
-       sortOrder = function( a, b ) {
-               if ( !a.ownerDocument || !b.ownerDocument ) {
-                       if ( a == b ) {
-                               hasDuplicate = true;
-                       }
-                       return 0;
-               }
-
-               var aRange = a.ownerDocument.createRange(), bRange = b.ownerDocument.createRange();
-               aRange.setStart(a, 0);
-               aRange.setEnd(a, 0);
-               bRange.setStart(b, 0);
-               bRange.setEnd(b, 0);
-               var ret = aRange.compareBoundaryPoints(Range.START_TO_END, bRange);
-               if ( ret === 0 ) {
-                       hasDuplicate = true;
-               }
-               return ret;
-       };
-}
-
-(function(){
-       var form = document.createElement("div"),
-               id = "script" + (new Date).getTime();
-       form.innerHTML = "<a name='" + id + "'/>";
-
-       var root = document.documentElement;
-       root.insertBefore( form, root.firstChild );
-
-       if ( !!document.getElementById( id ) ) {
-               Expr.find.ID = function(match, context, isXML){
-                       if ( typeof context.getElementById !== "undefined" && !isXML ) {
-                               var m = context.getElementById(match[1]);
-                               return m ? m.id === match[1] || typeof m.getAttributeNode !== "undefined" && m.getAttributeNode("id").nodeValue === match[1] ? [m] : undefined : [];
-                       }
-               };
-
-               Expr.filter.ID = function(elem, match){
-                       var node = typeof elem.getAttributeNode !== "undefined" && elem.getAttributeNode("id");
-                       return elem.nodeType === 1 && node && node.nodeValue === match;
-               };
-       }
-
-       root.removeChild( form );
-       root = form = null; // release memory in IE
-})();
-
-(function(){
-
-       var div = document.createElement("div");
-       div.appendChild( document.createComment("") );
-
-       if ( div.getElementsByTagName("*").length > 0 ) {
-               Expr.find.TAG = function(match, context){
-                       var results = context.getElementsByTagName(match[1]);
-
-                       if ( match[1] === "*" ) {
-                               var tmp = [];
-
-                               for ( var i = 0; results[i]; i++ ) {
-                                       if ( results[i].nodeType === 1 ) {
-                                               tmp.push( results[i] );
-                                       }
-                               }
-
-                               results = tmp;
-                       }
-
-                       return results;
-               };
-       }
-
-       div.innerHTML = "<a href='#'></a>";
-       if ( div.firstChild && typeof div.firstChild.getAttribute !== "undefined" &&
-                       div.firstChild.getAttribute("href") !== "#" ) {
-               Expr.attrHandle.href = function(elem){
-                       return elem.getAttribute("href", 2);
-               };
-       }
-
-       div = null; // release memory in IE
-})();
-
-if ( document.querySelectorAll ) (function(){
-       var oldSizzle = Sizzle, div = document.createElement("div");
-       div.innerHTML = "<p class='TEST'></p>";
-
-       if ( div.querySelectorAll && div.querySelectorAll(".TEST").length === 0 ) {
-               return;
-       }
-
-       Sizzle = function(query, context, extra, seed){
-               context = context || document;
-
-               if ( !seed && context.nodeType === 9 && !isXML(context) ) {
-                       try {
-                               return makeArray( context.querySelectorAll(query), extra );
-                       } catch(e){}
-               }
-
-               return oldSizzle(query, context, extra, seed);
-       };
-
-       for ( var prop in oldSizzle ) {
-               Sizzle[ prop ] = oldSizzle[ prop ];
-       }
-
-       div = null; // release memory in IE
-})();
-
-if ( document.getElementsByClassName && document.documentElement.getElementsByClassName ) (function(){
-       var div = document.createElement("div");
-       div.innerHTML = "<div class='test e'></div><div class='test'></div>";
-
-       if ( div.getElementsByClassName("e").length === 0 )
-               return;
-
-       div.lastChild.className = "e";
-
-       if ( div.getElementsByClassName("e").length === 1 )
-               return;
-
-       Expr.order.splice(1, 0, "CLASS");
-       Expr.find.CLASS = function(match, context, isXML) {
-               if ( typeof context.getElementsByClassName !== "undefined" && !isXML ) {
-                       return context.getElementsByClassName(match[1]);
-               }
-       };
-
-       div = null; // release memory in IE
-})();
-
-function dirNodeCheck( dir, cur, doneName, checkSet, nodeCheck, isXML ) {
-       var sibDir = dir == "previousSibling" && !isXML;
-       for ( var i = 0, l = checkSet.length; i < l; i++ ) {
-               var elem = checkSet[i];
-               if ( elem ) {
-                       if ( sibDir && elem.nodeType === 1 ){
-                               elem.sizcache = doneName;
-                               elem.sizset = i;
-                       }
-                       elem = elem[dir];
-                       var match = false;
-
-                       while ( elem ) {
-                               if ( elem.sizcache === doneName ) {
-                                       match = checkSet[elem.sizset];
-                                       break;
-                               }
-
-                               if ( elem.nodeType === 1 && !isXML ){
-                                       elem.sizcache = doneName;
-                                       elem.sizset = i;
-                               }
-
-                               if ( elem.nodeName === cur ) {
-                                       match = elem;
-                                       break;
-                               }
-
-                               elem = elem[dir];
-                       }
-
-                       checkSet[i] = match;
-               }
-       }
-}
-
-function dirCheck( dir, cur, doneName, checkSet, nodeCheck, isXML ) {
-       var sibDir = dir == "previousSibling" && !isXML;
-       for ( var i = 0, l = checkSet.length; i < l; i++ ) {
-               var elem = checkSet[i];
-               if ( elem ) {
-                       if ( sibDir && elem.nodeType === 1 ) {
-                               elem.sizcache = doneName;
-                               elem.sizset = i;
-                       }
-                       elem = elem[dir];
-                       var match = false;
-
-                       while ( elem ) {
-                               if ( elem.sizcache === doneName ) {
-                                       match = checkSet[elem.sizset];
-                                       break;
-                               }
-
-                               if ( elem.nodeType === 1 ) {
-                                       if ( !isXML ) {
-                                               elem.sizcache = doneName;
-                                               elem.sizset = i;
-                                       }
-                                       if ( typeof cur !== "string" ) {
-                                               if ( elem === cur ) {
-                                                       match = true;
-                                                       break;
-                                               }
-
-                                       } else if ( Sizzle.filter( cur, [elem] ).length > 0 ) {
-                                               match = elem;
-                                               break;
-                                       }
-                               }
-
-                               elem = elem[dir];
-                       }
-
-                       checkSet[i] = match;
-               }
-       }
-}
-
-var contains = document.compareDocumentPosition ?  function(a, b){
-       return a.compareDocumentPosition(b) & 16;
-} : function(a, b){
-       return a !== b && (a.contains ? a.contains(b) : true);
-};
-
-var isXML = function(elem){
-       return elem.nodeType === 9 && elem.documentElement.nodeName !== "HTML" ||
-               !!elem.ownerDocument && elem.ownerDocument.documentElement.nodeName !== "HTML";
-};
-
-var posProcess = function(selector, context){
-       var tmpSet = [], later = "", match,
-               root = context.nodeType ? [context] : context;
-
-       while ( (match = Expr.match.PSEUDO.exec( selector )) ) {
-               later += match[0];
-               selector = selector.replace( Expr.match.PSEUDO, "" );
-       }
-
-       selector = Expr.relative[selector] ? selector + "*" : selector;
-
-       for ( var i = 0, l = root.length; i < l; i++ ) {
-               Sizzle( selector, root[i], tmpSet );
-       }
-
-       return Sizzle.filter( later, tmpSet );
-};
-
-
-window.Sizzle = Sizzle;
-
-})();
-
-;(function(engine) {
-  var extendElements = Prototype.Selector.extendElements;
-
-  function select(selector, scope) {
-    return extendElements(engine(selector, scope || document));
-  }
-
-  function match(element, selector) {
-    return engine.matches(selector, [element]).length == 1;
-  }
-
-  Prototype.Selector.engine = engine;
-  Prototype.Selector.select = select;
-  Prototype.Selector.match = match;
-})(Sizzle);
-
-window.Sizzle = Prototype._original_property;
-delete Prototype._original_property;
-
-var Form = {
-  reset: function(form) {
-    form = $(form);
-    form.reset();
-    return form;
-  },
-
-  serializeElements: function(elements, options) {
-    if (typeof options != 'object') options = { hash: !!options };
-    else if (Object.isUndefined(options.hash)) options.hash = true;
-    var key, value, submitted = false, submit = options.submit, accumulator, initial;
-
-    if (options.hash) {
-      initial = {};
-      accumulator = function(result, key, value) {
-        if (key in result) {
-          if (!Object.isArray(result[key])) result[key] = [result[key]];
-          result[key].push(value);
-        } else result[key] = value;
-        return result;
-      };
-    } else {
-      initial = '';
-      accumulator = function(result, key, value) {
-        return result + (result ? '&' : '') + encodeURIComponent(key) + '=' + encodeURIComponent(value);
-      }
-    }
-
-    return elements.inject(initial, function(result, element) {
-      if (!element.disabled && element.name) {
-        key = element.name; value = $(element).getValue();
-        if (value != null && element.type != 'file' && (element.type != 'submit' || (!submitted &&
-            submit !== false && (!submit || key == submit) && (submitted = true)))) {
-          result = accumulator(result, key, value);
-        }
-      }
-      return result;
-    });
-  }
-};
-
-Form.Methods = {
-  serialize: function(form, options) {
-    return Form.serializeElements(Form.getElements(form), options);
-  },
-
-  getElements: function(form) {
-    var elements = $(form).getElementsByTagName('*'),
-        element,
-        arr = [ ],
-        serializers = Form.Element.Serializers;
-    for (var i = 0; element = elements[i]; i++) {
-      arr.push(element);
-    }
-    return arr.inject([], function(elements, child) {
-      if (serializers[child.tagName.toLowerCase()])
-        elements.push(Element.extend(child));
-      return elements;
-    })
-  },
-
-  getInputs: function(form, typeName, name) {
-    form = $(form);
-    var inputs = form.getElementsByTagName('input');
-
-    if (!typeName && !name) return $A(inputs).map(Element.extend);
-
-    for (var i = 0, matchingInputs = [], length = inputs.length; i < length; i++) {
-      var input = inputs[i];
-      if ((typeName && input.type != typeName) || (name && input.name != name))
-        continue;
-      matchingInputs.push(Element.extend(input));
-    }
-
-    return matchingInputs;
-  },
-
-  disable: function(form) {
-    form = $(form);
-    Form.getElements(form).invoke('disable');
-    return form;
-  },
-
-  enable: function(form) {
-    form = $(form);
-    Form.getElements(form).invoke('enable');
-    return form;
-  },
-
-  findFirstElement: function(form) {
-    var elements = $(form).getElements().findAll(function(element) {
-      return 'hidden' != element.type && !element.disabled;
-    });
-    var firstByIndex = elements.findAll(function(element) {
-      return element.hasAttribute('tabIndex') && element.tabIndex >= 0;
-    }).sortBy(function(element) { return element.tabIndex }).first();
-
-    return firstByIndex ? firstByIndex : elements.find(function(element) {
-      return /^(?:input|select|textarea)$/i.test(element.tagName);
-    });
-  },
-
-  focusFirstElement: function(form) {
-    form = $(form);
-    var element = form.findFirstElement();
-    if (element) element.activate();
-    return form;
-  },
-
-  request: function(form, options) {
-    form = $(form), options = Object.clone(options || { });
-
-    var params = options.parameters, action = form.readAttribute('action') || '';
-    if (action.blank()) action = window.location.href;
-    options.parameters = form.serialize(true);
-
-    if (params) {
-      if (Object.isString(params)) params = params.toQueryParams();
-      Object.extend(options.parameters, params);
-    }
-
-    if (form.hasAttribute('method') && !options.method)
-      options.method = form.method;
-
-    return new Ajax.Request(action, options);
-  }
-};
-
-/*--------------------------------------------------------------------------*/
-
-
-Form.Element = {
-  focus: function(element) {
-    $(element).focus();
-    return element;
-  },
-
-  select: function(element) {
-    $(element).select();
-    return element;
-  }
-};
-
-Form.Element.Methods = {
-
-  serialize: function(element) {
-    element = $(element);
-    if (!element.disabled && element.name) {
-      var value = element.getValue();
-      if (value != undefined) {
-        var pair = { };
-        pair[element.name] = value;
-        return Object.toQueryString(pair);
-      }
-    }
-    return '';
-  },
-
-  getValue: function(element) {
-    element = $(element);
-    var method = element.tagName.toLowerCase();
-    return Form.Element.Serializers[method](element);
-  },
-
-  setValue: function(element, value) {
-    element = $(element);
-    var method = element.tagName.toLowerCase();
-    Form.Element.Serializers[method](element, value);
-    return element;
-  },
-
-  clear: function(element) {
-    $(element).value = '';
-    return element;
-  },
-
-  present: function(element) {
-    return $(element).value != '';
-  },
-
-  activate: function(element) {
-    element = $(element);
-    try {
-      element.focus();
-      if (element.select && (element.tagName.toLowerCase() != 'input' ||
-          !(/^(?:button|reset|submit)$/i.test(element.type))))
-        element.select();
-    } catch (e) { }
-    return element;
-  },
-
-  disable: function(element) {
-    element = $(element);
-    element.disabled = true;
-    return element;
-  },
-
-  enable: function(element) {
-    element = $(element);
-    element.disabled = false;
-    return element;
-  }
-};
-
-/*--------------------------------------------------------------------------*/
-
-var Field = Form.Element;
-
-var $F = Form.Element.Methods.getValue;
-
-/*--------------------------------------------------------------------------*/
-
-Form.Element.Serializers = (function() {
-  function input(element, value) {
-    switch (element.type.toLowerCase()) {
-      case 'checkbox':
-      case 'radio':
-        return inputSelector(element, value);
-      default:
-        return valueSelector(element, value);
-    }
-  }
-
-  function inputSelector(element, value) {
-    if (Object.isUndefined(value))
-      return element.checked ? element.value : null;
-    else element.checked = !!value;
-  }
-
-  function valueSelector(element, value) {
-    if (Object.isUndefined(value)) return element.value;
-    else element.value = value;
-  }
-
-  function select(element, value) {
-    if (Object.isUndefined(value))
-      return (element.type === 'select-one' ? selectOne : selectMany)(element);
-
-    var opt, currentValue, single = !Object.isArray(value);
-    for (var i = 0, length = element.length; i < length; i++) {
-      opt = element.options[i];
-      currentValue = this.optionValue(opt);
-      if (single) {
-        if (currentValue == value) {
-          opt.selected = true;
-          return;
-        }
-      }
-      else opt.selected = value.include(currentValue);
-    }
-  }
-
-  function selectOne(element) {
-    var index = element.selectedIndex;
-    return index >= 0 ? optionValue(element.options[index]) : null;
-  }
-
-  function selectMany(element) {
-    var values, length = element.length;
-    if (!length) return null;
-
-    for (var i = 0, values = []; i < length; i++) {
-      var opt = element.options[i];
-      if (opt.selected) values.push(optionValue(opt));
-    }
-    return values;
-  }
-
-  function optionValue(opt) {
-    return Element.hasAttribute(opt, 'value') ? opt.value : opt.text;
-  }
-
-  return {
-    input:         input,
-    inputSelector: inputSelector,
-    textarea:      valueSelector,
-    select:        select,
-    selectOne:     selectOne,
-    selectMany:    selectMany,
-    optionValue:   optionValue,
-    button:        valueSelector
-  };
-})();
-
-/*--------------------------------------------------------------------------*/
-
-
-Abstract.TimedObserver = Class.create(PeriodicalExecuter, {
-  initialize: function($super, element, frequency, callback) {
-    $super(callback, frequency);
-    this.element   = $(element);
-    this.lastValue = this.getValue();
-  },
-
-  execute: function() {
-    var value = this.getValue();
-    if (Object.isString(this.lastValue) && Object.isString(value) ?
-        this.lastValue != value : String(this.lastValue) != String(value)) {
-      this.callback(this.element, value);
-      this.lastValue = value;
-    }
-  }
-});
-
-Form.Element.Observer = Class.create(Abstract.TimedObserver, {
-  getValue: function() {
-    return Form.Element.getValue(this.element);
-  }
-});
-
-Form.Observer = Class.create(Abstract.TimedObserver, {
-  getValue: function() {
-    return Form.serialize(this.element);
-  }
-});
-
-/*--------------------------------------------------------------------------*/
-
-Abstract.EventObserver = Class.create({
-  initialize: function(element, callback) {
-    this.element  = $(element);
-    this.callback = callback;
-
-    this.lastValue = this.getValue();
-    if (this.element.tagName.toLowerCase() == 'form')
-      this.registerFormCallbacks();
-    else
-      this.registerCallback(this.element);
-  },
-
-  onElementEvent: function() {
-    var value = this.getValue();
-    if (this.lastValue != value) {
-      this.callback(this.element, value);
-      this.lastValue = value;
-    }
-  },
-
-  registerFormCallbacks: function() {
-    Form.getElements(this.element).each(this.registerCallback, this);
-  },
-
-  registerCallback: function(element) {
-    if (element.type) {
-      switch (element.type.toLowerCase()) {
-        case 'checkbox':
-        case 'radio':
-          Event.observe(element, 'click', this.onElementEvent.bind(this));
-          break;
-        default:
-          Event.observe(element, 'change', this.onElementEvent.bind(this));
-          break;
-      }
-    }
-  }
-});
-
-Form.Element.EventObserver = Class.create(Abstract.EventObserver, {
-  getValue: function() {
-    return Form.Element.getValue(this.element);
-  }
-});
-
-Form.EventObserver = Class.create(Abstract.EventObserver, {
-  getValue: function() {
-    return Form.serialize(this.element);
-  }
-});
-(function() {
-
-  var Event = {
-    KEY_BACKSPACE: 8,
-    KEY_TAB:       9,
-    KEY_RETURN:   13,
-    KEY_ESC:      27,
-    KEY_LEFT:     37,
-    KEY_UP:       38,
-    KEY_RIGHT:    39,
-    KEY_DOWN:     40,
-    KEY_DELETE:   46,
-    KEY_HOME:     36,
-    KEY_END:      35,
-    KEY_PAGEUP:   33,
-    KEY_PAGEDOWN: 34,
-    KEY_INSERT:   45,
-
-    cache: {}
-  };
-
-  var docEl = document.documentElement;
-  var MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED = 'onmouseenter' in docEl
-    && 'onmouseleave' in docEl;
-
-
-
-  var isIELegacyEvent = function(event) { return false; };
-
-  if (window.attachEvent) {
-    if (window.addEventListener) {
-      isIELegacyEvent = function(event) {
-        return !(event instanceof window.Event);
-      };
-    } else {
-      isIELegacyEvent = function(event) { return true; };
-    }
-  }
-
-  var _isButton;
-
-  function _isButtonForDOMEvents(event, code) {
-    return event.which ? (event.which === code + 1) : (event.button === code);
-  }
-
-  var legacyButtonMap = { 0: 1, 1: 4, 2: 2 };
-  function _isButtonForLegacyEvents(event, code) {
-    return event.button === legacyButtonMap[code];
-  }
-
-  function _isButtonForWebKit(event, code) {
-    switch (code) {
-      case 0: return event.which == 1 && !event.metaKey;
-      case 1: return event.which == 2 || (event.which == 1 && event.metaKey);
-      case 2: return event.which == 3;
-      default: return false;
-    }
-  }
-
-  if (window.attachEvent) {
-    if (!window.addEventListener) {
-      _isButton = _isButtonForLegacyEvents;
-    } else {
-      _isButton = function(event, code) {
-        return isIELegacyEvent(event) ? _isButtonForLegacyEvents(event, code) :
-         _isButtonForDOMEvents(event, code);
-      }
-    }
-  } else if (Prototype.Browser.WebKit) {
-    _isButton = _isButtonForWebKit;
-  } else {
-    _isButton = _isButtonForDOMEvents;
-  }
-
-  function isLeftClick(event)   { return _isButton(event, 0) }
-
-  function isMiddleClick(event) { return _isButton(event, 1) }
-
-  function isRightClick(event)  { return _isButton(event, 2) }
-
-  function element(event) {
-    event = Event.extend(event);
-
-    var node = event.target, type = event.type,
-     currentTarget = event.currentTarget;
-
-    if (currentTarget && currentTarget.tagName) {
-      if (type === 'load' || type === 'error' ||
-        (type === 'click' && currentTarget.tagName.toLowerCase() === 'input'
-          && currentTarget.type === 'radio'))
-            node = currentTarget;
-    }
-
-    if (node.nodeType == Node.TEXT_NODE)
-      node = node.parentNode;
-
-    return Element.extend(node);
-  }
-
-  function findElement(event, expression) {
-    var element = Event.element(event);
-
-    if (!expression) return element;
-    while (element) {
-      if (Object.isElement(element) && Prototype.Selector.match(element, expression)) {
-        return Element.extend(element);
-      }
-      element = element.parentNode;
-    }
-  }
-
-  function pointer(event) {
-    return { x: pointerX(event), y: pointerY(event) };
-  }
-
-  function pointerX(event) {
-    var docElement = document.documentElement,
-     body = document.body || { scrollLeft: 0 };
-
-    return event.pageX || (event.clientX +
-      (docElement.scrollLeft || body.scrollLeft) -
-      (docElement.clientLeft || 0));
-  }
-
-  function pointerY(event) {
-    var docElement = document.documentElement,
-     body = document.body || { scrollTop: 0 };
-
-    return  event.pageY || (event.clientY +
-       (docElement.scrollTop || body.scrollTop) -
-       (docElement.clientTop || 0));
-  }
-
-
-  function stop(event) {
-    Event.extend(event);
-    event.preventDefault();
-    event.stopPropagation();
-
-    event.stopped = true;
-  }
-
-
-  Event.Methods = {
-    isLeftClick:   isLeftClick,
-    isMiddleClick: isMiddleClick,
-    isRightClick:  isRightClick,
-
-    element:     element,
-    findElement: findElement,
-
-    pointer:  pointer,
-    pointerX: pointerX,
-    pointerY: pointerY,
-
-    stop: stop
-  };
-
-  var methods = Object.keys(Event.Methods).inject({ }, function(m, name) {
-    m[name] = Event.Methods[name].methodize();
-    return m;
-  });
-
-  if (window.attachEvent) {
-    function _relatedTarget(event) {
-      var element;
-      switch (event.type) {
-        case 'mouseover':
-        case 'mouseenter':
-          element = event.fromElement;
-          break;
-        case 'mouseout':
-        case 'mouseleave':
-          element = event.toElement;
-          break;
-        default:
-          return null;
-      }
-      return Element.extend(element);
-    }
-
-    var additionalMethods = {
-      stopPropagation: function() { this.cancelBubble = true },
-      preventDefault:  function() { this.returnValue = false },
-      inspect: function() { return '[object Event]' }
-    };
-
-    Event.extend = function(event, element) {
-      if (!event) return false;
-
-      if (!isIELegacyEvent(event)) return event;
-
-      if (event._extendedByPrototype) return event;
-      event._extendedByPrototype = Prototype.emptyFunction;
-
-      var pointer = Event.pointer(event);
-
-      Object.extend(event, {
-        target: event.srcElement || element,
-        relatedTarget: _relatedTarget(event),
-        pageX:  pointer.x,
-        pageY:  pointer.y
-      });
-
-      Object.extend(event, methods);
-      Object.extend(event, additionalMethods);
-
-      return event;
-    };
-  } else {
-    Event.extend = Prototype.K;
-  }
-
-  if (window.addEventListener) {
-    Event.prototype = window.Event.prototype || document.createEvent('HTMLEvents').__proto__;
-    Object.extend(Event.prototype, methods);
-  }
-
-  function _createResponder(element, eventName, handler) {
-    var registry = Element.retrieve(element, 'prototype_event_registry');
-
-    if (Object.isUndefined(registry)) {
-      CACHE.push(element);
-      registry = Element.retrieve(element, 'prototype_event_registry', $H());
-    }
-
-    var respondersForEvent = registry.get(eventName);
-    if (Object.isUndefined(respondersForEvent)) {
-      respondersForEvent = [];
-      registry.set(eventName, respondersForEvent);
-    }
-
-    if (respondersForEvent.pluck('handler').include(handler)) return false;
-
-    var responder;
-    if (eventName.include(":")) {
-      responder = function(event) {
-        if (Object.isUndefined(event.eventName))
-          return false;
-
-        if (event.eventName !== eventName)
-          return false;
-
-        Event.extend(event, element);
-        handler.call(element, event);
-      };
-    } else {
-      if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED &&
-       (eventName === "mouseenter" || eventName === "mouseleave")) {
-        if (eventName === "mouseenter" || eventName === "mouseleave") {
-          responder = function(event) {
-            Event.extend(event, element);
-
-            var parent = event.relatedTarget;
-            while (parent && parent !== element) {
-              try { parent = parent.parentNode; }
-              catch(e) { parent = element; }
-            }
-
-            if (parent === element) return;
-
-            handler.call(element, event);
-          };
-        }
-      } else {
-        responder = function(event) {
-          Event.extend(event, element);
-          handler.call(element, event);
-        };
-      }
-    }
-
-    responder.handler = handler;
-    respondersForEvent.push(responder);
-    return responder;
-  }
-
-  function _destroyCache() {
-    for (var i = 0, length = CACHE.length; i < length; i++) {
-      Event.stopObserving(CACHE[i]);
-      CACHE[i] = null;
-    }
-  }
-
-  var CACHE = [];
-
-  if (Prototype.Browser.IE)
-    window.attachEvent('onunload', _destroyCache);
-
-  if (Prototype.Browser.WebKit)
-    window.addEventListener('unload', Prototype.emptyFunction, false);
-
-
-  var _getDOMEventName = Prototype.K,
-      translations = { mouseenter: "mouseover", mouseleave: "mouseout" };
-
-  if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED) {
-    _getDOMEventName = function(eventName) {
-      return (translations[eventName] || eventName);
-    };
-  }
-
-  function observe(element, eventName, handler) {
-    element = $(element);
-
-    var responder = _createResponder(element, eventName, handler);
-
-    if (!responder) return element;
-
-    if (eventName.include(':')) {
-      if (element.addEventListener)
-        element.addEventListener("dataavailable", responder, false);
-      else {
-        element.attachEvent("ondataavailable", responder);
-        element.attachEvent("onlosecapture", responder);
-      }
-    } else {
-      var actualEventName = _getDOMEventName(eventName);
-
-      if (element.addEventListener)
-        element.addEventListener(actualEventName, responder, false);
-      else
-        element.attachEvent("on" + actualEventName, responder);
-    }
-
-    return element;
-  }
-
-  function stopObserving(element, eventName, handler) {
-    element = $(element);
-
-    var registry = Element.retrieve(element, 'prototype_event_registry');
-    if (!registry) return element;
-
-    if (!eventName) {
-      registry.each( function(pair) {
-        var eventName = pair.key;
-        stopObserving(element, eventName);
-      });
-      return element;
-    }
-
-    var responders = registry.get(eventName);
-    if (!responders) return element;
-
-    if (!handler) {
-      responders.each(function(r) {
-        stopObserving(element, eventName, r.handler);
-      });
-      return element;
-    }
-
-    var i = responders.length, responder;
-    while (i--) {
-      if (responders[i].handler === handler) {
-        responder = responders[i];
-        break;
-      }
-    }
-    if (!responder) return element;
-
-    if (eventName.include(':')) {
-      if (element.removeEventListener)
-        element.removeEventListener("dataavailable", responder, false);
-      else {
-        element.detachEvent("ondataavailable", responder);
-        element.detachEvent("onlosecapture", responder);
-      }
-    } else {
-      var actualEventName = _getDOMEventName(eventName);
-      if (element.removeEventListener)
-        element.removeEventListener(actualEventName, responder, false);
-      else
-        element.detachEvent('on' + actualEventName, responder);
-    }
-
-    registry.set(eventName, responders.without(responder));
-
-    return element;
-  }
-
-  function fire(element, eventName, memo, bubble) {
-    element = $(element);
-
-    if (Object.isUndefined(bubble))
-      bubble = true;
-
-    if (element == document && document.createEvent && !element.dispatchEvent)
-      element = document.documentElement;
-
-    var event;
-    if (document.createEvent) {
-      event = document.createEvent('HTMLEvents');
-      event.initEvent('dataavailable', bubble, true);
-    } else {
-      event = document.createEventObject();
-      event.eventType = bubble ? 'ondataavailable' : 'onlosecapture';
-    }
-
-    event.eventName = eventName;
-    event.memo = memo || { };
-
-    if (document.createEvent)
-      element.dispatchEvent(event);
-    else
-      element.fireEvent(event.eventType, event);
-
-    return Event.extend(event);
-  }
-
-  Event.Handler = Class.create({
-    initialize: function(element, eventName, selector, callback) {
-      this.element   = $(element);
-      this.eventName = eventName;
-      this.selector  = selector;
-      this.callback  = callback;
-      this.handler   = this.handleEvent.bind(this);
-    },
-
-    start: function() {
-      Event.observe(this.element, this.eventName, this.handler);
-      return this;
-    },
-
-    stop: function() {
-      Event.stopObserving(this.element, this.eventName, this.handler);
-      return this;
-    },
-
-    handleEvent: function(event) {
-      var element = Event.findElement(event, this.selector);
-      if (element) this.callback.call(this.element, event, element);
-    }
-  });
-
-  function on(element, eventName, selector, callback) {
-    element = $(element);
-    if (Object.isFunction(selector) && Object.isUndefined(callback)) {
-      callback = selector, selector = null;
-    }
-
-    return new Event.Handler(element, eventName, selector, callback).start();
-  }
-
-  Object.extend(Event, Event.Methods);
-
-  Object.extend(Event, {
-    fire:          fire,
-    observe:       observe,
-    stopObserving: stopObserving,
-    on:            on
-  });
-
-  Element.addMethods({
-    fire:          fire,
-
-    observe:       observe,
-
-    stopObserving: stopObserving,
-
-    on:            on
-  });
-
-  Object.extend(document, {
-    fire:          fire.methodize(),
-
-    observe:       observe.methodize(),
-
-    stopObserving: stopObserving.methodize(),
-
-    on:            on.methodize(),
-
-    loaded:        false
-  });
-
-  if (window.Event) Object.extend(window.Event, Event);
-  else window.Event = Event;
-})();
-
-(function() {
-  /* Support for the DOMContentLoaded event is based on work by Dan Webb,
-     Matthias Miller, Dean Edwards, John Resig, and Diego Perini. */
-
-  var timer;
-
-  function fireContentLoadedEvent() {
-    if (document.loaded) return;
-    if (timer) window.clearTimeout(timer);
-    document.loaded = true;
-    document.fire('dom:loaded');
-  }
-
-  function checkReadyState() {
-    if (document.readyState === 'complete') {
-      document.stopObserving('readystatechange', checkReadyState);
-      fireContentLoadedEvent();
-    }
-  }
-
-  function pollDoScroll() {
-    try { document.documentElement.doScroll('left'); }
-    catch(e) {
-      timer = pollDoScroll.defer();
-      return;
-    }
-    fireContentLoadedEvent();
-  }
-
-  if (document.addEventListener) {
-    document.addEventListener('DOMContentLoaded', fireContentLoadedEvent, false);
-  } else {
-    document.observe('readystatechange', checkReadyState);
-    if (window == top)
-      timer = pollDoScroll.defer();
-  }
-
-  Event.observe(window, 'load', fireContentLoadedEvent);
-})();
-
-Element.addMethods();
-
-/*------------------------------- DEPRECATED -------------------------------*/
-
-Hash.toQueryString = Object.toQueryString;
-
-var Toggle = { display: Element.toggle };
-
-Element.Methods.childOf = Element.Methods.descendantOf;
-
-var Insertion = {
-  Before: function(element, content) {
-    return Element.insert(element, {before:content});
-  },
-
-  Top: function(element, content) {
-    return Element.insert(element, {top:content});
-  },
-
-  Bottom: function(element, content) {
-    return Element.insert(element, {bottom:content});
-  },
-
-  After: function(element, content) {
-    return Element.insert(element, {after:content});
-  }
-};
-
-var $continue = new Error('"throw $continue" is deprecated, use "return" instead');
-
-var Position = {
-  includeScrollOffsets: false,
-
-  prepare: function() {
-    this.deltaX =  window.pageXOffset
-                || document.documentElement.scrollLeft
-                || document.body.scrollLeft
-                || 0;
-    this.deltaY =  window.pageYOffset
-                || document.documentElement.scrollTop
-                || document.body.scrollTop
-                || 0;
-  },
-
-  within: function(element, x, y) {
-    if (this.includeScrollOffsets)
-      return this.withinIncludingScrolloffsets(element, x, y);
-    this.xcomp = x;
-    this.ycomp = y;
-    this.offset = Element.cumulativeOffset(element);
-
-    return (y >= this.offset[1] &&
-            y <  this.offset[1] + element.offsetHeight &&
-            x >= this.offset[0] &&
-            x <  this.offset[0] + element.offsetWidth);
-  },
-
-  withinIncludingScrolloffsets: function(element, x, y) {
-    var offsetcache = Element.cumulativeScrollOffset(element);
-
-    this.xcomp = x + offsetcache[0] - this.deltaX;
-    this.ycomp = y + offsetcache[1] - this.deltaY;
-    this.offset = Element.cumulativeOffset(element);
-
-    return (this.ycomp >= this.offset[1] &&
-            this.ycomp <  this.offset[1] + element.offsetHeight &&
-            this.xcomp >= this.offset[0] &&
-            this.xcomp <  this.offset[0] + element.offsetWidth);
-  },
-
-  overlap: function(mode, element) {
-    if (!mode) return 0;
-    if (mode == 'vertical')
-      return ((this.offset[1] + element.offsetHeight) - this.ycomp) /
-        element.offsetHeight;
-    if (mode == 'horizontal')
-      return ((this.offset[0] + element.offsetWidth) - this.xcomp) /
-        element.offsetWidth;
-  },
-
-
-  cumulativeOffset: Element.Methods.cumulativeOffset,
-
-  positionedOffset: Element.Methods.positionedOffset,
-
-  absolutize: function(element) {
-    Position.prepare();
-    return Element.absolutize(element);
-  },
-
-  relativize: function(element) {
-    Position.prepare();
-    return Element.relativize(element);
-  },
-
-  realOffset: Element.Methods.cumulativeScrollOffset,
-
-  offsetParent: Element.Methods.getOffsetParent,
-
-  page: Element.Methods.viewportOffset,
-
-  clone: function(source, target, options) {
-    options = options || { };
-    return Element.clonePosition(target, source, options);
-  }
-};
-
-/*--------------------------------------------------------------------------*/
-
-if (!document.getElementsByClassName) document.getElementsByClassName = function(instanceMethods){
-  function iter(name) {
-    return name.blank() ? null : "[contains(concat(' ', @class, ' '), ' " + name + " ')]";
-  }
-
-  instanceMethods.getElementsByClassName = Prototype.BrowserFeatures.XPath ?
-  function(element, className) {
-    className = className.toString().strip();
-    var cond = /\s/.test(className) ? $w(className).map(iter).join('') : iter(className);
-    return cond ? document._getElementsByXPath('.//*' + cond, element) : [];
-  } : function(element, className) {
-    className = className.toString().strip();
-    var elements = [], classNames = (/\s/.test(className) ? $w(className) : null);
-    if (!classNames && !className) return elements;
-
-    var nodes = $(element).getElementsByTagName('*');
-    className = ' ' + className + ' ';
-
-    for (var i = 0, child, cn; child = nodes[i]; i++) {
-      if (child.className && (cn = ' ' + child.className + ' ') && (cn.include(className) ||
-          (classNames && classNames.all(function(name) {
-            return !name.toString().blank() && cn.include(' ' + name + ' ');
-          }))))
-        elements.push(Element.extend(child));
-    }
-    return elements;
-  };
-
-  return function(className, parentElement) {
-    return $(parentElement || document.body).getElementsByClassName(className);
-  };
-}(Element.Methods);
-
-/*--------------------------------------------------------------------------*/
-
-Element.ClassNames = Class.create();
-Element.ClassNames.prototype = {
-  initialize: function(element) {
-    this.element = $(element);
-  },
-
-  _each: function(iterator) {
-    this.element.className.split(/\s+/).select(function(name) {
-      return name.length > 0;
-    })._each(iterator);
-  },
-
-  set: function(className) {
-    this.element.className = className;
-  },
-
-  add: function(classNameToAdd) {
-    if (this.include(classNameToAdd)) return;
-    this.set($A(this).concat(classNameToAdd).join(' '));
-  },
-
-  remove: function(classNameToRemove) {
-    if (!this.include(classNameToRemove)) return;
-    this.set($A(this).without(classNameToRemove).join(' '));
-  },
-
-  toString: function() {
-    return $A(this).join(' ');
-  }
-};
-
-Object.extend(Element.ClassNames.prototype, Enumerable);
-
-/*--------------------------------------------------------------------------*/
-
-(function() {
-  window.Selector = Class.create({
-    initialize: function(expression) {
-      this.expression = expression.strip();
-    },
-
-    findElements: function(rootElement) {
-      return Prototype.Selector.select(this.expression, rootElement);
-    },
-
-    match: function(element) {
-      return Prototype.Selector.match(element, this.expression);
-    },
-
-    toString: function() {
-      return this.expression;
-    },
-
-    inspect: function() {
-      return "#<Selector: " + this.expression + ">";
-    }
-  });
-
-  Object.extend(Selector, {
-    matchElements: function(elements, expression) {
-      var match = Prototype.Selector.match,
-          results = [];
-
-      for (var i = 0, length = elements.length; i < length; i++) {
-        var element = elements[i];
-        if (match(element, expression)) {
-          results.push(Element.extend(element));
-        }
-      }
-      return results;
-    },
-
-    findElement: function(elements, expression, index) {
-      index = index || 0;
-      var matchIndex = 0, element;
-      for (var i = 0, length = elements.length; i < length; i++) {
-        element = elements[i];
-        if (Prototype.Selector.match(element, expression) && index === matchIndex++) {
-          return Element.extend(element);
-        }
-      }
-    },
-
-    findChildElements: function(element, expressions) {
-      var selector = expressions.toArray().join(', ');
-      return Prototype.Selector.select(selector, element || document);
-    }
-  });
-})();
diff --git a/ricoClient/js/minsrc/ricoCalendar.js b/ricoClient/js/minsrc/ricoCalendar.js
deleted file mode 100644 (file)
index 7978ca1..0000000
+++ /dev/null
@@ -1,549 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-//  Inspired by code originally written by Tan Ling Wee on 2 Dec 2001
-
-Rico.CalendarControl = function(id,options) {
-  this.initialize(id,options);
-};
-
-Rico.CalendarControl.prototype = {
-/**
- * @class Implements a pop-up Gregorian calendar.
- * Dates of adoption of the Gregorian calendar vary by country - accurate as a US & British calendar from 14 Sept 1752 to present.
- * Mark special dates with calls to addHoliday()
- * @extends Rico.Popup
- * @constructs
- * @param id unique identifier
- * @param options object may contain any of the following:<dl>
- *   <dt>startAt       </dt><dd> week starts with 0=sunday, 1=monday? default=0</dd>
- *   <dt>showWeekNumber</dt><dd> show week number in first column? default=0</dd>
- *   <dt>showToday     </dt><dd> show "Today is..." in footer? default=1</dd>
- *   <dt>dateFmt       </dt><dd> date format for return value (one of values accepted by {@link Date#formatDate}), default=ISO8601</dd>
- *   <dt>minDate       </dt><dd> earliest selectable date? default=today-50 years</dd>
- *   <dt>maxDate       </dt><dd> last selectable date? default=today+50 years</dd>
- *</dl>
- */
-  initialize: function(id,options) {
-    this.id=id;
-    var today=new Date();
-    Rico.extend(this, new Rico.Popup());
-    Rico.extend(this.options, {
-      ignoreClicks:true,
-      startAt : 0,
-      showWeekNumber : 0,
-      showToday : 1,
-      dateFmt : 'ISO8601',
-      minDate : new Date(today.getFullYear()-50,0,1),
-      maxDate : new Date(today.getFullYear()+50,11,31)
-    });
-    Rico.extend(this.options, options || {});
-    /**
-     * alias for closePopup
-     * @function
-     */
-    this.close=this.closePopup;
-    this.bPageLoaded=false;
-    this.img=[];
-    this.Holidays={};
-    this.weekString=Rico.getPhraseById("calWeekHdg");
-    this.re=/^\s*(\w+)(\W)(\w+)(\W)(\w+)/i;
-    this.setDateFmt(this.options.dateFmt);
-  },
-
-
-  setDateFmt: function(fmt) {
-    this.dateFmt=(fmt=='rico') ? Rico.dateFmt : fmt;
-    Rico.log(this.id+' date format set to '+this.dateFmt);
-    this.dateParts={};
-    if (this.re.exec(this.dateFmt)) {
-      this.dateParts[RegExp.$1]=0;
-      this.dateParts[RegExp.$3]=1;
-      this.dateParts[RegExp.$5]=2;
-    }
-  },
-  
-/**
- * Call before displaying calendar to highlight special days
- * @param d day (1-31)
- * @param m month (1-12)
- * @param y year (0 implies a repeating holiday)
- * @param desc description
- * @param bgColor background color for cell displaying this day (CSS value, defaults to '#DDF')
- * @param txtColor text color for cell displaying this day (CSS value), if not specified it is displayed with the same color as other days
- */
-  addHoliday : function(d, m, y, desc, bgColor, txtColor) {
-    this.Holidays[this.holidayKey(y,m-1,d)]={desc:desc, txtColor:txtColor, bgColor:bgColor || '#DDF'};
-  },
-  
-/** @private */
-  holidayKey : function(y,m,d) {
-    return 'h'+Rico.zFill(y,4)+Rico.zFill(m,2)+Rico.zFill(d,2);
-  },
-
-  atLoad : function() {
-    Rico.log('Calendar#atLoad: '+this.id);
-    this.createContainer();
-    this.container.id=this.id;
-    Rico.addClass(this.container, Rico.theme.calendar || 'ricoCalContainer');
-
-    this.maintab=document.createElement("table");
-    this.maintab.cellSpacing=2;
-    this.maintab.cellPadding=0;
-    this.maintab.border=0;
-    this.maintab.style.borderCollapse='separate';
-    this.maintab.className='ricoCalTab';
-    if (Rico.theme.calendarTable) Rico.addClass(this.maintab,Rico.theme.calendarTable)
-    this.tbody=Rico.getTBody(this.maintab);
-
-    var r,c,d,i,j,img,dow,a,s,tab;
-    this.colStart=this.options.showWeekNumber ? 1 : 0;
-    for (i=0; i<7; i++) {
-      r=this.tbody.insertRow(-1);
-      r.className='row'+i;
-      for (c=0; c<7+this.colStart; c++) {
-        r.insertCell(-1);
-      }
-    }
-    r=this.tbody.rows[0];
-    r.className='ricoCalDayNames';
-    if (this.options.showWeekNumber) {
-      r.cells[0].innerHTML=this.weekString;
-      for (i=0; i<7; i++) {
-        this.tbody.rows[i].cells[0].className='ricoCalWeekNum';
-      }
-    }
-    this.styles=[];
-    for (i=0; i<7; i++) {
-      dow=(i+this.options.startAt) % 7;
-      r.cells[i+this.colStart].innerHTML=Rico.dayAbbr(dow);
-      this.styles[i]='ricoCal'+dow;
-    }
-    
-    // table header (navigation controls)
-    this.thead=this.maintab.createTHead();
-    r=this.thead.insertRow(-1);
-    c=r.appendChild(document.createElement("th"));
-    c.colSpan=7+this.colStart;
-    d=c.appendChild(document.createElement("div"));
-    //d.style.padding='3px';
-    d.className=Rico.theme.calendarHeading || 'RicoCalHeading';
-    
-    d.appendChild(this._createTitleSection('Month'));
-    d.appendChild(this._createTitleSection('Year'));
-    new Rico.HoverSet(d.getElementsByTagName('a'));
-    new Rico.HoverSet(this.tbody.getElementsByTagName('td'),{ hoverNodes: function(e) { return e.innerHTML.match(/^\d+$/) ? [e] : []; } });
-    d.appendChild(Rico.closeButton(Rico.eventHandle(this,'close')));
-
-    // table footer (today)
-    if (this.options.showToday) {
-      this.tfoot=this.maintab.createTFoot();
-      r=this.tfoot.insertRow(-1);
-      this.todayCell=r.insertCell(-1);
-      this.todayCell.colSpan=7+this.colStart;
-      if (Rico.theme.calendarFooter) Rico.addClass(this.todayCell,Rico.theme.calendarFooter);
-      Rico.eventBind(this.todayCell,"click", Rico.eventHandle(this,'selectNow'), false);
-    }
-    this.content.appendChild(this.maintab);
-    var ie6=Rico.isIE && Rico.ieVersion < 7;
-    var selectOptions={shadow: !ie6};
-    
-    // month selector
-    this.monthPopup=new Rico.Popup(document.createElement("div"),selectOptions);
-    this.monthPopup.closePopup();
-    tab=document.createElement("table");
-    tab.className='ricoCalMenu';
-    if (Rico.theme.calendarPopdown) Rico.addClass(tab,Rico.theme.calendarPopdown);
-    tab.cellPadding=2;
-    tab.cellSpacing=0;
-    tab.border=0;
-    tab.style.borderCollapse='separate';
-    tab.style.margin='0px';
-    for (i=0; i<4; i++) {
-      r=tab.insertRow(-1);
-      for (j=0; j<3; j++) {
-        c=r.insertCell(-1);
-        a=document.createElement("a");
-        a.innerHTML=Rico.monthAbbr(i*3+j);
-        a.name=i*3+j;
-        if (Rico.theme.calendarDay) Rico.addClass(a,Rico.theme.calendarDay);
-        c.appendChild(a);
-        Rico.eventBind(a,"click", Rico.eventHandle(this,'selectMonth'), false);
-      }
-    }
-    new Rico.HoverSet(tab.getElementsByTagName('a'));
-    this.monthPopup.content.appendChild(tab);
-    this.content.appendChild(this.monthPopup.container);
-    
-    // year selector
-    this.yearPopup=new Rico.Popup(document.createElement("div"),selectOptions);
-    this.yearPopup.closePopup();
-    this.yearPopup.content.className='ricoCalYearPrompt';
-    if (Rico.theme.calendarPopdown) Rico.addClass(this.yearPopup.content,Rico.theme.calendarPopdown);
-    var tab=document.createElement("table");
-    tab.cellPadding=2;
-    tab.cellSpacing=0;
-    tab.border=0;
-    tab.style.borderCollapse='separate';
-    tab.style.margin='0px';
-    r=tab.insertRow(-1);
-    this.yearLabel=r.insertCell(-1);
-    this.yearLabel.colSpan=3;
-    r=tab.insertRow(-1);
-    c=r.insertCell(-1);
-    this.yearInput=c.appendChild(document.createElement("input"));
-    this.yearInput.maxlength=4;
-    this.yearInput.size=4;
-    Rico.eventBind(this.yearInput,"keypress", Rico.eventHandle(this,'yearKey'), false);
-    c=r.insertCell(-1);
-    var a=Rico.floatButton('Checkmark', Rico.eventHandle(this,'processPopUpYear'));
-    Rico.setStyle(a.firstChild,{ margin:"0px", padding:"0px", border:"none" });
-    c.appendChild(a);
-    c=r.insertCell(-1);
-    a=Rico.floatButton('Cancel', Rico.eventHandle(this,'popDownYear'));
-    Rico.setStyle(a.firstChild,{ margin:"0px", padding:"0px", border:"none" });
-    c.appendChild(a);
-    this.yearPopup.content.appendChild(tab);
-    this.content.appendChild(this.yearPopup.container);
-
-    //this.yearLabel.className='ricoCalYearPromptText';
-
-    // fix anchors so they work in IE6
-    a=this.content.getElementsByTagName('a');
-    for (i=0; i<a.length; i++) {
-      a[i].href='javascript:void(0)';
-    }
-    
-    Rico.eventBind(this.tbody,"click", Rico.eventHandle(this,'saveAndClose'));
-    this.close();
-    this.bPageLoaded=true;
-  },
-
-  _createTitleSection : function(section) {
-    var s=document.createElement("span");
-    s.className='RicoCal'+section+'Heading';
-
-    var a=s.appendChild(document.createElement("a"));
-    a.className='Rico_leftArrow';
-    if (Rico.theme.leftArrowAnchor) Rico.addClass(a,Rico.theme.leftArrowAnchor);
-    a.appendChild(this.createNavArrow('dec'+section,'left'));
-
-    a=s.appendChild(document.createElement("a"));
-    a.style.display='inline';
-    Rico.eventBind(a,"click", Rico.eventHandle(this,'popUp'+section), false);
-    this['title'+section]=a;
-
-    a=s.appendChild(document.createElement("a"));
-    a.className='Rico_rightArrow';
-    if (Rico.theme.rightArrowAnchor) Rico.addClass(a,Rico.theme.rightArrowAnchor);
-    a.appendChild(this.createNavArrow('inc'+section,'right'));
-    return s
-  },
-  
-  selectNow : function() {
-    var today = new Date();
-    this.dateNow  = today.getDate();
-    this.monthNow = today.getMonth();
-    this.yearNow  = today.getFullYear();
-    this.monthSelected=this.monthNow;
-    this.yearSelected=this.yearNow;
-    this.constructCalendar();
-  },
-  
-/** @private */
-  createNavArrow: function(funcname,gifname) {
-    var img;
-    img=document.createElement("span");
-    img.className=Rico.theme[gifname+'Arrow'] || 'Rico_'+gifname+'Arrow';
-    Rico.eventBind(img,"click", Rico.eventHandle(this,funcname), false);
-    return img;
-  },
-
-/**
- * @returns true if yr/mo is within minDate/MaxDate
- */
-  isValidMonth : function(yr,mo) {
-    if (yr < this.options.minDate.getFullYear()) return false;
-    if (yr == this.options.minDate.getFullYear() && mo < this.options.minDate.getMonth()) return false;
-    if (yr > this.options.maxDate.getFullYear()) return false;
-    if (yr == this.options.maxDate.getFullYear() && mo > this.options.maxDate.getMonth()) return false;
-    return true;
-  },
-
-  incMonth : function() {
-    var newMonth=this.monthSelected+1;
-    var newYear=this.yearSelected;
-    if (newMonth>11) {
-      newMonth=0;
-      newYear++;
-    }
-    if (!this.isValidMonth(newYear,newMonth)) return;
-    this.monthSelected=newMonth;
-    this.yearSelected=newYear;
-    this.constructCalendar();
-  },
-
-  decMonth : function() {
-    var newMonth=this.monthSelected-1;
-    var newYear=this.yearSelected;
-    if (newMonth<0) {
-      newMonth=11;
-      newYear--;
-    }
-    if (!this.isValidMonth(newYear,newMonth)) return;
-    this.monthSelected=newMonth;
-    this.yearSelected=newYear;
-    this.constructCalendar();
-  },
-  
-/** @private */
-  selectMonth : function(e) {
-    var el=Rico.eventElement(e);
-    this.monthSelected=parseInt(el.name,10);
-    this.constructCalendar();
-    Rico.eventStop(e);
-  },
-
-  popUpMonth : function(e) {
-    if (this.monthPopup.visible()) {
-      this.popDownMonth();
-      return;
-    }
-    this.popDownYear();
-    this.monthPopup.openPopup(this.titleMonth.parentNode.offsetLeft, this.thead.offsetHeight+2);
-    Rico.eventStop(e);
-    return false;
-  },
-
-  popDownMonth : function() {
-    this.monthPopup.closePopup();
-  },
-
-  popDownYear : function() {
-    this.yearPopup.closePopup();
-    this.yearInput.disabled=true;  // make sure this does not get submitted
-  },
-
-/**
- * Prompt for year
- */
-  popUpYear : function(e) {
-    if (this.yearPopup.visible()) {
-      this.popDownYear();
-      return;
-    }
-    this.popDownMonth();
-    this.yearPopup.openPopup(90, this.thead.offsetHeight+2);
-    this.yearLabel.innerHTML=Rico.getPhraseById("calYearRange",this.options.minDate.getFullYear(),this.options.maxDate.getFullYear());
-    this.yearInput.disabled=false;
-    this.yearInput.value='';   // this.yearSelected
-    this.yearInput.focus();
-    Rico.eventStop(e);
-    return false;
-  },
-  
-  yearKey : function(e) {
-    switch (Rico.eventKey(e)) {
-      case 27: this.popDownYear(); Rico.eventStop(e); return false;
-      case 13: this.processPopUpYear(); Rico.eventStop(e); return false;
-    }
-    return true;
-  },
-  
-  processPopUpYear : function() {
-    var newYear=this.yearInput.value;
-    newYear=parseInt(newYear,10);
-    if (isNaN(newYear) || newYear<this.options.minDate.getFullYear() || newYear>this.options.maxDate.getFullYear()) {
-      alert(Rico.getPhraseById("calInvalidYear"));
-    } else {
-      this.yearSelected=newYear;
-      this.popDownYear();
-      this.constructCalendar();
-    }
-  },
-  
-  incYear : function() {
-    if (this.yearSelected>=this.options.maxDate.getFullYear()) return;
-    this.yearSelected++;
-    this.constructCalendar();
-  },
-
-  decYear : function() {
-    if (this.yearSelected<=this.options.minDate.getFullYear()) return;
-    this.yearSelected--;
-    this.constructCalendar();
-  },
-
-  // tried a number of different week number functions posted on the net
-  // this is the only one that produced consistent results when comparing week numbers for December and the following January
-  WeekNbr : function(year,month,day) {
-    var when = new Date(year,month,day);
-    var newYear = new Date(year,0,1);
-    var offset = 7 + 1 - newYear.getDay();
-    if (offset == 8) offset = 1;
-    var daynum = ((Date.UTC(year,when.getMonth(),when.getDate(),0,0,0) - Date.UTC(year,0,1,0,0,0)) /1000/60/60/24) + 1;
-    var weeknum = Math.floor((daynum-offset+7)/7);
-    if (weeknum == 0) {
-      year--;
-      var prevNewYear = new Date(year,0,1);
-      var prevOffset = 7 + 1 - prevNewYear.getDay();
-      weeknum = (prevOffset == 2 || prevOffset == 8) ? 53 : 52;
-    }
-    return weeknum;
-  },
-
-  constructCalendar : function() {
-    var aNumDays = [31,0,31,30,31,30,31,31,30,31,30,31];
-    var startDate = new Date (this.yearSelected,this.monthSelected,1);
-    var endDate,numDaysInMonth,i,colnum;
-
-    if (typeof this.monthSelected!='number' || this.monthSelected>=12 || this.monthSelected<0) {
-      alert('ERROR in calendar: monthSelected='+this.monthSelected);
-      return;
-    }
-
-    if (this.monthSelected==1) {
-      endDate = new Date (this.yearSelected,this.monthSelected+1,1);
-      endDate = new Date (endDate - (24*60*60*1000));
-      numDaysInMonth = endDate.getDate();
-    } else {
-      numDaysInMonth = aNumDays[this.monthSelected];
-    }
-    var dayPointer = startDate.getDay() - this.options.startAt;
-    if (dayPointer<0) dayPointer+=7;
-    this.popDownMonth();
-    this.popDownYear();
-
-    //this.bgcolor=Rico.getStyle(this.tbody,'background-color');
-    //this.bgcolor=this.bgcolor.replace(/\"/g,'');
-    if (this.options.showWeekNumber) {
-      for (i=1; i<7; i++) {
-        this.tbody.rows[i].cells[0].innerHTML='&nbsp;';
-      }
-    }
-    for ( i=0; i<dayPointer; i++ ) {
-      this.resetCell(this.tbody.rows[1].cells[i+this.colStart]);
-    }
-
-    for ( var datePointer=1,r=1; datePointer<=numDaysInMonth; datePointer++,dayPointer++ ) {
-      colnum=dayPointer % 7;
-      if (this.options.showWeekNumber && colnum==0) {
-        this.tbody.rows[r].cells[0].innerHTML=this.WeekNbr(this.yearSelected,this.monthSelected,datePointer);
-      }
-      var c=this.tbody.rows[r].cells[colnum+this.colStart];
-      c.innerHTML=datePointer;
-      c.className=this.styles[colnum];
-      if ((datePointer==this.dateNow)&&(this.monthSelected==this.monthNow)&&(this.yearSelected==this.yearNow)) {
-        Rico.addClass(c,Rico.theme.calendarToday || 'ricoCalToday');
-      }
-      if (Rico.theme.calendarDay) Rico.addClass(c,Rico.theme.calendarDay);
-      if ((datePointer==this.odateSelected) && (this.monthSelected==this.omonthSelected) && (this.yearSelected==this.oyearSelected)) {
-        Rico.addClass(c,Rico.theme.calendarSelectedDay || 'ricoSelectedDay');
-      }
-      var h=this.Holidays[this.holidayKey(this.yearSelected,this.monthSelected,datePointer)];
-      if (!h)  {
-        h=this.Holidays[this.holidayKey(0,this.monthSelected,datePointer)];
-      }
-      c.style.color=h ? h.txtColor : '';
-      c.style.backgroundColor=h ? h.bgColor : '';
-      c.title=h ? h.desc : '';
-      if (colnum==6) r++;
-    }
-    while (dayPointer<42) {
-      colnum=dayPointer % 7;
-      this.resetCell(this.tbody.rows[r].cells[colnum+this.colStart]);
-      dayPointer++;
-      if (colnum==6) r++;
-    }
-
-    this.titleMonth.innerHTML = Rico.monthAbbr(this.monthSelected);
-    this.titleYear.innerHTML = this.yearSelected;
-    if (this.todayCell) {
-      this.todayCell.innerHTML = Rico.getPhraseById("calToday",this.dateNow,Rico.monthAbbr(this.monthNow),this.yearNow,this.monthNow+1);
-    }
-  },
-  
-/** @private */
-  resetCell: function(c) {
-    c.innerHTML="&nbsp;";
-    c.className='ricoCalEmpty';
-    c.style.color='';
-    c.style.backgroundColor='';
-    c.title='';
-  },
-  
-/** @private */
-  saveAndClose : function(e) {
-    Rico.eventStop(e);
-    var el=Rico.eventElement(e);
-    var s=el.innerHTML.replace(/&nbsp;/g,'');
-    if (s=='' || el.className=='ricoCalWeekNum') return;
-    var day=parseInt(s,10);
-    if (isNaN(day)) return;
-    var d=new Date(this.yearSelected,this.monthSelected,day);
-    var dateStr=Rico.formatDate(d,this.dateFmt=='ISO8601' ? 'yyyy-mm-dd' : this.dateFmt);
-    if (this.returnValue) {
-      this.returnValue(dateStr);
-      this.close();
-    }
-  },
-
-  open : function(curval) {
-    if (!this.bPageLoaded) return;
-    var today = new Date();
-    this.dateNow  = today.getDate();
-    this.monthNow = today.getMonth();
-    this.yearNow  = today.getFullYear();
-    this.oyearSelected = -1;
-    if (typeof curval=='object') {
-      this.odateSelected  = curval.getDate();
-      this.omonthSelected = curval.getMonth();
-      this.oyearSelected  = curval.getFullYear();
-    } else if (this.dateFmt=='ISO8601') {
-      var d=Rico.setISO8601(curval);
-      if (d) {
-        this.odateSelected  = d.getDate();
-        this.omonthSelected = d.getMonth();
-        this.oyearSelected  = d.getFullYear();
-      }
-    } else if (this.re.exec(curval)) {
-      var aDate = [ RegExp.$1, RegExp.$3, RegExp.$5 ];
-      this.odateSelected  = parseInt(aDate[this.dateParts.dd], 10);
-      this.omonthSelected = parseInt(aDate[this.dateParts.mm], 10) - 1;
-      this.oyearSelected  = parseInt(aDate[this.dateParts.yyyy], 10);
-      if (this.oyearSelected < 100) {
-        // apply a century to 2-digit years
-        this.oyearSelected+=this.yearNow - (this.yearNow % 100);
-        var maxyr=this.options.maxDate.getFullYear();
-        while (this.oyearSelected > maxyr) this.oyearSelected-=100;
-      }
-    } else {
-      if (curval) {
-        alert('ERROR: invalid date passed to calendar ('+curval+')');
-      }
-    }
-    if (this.oyearSelected > 0) {
-      this.dateSelected=this.odateSelected;
-      this.monthSelected=this.omonthSelected;
-      this.yearSelected=this.oyearSelected;
-    } else {
-      this.dateSelected=this.dateNow;
-      this.monthSelected=this.monthNow;
-      this.yearSelected=this.yearNow;
-    }
-    this.constructCalendar();
-    this.openPopup();
-  }
-};
diff --git a/ricoClient/js/minsrc/ricoColorPicker.js b/ricoClient/js/minsrc/ricoColorPicker.js
deleted file mode 100644 (file)
index 36c2a3e..0000000
+++ /dev/null
@@ -1,110 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-// ===================================================================
-// Adapted to Rico by Matt Brown from code 
-//   published by Matt Kruse http://www.mattkruse.com/
-// ===================================================================
-
-Rico.ColorPicker = function(id,options) {
-  this.initialize(id,options);
-};
-
-Rico.ColorPicker.prototype = {
-/**
- * @class Implements a pop-up color picker control.
- * @extends Rico.Popup
- * @constructs
- * @param id unique identifier
- * @param options object may contain any of the following:<dl>
- *   <dt>showColorCode</dt><dd> show hex color code as user hovers over color grid? default=false</dd>
- *   <dt>cellsPerRow  </dt><dd> number of colors per row in the grid? default=18</dd>
- *   <dt>palette      </dt><dd> array of 6 digit hex values, default=216 "web safe" colors</dd>
- *</dl>
- */
-  initialize: function(id,options) {
-    this.id=id;
-    this.currentValue = "#FFFFFF";
-    Rico.extend(this, new Rico.Popup());
-    Rico.extend(this.options, {
-      showColorCode : false,
-      cellsPerRow   : 18,
-      palette       : []
-    });
-    var hexvals=['00','33','66','99','CC','FF'];
-    for (var g=0; g<hexvals.length; g++) {
-      for (var r=0; r<hexvals.length; r++) {
-        for (var b=0; b<hexvals.length; b++) {
-          this.options.palette.push(hexvals[r]+hexvals[g]+hexvals[b]);
-        }
-      }
-    }
-    Rico.extend(this.options, options || {});
-  },
-
-  atLoad : function() {
-    this.createContainer();
-    this.content.className='ricoColorPicker';
-    var width = this.options.cellsPerRow;
-    var cp_contents = "<TABLE BORDER='1' CELLSPACING='1' CELLPADDING='0'>";
-    for (var i=0; i<this.options.palette.length; i++) {
-      if ((i % width) == 0) { cp_contents += "<TR>"; }
-      cp_contents += '<TD BGCOLOR="#'+this.options.palette[i]+'">&nbsp;</TD>';
-      if ( ((i+1)>=this.options.palette.length) || (((i+1) % width) == 0)) {
-        cp_contents += "</TR>";
-      }
-    }
-    var halfwidth = Math.floor(width/2);
-    if (this.options.showColorCode) {
-      cp_contents += "<TR><TD COLSPAN='"+halfwidth+"' ID='colorPickerSelectedColor'>&nbsp;</TD><TD COLSPAN='"+(width-halfwidth)+"' ALIGN='CENTER' ID='colorPickerSelectedColorValue'>#FFFFFF</TD></TR>";
-    } else {
-      cp_contents += "<TR><TD COLSPAN='"+width+"' ID='colorPickerSelectedColor'>&nbsp;</TD></TR>";
-    }
-    cp_contents += "</TABLE>";
-    this.content.innerHTML=cp_contents;
-    /**
-     * alias for openPopup
-     * @function
-     */
-    this.open=this.openPopup;
-    /**
-     * alias for closePopup
-     * @function
-     */
-    this.close=this.closePopup;
-    Rico.eventBind(this.container,"mouseover", Rico.eventHandle(this,'highlightColor'), false);
-    Rico.eventBind(this.container,"click", Rico.eventHandle(this,'selectColor'), false);
-    this.close();
-  },
-
-/** @private */
-  selectColor: function(e) {
-    Rico.eventStop(e);
-    if (this.returnValue) this.returnValue(this.currentValue);
-    this.close();
-  },
-
-/* This function runs when you move your mouse over a color block */
-/** @private */
-  highlightColor: function(e) {
-    var elem = Rico.eventElement(e);
-    if (!elem.tagName || elem.tagName.toLowerCase() != 'td') return;
-    var c=Rico.Color.createColorFromBackground(elem).toString();
-    this.currentValue = c;
-    Rico.setStyle('colorPickerSelectedColor', {backgroundColor:c});
-    var d = Rico.$("colorPickerSelectedColorValue");
-    if (d) d.innerHTML = c;
-  }
-};
diff --git a/ricoClient/js/minsrc/ricoDragDrop.js b/ricoClient/js/minsrc/ricoDragDrop.js
deleted file mode 100644 (file)
index 5eafd1a..0000000
+++ /dev/null
@@ -1,577 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-Rico.dndMgrList = [];
-
-Rico.registerDraggable = function(aDraggable, mgrIdx) {
-  if (typeof mgrIdx != 'number') mgrIdx=0;
-  if (typeof Rico.dndMgrList[mgrIdx] != 'object')
-    Rico.dndMgrList[mgrIdx] = new Rico.dndMgr();
-  Rico.dndMgrList[mgrIdx].registerDraggable(aDraggable);
-};
-
-Rico.registerDropZone = function(aDropZone, mgrIdx) {
-  if (typeof mgrIdx != 'number') mgrIdx=0;
-  if (typeof Rico.dndMgrList[mgrIdx] != 'object')
-    Rico.dndMgrList[mgrIdx] = new Rico.dndMgr();
-  Rico.dndMgrList[mgrIdx].registerDropZone(aDropZone);
-};
-
-Rico.dndMgr = function() {
-  this.initialize();
-};
-
-Rico.dndMgr.prototype = {
-/**
- * @class Implements drag-n-drop manager -- a group of linked draggables and drop zones
- * @constructs
- */
-   initialize: function() {
-      this.dropZones                = [];
-      this.draggables               = [];
-      this.currentDragObjects       = [];
-      this.dragElement              = null;
-      this.lastSelectedDraggable    = null;
-      this.currentDragObjectVisible = false;
-      this.interestedInMotionEvents = false;
-      this._mouseDown = Rico.eventHandle(this,'_mouseDownHandler');
-      this._mouseMove = Rico.eventHandle(this,'_mouseMoveHandler');
-      this._mouseUp = Rico.eventHandle(this,'_mouseUpHandler');
-   },
-
-   registerDropZone: function(aDropZone) {
-      this.dropZones[ this.dropZones.length ] = aDropZone;
-   },
-
-   deregisterDropZone: function(aDropZone) {
-      var newDropZones = new Array();
-      var j = 0;
-      for ( var i = 0 ; i < this.dropZones.length ; i++ ) {
-         if ( this.dropZones[i] != aDropZone )
-            newDropZones[j++] = this.dropZones[i];
-      }
-
-      this.dropZones = newDropZones;
-   },
-
-   clearDropZones: function() {
-      this.dropZones = new Array();
-   },
-
-   registerDraggable: function( aDraggable ) {
-      this.draggables[ this.draggables.length ] = aDraggable;
-      var htmlElement = aDraggable.getMouseDownHTMLElement();
-      if ( htmlElement != null ) {
-         htmlElement.ricoDraggable = aDraggable;
-         Rico.eventBind(htmlElement, "mousedown", Rico.eventHandle(this,'_attachEvents'));
-         Rico.eventBind(htmlElement, "mousedown", this._mouseDown);
-      }
-   },
-
-   clearSelection: function() {
-      for ( var i = 0 ; i < this.currentDragObjects.length ; i++ )
-         this.currentDragObjects[i].deselect();
-      this.currentDragObjects = new Array();
-      this.lastSelectedDraggable = null;
-   },
-
-   hasSelection: function() {
-      return this.currentDragObjects.length > 0;
-   },
-
-   setStartDragFromElement: function( e, mouseDownElement ) {
-      this.origPos = Rico.cumulativeOffset(mouseDownElement);
-      var coord=Rico.eventClient(e);
-      this.startx = coord.x - this.origPos.left;
-      this.starty = coord.y - this.origPos.top;
-
-      this.interestedInMotionEvents = this.hasSelection();
-      Rico.eventStop(e);
-   },
-
-   updateSelection: function( draggable, extendSelection ) {
-      if ( ! extendSelection )
-         this.clearSelection();
-
-      if ( draggable.isSelected() ) {
-         this.currentDragObjects=this.currentDragObjects.without(draggable);
-         draggable.deselect();
-         if ( draggable == this.lastSelectedDraggable )
-            this.lastSelectedDraggable = null;
-      }
-      else {
-         draggable.select();
-         if ( draggable.isSelected() ) {
-           this.currentDragObjects.push(draggable);
-           this.lastSelectedDraggable = draggable;
-         }
-      }
-   },
-
-   _mouseDownHandler: function(e) {
-      // if not button 1 ignore it...
-      if (!Rico.eventLeftClick(e)) return;
-
-      var eventTarget      = Rico.eventElement(e);
-      var draggableObject  = eventTarget.ricoDraggable;
-
-      var candidate = eventTarget;
-      while (draggableObject == null && candidate.parentNode) {
-         candidate = candidate.parentNode;
-         draggableObject = candidate.ricoDraggable;
-      }
-
-      if ( draggableObject == null ) return;
-
-      this.updateSelection( draggableObject, e.ctrlKey );
-
-      // clear the drop zones postion cache...
-      if ( this.hasSelection() ) {
-         for ( var i = 0 ; i < this.dropZones.length ; i++ )
-            this.dropZones[i].clearPositionCache();
-      }
-      this.setStartDragFromElement( e, draggableObject.getMouseDownHTMLElement() );
-   },
-
-
-   _mouseMoveHandler: function(e) {
-      if ( !this.interestedInMotionEvents ) {
-         return;
-      }
-
-      if ( ! this.hasSelection() )
-         return;
-
-      if ( ! this.currentDragObjectVisible )
-         this._startDrag(e);
-
-      if ( !this.activatedDropZones )
-         this._activateRegisteredDropZones();
-
-      this._updateDraggableLocation(e);
-      this._updateDropZonesHover(e);
-
-      Rico.eventStop(e);
-   },
-
-   _makeDraggableObjectVisible: function(e) {
-      if ( !this.hasSelection() )
-         return;
-
-      var dragElement;
-      if ( this.currentDragObjects.length > 1 )
-         dragElement = this.currentDragObjects[0].getMultiObjectDragGUI(this.currentDragObjects);
-      else
-         dragElement = this.currentDragObjects[0].getSingleObjectDragGUI();
-
-      // go ahead and absolute position it...
-      this.dragElemPosition=Rico.getStyle(dragElement, "position");
-      if (this.dragElemPosition != "absolute")
-         dragElement.style.position = "absolute";
-
-      // need to parent him into the document...
-      if ( dragElement.parentNode == null || dragElement.parentNode.nodeType == 11 )
-         document.body.appendChild(dragElement);
-
-      this.dragElement = dragElement;
-      this._updateDraggableLocation(e);
-
-      this.currentDragObjectVisible = true;
-   },
-
-   _leftOffset: function(e) {
-          return e.offsetX ? document.body.scrollLeft : 0;
-       },
-
-   _topOffset: function(e) {
-          return e.offsetY ? document.body.scrollTop : 0;
-       },
-
-
-   _updateDraggableLocation: function(e) {
-      var dragObjectStyle = this.dragElement.style;
-      var coord=Rico.eventClient(e);
-      dragObjectStyle.left = (coord.x + this._leftOffset(e) - this.startx) + "px";
-      dragObjectStyle.top  = (coord.y + this._topOffset(e) - this.starty) + "px";
-   },
-
-   _updateDropZonesHover: function(e) {
-      var i,n = this.dropZones.length;
-      for ( i = 0 ; i < n ; i++ ) {
-         if ( ! this._mousePointInDropZone( e, this.dropZones[i] ) )
-            this.dropZones[i].hideHover();
-      }
-
-      for ( i = 0 ; i < n ; i++ ) {
-         if ( this._mousePointInDropZone( e, this.dropZones[i] ) ) {
-            if ( this.dropZones[i].canAccept(this.currentDragObjects) )
-               this.dropZones[i].showHover();
-         }
-      }
-   },
-
-   _startDrag: function(e) {
-      for ( var i = 0 ; i < this.currentDragObjects.length ; i++ )
-         this.currentDragObjects[i].startDrag();
-      this._makeDraggableObjectVisible(e);
-   },
-
-   _mouseUpHandler: function(e) {
-      if ( ! this.hasSelection() ) return;
-      if (!Rico.eventLeftClick(e)) return;
-
-      this.interestedInMotionEvents = false;
-
-      if ( this._placeDraggableInDropZone(e) )
-         this._completeDropOperation(e);
-      else if (this.dragElement != null) {
-         Rico.eventStop(e);
-         var self=this;
-         Rico.animate(this.dragElement,
-                      {duration: 300, onEnd: function() { self._doCancelDragProcessing(); } },
-                      {left:this.origPos.left, top:this.origPos.top});
-      }
-
-     Rico.eventUnbind(document.body, "mousemove", this._mouseMove);
-     Rico.eventUnbind(document.body, "mouseup",  this._mouseUp);
-   },
-
-   _retTrue: function () {
-      return true;
-   },
-
-   _completeDropOperation: function(e) {
-      if ( this.dragElement != this.currentDragObjects[0].getMouseDownHTMLElement() ) {
-         if ( this.dragElement.parentNode != null )
-            this.dragElement.parentNode.removeChild(this.dragElement);
-      }
-
-      this._deactivateRegisteredDropZones();
-      this._endDrag();
-      this.clearSelection();
-      this.dragElement = null;
-      this.currentDragObjectVisible = false;
-      Rico.eventStop(e);
-   },
-
-   _doCancelDragProcessing: function() {
-      this._cancelDrag();
-      if ( this.dragElement == this.currentDragObjects[0].getMouseDownHTMLElement() ) {
-         this.dragElement.style.position=this.dragElemPosition;
-      } else {
-         if ( this.dragElement && this.dragElement.parentNode != null )
-            this.dragElement.parentNode.removeChild(this.dragElement);
-      }
-      this._deactivateRegisteredDropZones();
-      this.dragElement = null;
-      this.currentDragObjectVisible = false;
-   },
-
-   _placeDraggableInDropZone: function(e) {
-      var foundDropZone = false;
-      var n = this.dropZones.length;
-      for ( var i = 0 ; i < n ; i++ ) {
-         if ( this._mousePointInDropZone( e, this.dropZones[i] ) ) {
-            if ( this.dropZones[i].canAccept(this.currentDragObjects) ) {
-               this.dropZones[i].hideHover();
-               this.dropZones[i].accept(this.currentDragObjects);
-               foundDropZone = true;
-               break;
-            }
-         }
-      }
-
-      return foundDropZone;
-   },
-
-   _cancelDrag: function() {
-      for ( var i = 0 ; i < this.currentDragObjects.length ; i++ )
-         this.currentDragObjects[i].cancelDrag();
-   },
-
-   _endDrag: function() {
-      for ( var i = 0 ; i < this.currentDragObjects.length ; i++ )
-         this.currentDragObjects[i].endDrag();
-   },
-
-   _mousePointInDropZone: function( e, dropZone ) {
-
-      var absoluteRect = dropZone.getAbsoluteRect();
-      var coord=Rico.eventClient(e);
-
-      return coord.x  > absoluteRect.left + this._leftOffset(e) &&
-             coord.x  < absoluteRect.right + this._leftOffset(e) &&
-             coord.y  > absoluteRect.top + this._topOffset(e)   &&
-             coord.y  < absoluteRect.bottom + this._topOffset(e);
-   },
-
-   _activateRegisteredDropZones: function() {
-      var n = this.dropZones.length;
-      for ( var i = 0 ; i < n ; i++ ) {
-         var dropZone = this.dropZones[i];
-         if ( dropZone.canAccept(this.currentDragObjects) )
-            dropZone.activate();
-      }
-
-      this.activatedDropZones = true;
-   },
-
-   _deactivateRegisteredDropZones: function() {
-      var n = this.dropZones.length;
-      for ( var i = 0 ; i < n ; i++ )
-         this.dropZones[i].deactivate();
-      this.activatedDropZones = false;
-   },
-
-   _attachEvents: function () {
-     Rico.eventBind(document.body, "mousemove", this._mouseMove);
-     Rico.eventBind(document.body, "mouseup",  this._mouseUp);
-   }
-
-};
-
-
-Rico.Draggable = function(type, htmlElement) {
-  this.initialize(type, htmlElement);
-};
-
-Rico.Draggable.prototype = {
-/**
- * @class Implements behavior for a draggable element
- * @constructs
- */
-   initialize: function( type, htmlElement ) {
-      this.type          = type;
-      this.htmlElement   = Rico.$(htmlElement);
-      this.selected      = false;
-   },
-
-   /**
-    *   Returns the HTML element that should have a mouse down event
-    *   added to it in order to initiate a drag operation
-    **/
-   getMouseDownHTMLElement: function() {
-      return this.htmlElement;
-   },
-
-   select: function() {
-      this._select();
-   },
-
-   _select: function() {
-      this.selected = true;
-      if (this.showingSelected) return;
-      this.showingSelected = true;
-
-      var htmlElement = this.getMouseDownHTMLElement();
-      var color = Rico.Color.createColorFromBackground(htmlElement);
-      color.isBright() ? color.darken(0.033) : color.brighten(0.033);
-      this.saveBackground = Rico.getStyle(htmlElement, "backgroundColor", "background-color");
-      htmlElement.style.backgroundColor = color.asHex();
-   },
-
-   deselect: function() {
-      this.selected = false;
-      if (!this.showingSelected) return;
-      var htmlElement = this.getMouseDownHTMLElement();
-      htmlElement.style.backgroundColor = this.saveBackground;
-      this.showingSelected = false;
-   },
-
-   isSelected: function() {
-      return this.selected;
-   },
-
-   startDrag: function() {
-   },
-
-   cancelDrag: function() {
-   },
-
-   endDrag: function() {
-   },
-
-   getSingleObjectDragGUI: function() {
-      return this.htmlElement;
-   },
-
-   getMultiObjectDragGUI: function( draggables ) {
-      return this.htmlElement;
-   },
-
-   getDroppedGUI: function() {
-      return this.htmlElement;
-   },
-
-   toString: function() {
-      return this.type + ":" + this.htmlElement + ":";
-   }
-
-};
-
-
-Rico.LiveGridDraggable = function(grid, rownum, colnum) {
-  this.initialize(grid, rownum, colnum);
-};
-
-Rico.LiveGridDraggable.prototype = Rico.extend(new Rico.Draggable(), {
-/**
- * @class Enables draggable behavior for LiveGrid cells.
- * Called by LiveGrid#appendBlankRow for columns where canDrag is true.
- * @extends Rico.Draggable
- * @constructs
- */
-  initialize: function( grid, rownum, colnum) {\r
-    this.type        = 'RicoCell';\r
-    this.htmlElement = grid.cell(rownum,colnum);\r
-    this.liveGrid    = grid;\r
-    this.dragRow     = rownum;\r
-    this.dragCol     = colnum;\r
-  },\r
-\r
-  select: function() {
-    if (this.dragRow >= this.liveGrid.buffer.totalRows) return;
-    this.selected = true;
-    this.showingSelected = true;
-  },
-
-  deselect: function() {
-    this.selected = false;
-    this.showingSelected = false;
-  },
-
-  getSingleObjectDragGUI: function() {\r
-    var div = document.createElement("div");\r
-    div.className = 'LiveGridDraggable';\r
-    div.style.width = (this.htmlElement.offsetWidth - 10) + "px";\r
-    div.innerHTML=this.htmlElement.innerHTML;
-    return div;\r
-  }\r
-});\r
-
-
-Rico.Dropzone = function(htmlElement) {
-  this.initialize(htmlElement);
-};
-
-Rico.Dropzone.prototype = {
-/**
- * @class Implements behavior for a drop zone
- * @constructs
- */
-   initialize: function( htmlElement ) {
-      this.htmlElement  = Rico.$(htmlElement);
-      this.absoluteRect = null;
-   },
-
-   getHTMLElement: function() {
-      return this.htmlElement;
-   },
-
-   clearPositionCache: function() {
-      this.absoluteRect = null;
-   },
-
-   getAbsoluteRect: function() {
-      if ( this.absoluteRect == null ) {
-         var htmlElement = this.getHTMLElement();
-         var pos = Rico.viewportOffset(htmlElement);
-
-         this.absoluteRect = {
-            top:    pos.top,
-            left:   pos.left,
-            bottom: pos.top + htmlElement.offsetHeight,
-            right:  pos.left + htmlElement.offsetWidth
-         };
-      }
-      return this.absoluteRect;
-   },
-
-   activate: function() {
-      var htmlElement = this.getHTMLElement();
-      if (htmlElement == null  || this.showingActive)
-         return;
-
-      this.showingActive = true;
-      this.saveBackgroundColor = htmlElement.style.backgroundColor;
-
-      var fallbackColor = "#ffea84";
-      var currentColor = Rico.Color.createColorFromBackground(htmlElement);
-      if ( currentColor == null )
-         htmlElement.style.backgroundColor = fallbackColor;
-      else {
-         currentColor.isBright() ? currentColor.darken(0.2) : currentColor.brighten(0.2);
-         htmlElement.style.backgroundColor = currentColor.asHex();
-      }
-   },
-
-   deactivate: function() {
-      var htmlElement = this.getHTMLElement();
-      if (htmlElement == null || !this.showingActive)
-         return;
-
-      htmlElement.style.backgroundColor = this.saveBackgroundColor;
-      this.showingActive = false;
-      this.saveBackgroundColor = null;
-   },
-
-   showHover: function() {
-      var htmlElement = this.getHTMLElement();
-      if ( htmlElement == null || this.showingHover )
-         return;
-
-      this.saveBorderWidth = htmlElement.style.borderWidth;
-      this.saveBorderStyle = htmlElement.style.borderStyle;
-      this.saveBorderColor = htmlElement.style.borderColor;
-
-      this.showingHover = true;
-      htmlElement.style.borderWidth = "1px";
-      htmlElement.style.borderStyle = "solid";
-      //htmlElement.style.borderColor = "#ff9900";
-      htmlElement.style.borderColor = "#ffff00";
-   },
-
-   hideHover: function() {
-      var htmlElement = this.getHTMLElement();
-      if ( htmlElement == null || !this.showingHover )
-         return;
-
-      htmlElement.style.borderWidth = this.saveBorderWidth;
-      htmlElement.style.borderStyle = this.saveBorderStyle;
-      htmlElement.style.borderColor = this.saveBorderColor;
-      this.showingHover = false;
-   },
-
-   canAccept: function(draggableObjects) {
-      return true;
-   },
-
-   accept: function(draggableObjects) {
-      var htmlElement = this.getHTMLElement();
-      if ( htmlElement == null )
-         return;
-
-      var n = draggableObjects.length;
-      for ( var i = 0 ; i < n ; i++ ) {
-         var theGUI = draggableObjects[i].getDroppedGUI();
-         if ( Rico.getStyle( theGUI, "position" ) == "absolute" ) {
-            theGUI.style.position = "static";
-            theGUI.style.top = "";
-            theGUI.style.top = "";
-         }
-         htmlElement.appendChild(theGUI);
-      }
-   }
-};
diff --git a/ricoClient/js/minsrc/ricoGridCommon.js b/ricoClient/js/minsrc/ricoGridCommon.js
deleted file mode 100644 (file)
index bb64a92..0000000
+++ /dev/null
@@ -1,987 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-if(typeof Rico=='undefined') throw("GridCommon requires the Rico JavaScript framework");
-
-/**
- * Define methods that are common to both SimpleGrid and LiveGrid
- */
-Rico.GridCommon = {
-
-  baseInit: function() {
-    this.options = {
-      resizeBackground : 'resize.gif',
-      saveColumnInfo   : {width:true, filter:false, sort:false},  // save info in cookies?
-      cookiePrefix     : 'RicoGrid.',
-      allowColResize   : true,      // allow user to resize columns
-      windowResize     : true,      // Resize grid on window.resize event? Set to false when embedded in an accordian.
-      click            : null,
-      dblclick         : null,
-      contextmenu      : null,
-      menuEvent        : 'dblclick',  // event that triggers menus - click, dblclick, contextmenu, or none (no menus)
-      defaultWidth     : -1,          // if -1, then use unformatted column width
-      scrollBarWidth   : 19,          // this is the value used in positioning calculations, it does not actually change the width of the scrollbar
-      minScrollWidth   : 100,         // min scroll area width when width of frozen columns exceeds window width
-      frozenColumns    : 0,
-      exportWindow     : "height=400,width=500,scrollbars=1,menubar=1,resizable=1",
-      exportStyleList  : ['background-color','color','text-align','font-weight','font-size','font-family'],
-      exportImgTags    : false,       // applies to grid header and to SimpleGrid cells (not LiveGrid cells)
-      exportFormFields : true,
-      FilterLocation   : null,        // heading row number to place filters. -1=add a new heading row.
-      FilterAllToken   : '___ALL___', // select box value to use to indicate ALL
-      columnSpecs      : []
-    };
-    this.colWidths = [];
-    this.hdrCells=[];
-    this.headerColCnt=0;
-    this.headerRowIdx=0;       // row in header which gets resizers (no colspan's in this row)
-    this.tabs=new Array(2);
-    this.thead=new Array(2);
-    this.tbody=new Array(2);
-  },
-
-  attachMenuEvents: function() {
-    var i;
-    if (!this.options.menuEvent || this.options.menuEvent=='none') return;
-    this.hideScroll=navigator.userAgent.match(/Macintosh\b.*\b(Firefox|Camino)\b/i) || (Rico.isOpera && parseFloat(window.opera.version())<9.5);
-    this.options[this.options.menuEvent]=Rico.eventHandle(this,'handleMenuClick');
-    if (this.highlightDiv) {
-      switch (this.options.highlightElem) {
-        case 'cursorRow':
-          this.attachMenu(this.highlightDiv[0]);
-          break;
-        case 'cursorCell':
-          for (i=0; i<2; i++) {
-            this.attachMenu(this.highlightDiv[i]);
-          }
-          break;
-      }
-    }
-    for (i=0; i<2; i++) {
-      this.attachMenu(this.tbody[i]);
-    }
-  },
-
-  attachMenu: function(elem) {
-    if (this.options.click)
-      Rico.eventBind(elem, 'click', this.options.click, false);
-    if (this.options.dblclick) {
-      if (Rico.isWebKit || Rico.isOpera)
-        Rico.eventBind(elem, 'click', Rico.eventHandle(this,'handleDblClick'), false);
-      else
-        Rico.eventBind(elem, 'dblclick', this.options.dblclick, false);
-    }
-    if (this.options.contextmenu) {
-      if (Rico.isOpera || Rico.isKonqueror)
-        Rico.eventBind(elem, 'click', Rico.eventHandle(this,'handleContextMenu'), false);
-      else
-        Rico.eventBind(elem, 'contextmenu', this.options.contextmenu, false);
-    }
-  },
-
-/**
- * implement double-click for browsers that don't support a double-click event (e.g. Safari)
- */
-  handleDblClick: function(e) {
-    var elem=Rico.eventElement(e);
-    if (this.dblClickElem == elem) {
-      this.options.dblclick(e);
-    } else {
-      this.dblClickElem = elem;
-      this.safariTimer=Rico.runLater(300,this,'clearDblClick');
-    }
-  },
-
-  clearDblClick: function() {
-    this.dblClickElem=null;
-  },
-
-/**
- * implement right-click for browsers that don't support contextmenu event (e.g. Opera, Konqueror)
- * use control-click instead
- */
-  handleContextMenu: function(e) {
-    var b;
-    if( typeof( e.which ) == 'number' )
-      b = e.which; //Netscape compatible
-    else if( typeof( e.button ) == 'number' )
-      b = e.button; //DOM
-    else
-      return;
-    if (b==1 && e.ctrlKey) {
-      this.options.contextmenu(e);
-    }
-  },
-
-  cancelMenu: function() {
-    if (this.menu) this.menu.cancelmenu();
-  },
-
-/**
- * gather info from original headings
- */
-  getColumnInfo: function(hdrSrc) {
-    Rico.log('getColumnInfo: len='+hdrSrc.length);
-    if (hdrSrc.length == 0) return 0;
-    this.headerRowCnt=hdrSrc.length;
-    var r,c,colcnt;
-    for (r=0; r<this.headerRowCnt; r++) {
-      var headerRow = hdrSrc[r];
-      var headerCells=headerRow.cells;
-      if (r >= this.hdrCells.length) this.hdrCells[r]=[];
-      for (c=0; c<headerCells.length; c++) {
-        var obj={};
-        obj.cell=headerCells[c];
-        obj.colSpan=headerCells[c].colSpan || 1;  // Safari & Konqueror return default colspan of 0
-        if (this.options.defaultWidth < 0) obj.initWidth=headerCells[c].offsetWidth;
-        this.hdrCells[r].push(obj);
-      }
-      if (headerRow.id.slice(-5)=='_main') {
-        colcnt=this.hdrCells[r].length;
-        this.headerRowIdx=r;
-      }
-    }
-    if (!colcnt) {
-      this.headerRowIdx=this.headerRowCnt-1;
-      colcnt=this.hdrCells[this.headerRowIdx].length;
-    }
-    Rico.log("getColumnInfo: colcnt="+colcnt);
-    return colcnt;
-  },
-
-  addHeadingRow: function(className) {
-    var r=this.headerRowCnt++;
-    this.hdrCells[r]=[];
-    for( var h=0; h < 2; h++ ) {
-      var row = this.thead[h].insertRow(-1);
-      var newClass='ricoLG_hdg '+this.tableId+'_hdg'+r;
-      if (className) newClass+=' '+className;
-      row.className=newClass;
-      var limit= h==0 ? this.options.frozenColumns : this.headerColCnt-this.options.frozenColumns;
-      for( var c=0; c < limit; c++ ) {
-        var hdrCell=row.insertCell(-1);
-        var colDiv=Rico.wrapChildren(hdrCell,'ricoLG_col');
-        Rico.wrapChildren(colDiv,'ricoLG_cell');
-        this.hdrCells[r].push({cell:hdrCell,colSpan:1});
-      }
-    }
-    return r;
-  },
-  
-/**
- * create column array
- */
-  createColumnArray: function(columnType) {
-    this.direction=Rico.getStyle(this.outerDiv,'direction').toLowerCase();  // ltr or rtl
-    this.align=this.direction=='rtl' ? ['right','left'] : ['left','right'];
-    Rico.log('createColumnArray: dir='+this.direction);
-    this.columns = [];
-    for (var c=0; c < this.headerColCnt; c++) {
-      Rico.log("createColumnArray: c="+c);
-      var tabidx=c<this.options.frozenColumns ? 0 : 1;
-      var col=new Rico[columnType](this, c, this.hdrCells[this.headerRowIdx][c], tabidx);
-      this.columns.push(col);
-      if (c > 0) this.columns[c-1].next=col;
-    }
-    this.getCookie();
-    Rico.runLater(100,this,'insertResizers');  // avoids peek-a-boo bug in column 1 in IE6/7
-  },
-
-/**
- * Insert resizing handles
- */
-  insertResizers: function() {
-    if (!this.options.allowColResize) return;
-    for (var x=0;x<this.columns.length;x++) {
-      this.columns[x].insertResizer();
-    }
-  },
-
-/**
- * Create div structure
- */
-  createDivs: function() {
-    Rico.log("createDivs start");
-    this.outerDiv   = this.createDiv("outer");
-    if (Rico.theme.widget) Rico.addClass(this.outerDiv,Rico.theme.widget);
-    if (this.outerDiv.firstChild && this.outerDiv.firstChild.tagName && this.outerDiv.firstChild.tagName.toUpperCase()=='TABLE') {
-      this.structTab=this.outerDiv.firstChild;
-      this.structTabLeft=this.structTab.rows[0].cells[0];
-      this.structTabUR=this.structTab.rows[0].cells[1];
-      this.structTabLR=this.structTab.rows[1].cells[0];
-    } else {
-      this.structTab = document.createElement("table");
-      this.structTab.border=0;
-      this.structTab.cellPadding=0;
-      this.structTab.cellSpacing=0;
-      var tr1=this.structTab.insertRow(-1);
-      tr1.vAlign='top';
-      this.structTabLeft=tr1.insertCell(-1);
-      this.structTabLeft.rowSpan=2;
-      var tr2=this.structTab.insertRow(-1);
-      tr2.vAlign='top';
-      this.structTabUR=tr1.insertCell(-1);
-      this.structTabLR=tr2.insertCell(-1);
-      this.outerDiv.appendChild(this.structTab);
-    }
-    //this.structTabLR.style.overflow='hidden';
-    //if (Rico.isOpera) this.outerDiv.style.overflow="hidden";
-    this.frozenTabs = this.createDiv("frozenTabs",this.structTabLeft);
-    this.innerDiv   = this.createDiv("inner",this.structTabUR);
-    this.scrollDiv  = this.createDiv("scroll",this.structTabLR);
-    this.resizeDiv  = this.createDiv("resize",this.outerDiv,true);
-    this.exportDiv  = this.createDiv("export",this.outerDiv,true);
-
-    this.messagePopup=new Rico.Popup();
-    this.messagePopup.createContainer({hideOnEscape:false, hideOnClick:false, parent:this.outerDiv});
-    this.messagePopup.content.className='ricoLG_messageDiv';
-    if (Rico.theme.gridMessage) Rico.addClass(this.messagePopup.content,Rico.theme.gridMessage);
-
-    this.keywordPopup=new Rico.Window('', {zIndex:-1, parent:this.outerDiv});
-    Rico.addClass(this.keywordPopup.container, 'ricoLG_keywordDiv');
-    var instructions=this.keywordPopup.contentDiv.appendChild(document.createElement("p"));
-    instructions.innerHTML=Rico.getPhraseById("keywordPrompt");
-    this.keywordBox=this.keywordPopup.contentDiv.appendChild(document.createElement("input"));
-    this.keywordBox.size=20;
-    Rico.eventBind(this.keywordBox,"keypress", Rico.eventHandle(this,'keywordKey'), false);
-    this.keywordPopup.contentDiv.appendChild(Rico.floatButton('Checkmark', Rico.eventHandle(this,'processKeyword')));
-    var s=this.keywordPopup.contentDiv.appendChild(document.createElement("p"));
-    Rico.setStyle(s,{clear:'both'});
-
-    //this.frozenTabs.style[this.align[0]]='0px';
-    //this.innerDiv.style[this.align[0]]='0px';
-    Rico.log("createDivs end");
-  },
-  
-  keywordKey: function(e) {
-    switch (Rico.eventKey(e)) {
-      case 27: this.closeKeyword(); Rico.eventStop(e); return false;
-      case 13: this.processKeyword(); Rico.eventStop(e); return false;
-    }
-    return true;
-  },
-  
-  openKeyword: function(colnum) {
-    this.keywordCol=colnum;
-    this.keywordBox.value='';
-    this.keywordPopup.setTitle(this.columns[colnum].displayName);
-    this.keywordPopup.centerPopup();
-    this.keywordBox.focus();
-  },
-  
-  closeKeyword: function() {
-    this.keywordPopup.closePopup();
-    this.cancelMenu();
-  },
-  
-  processKeyword: function() {
-    var keyword=this.keywordBox.value;
-    this.closeKeyword();
-    this.columns[this.keywordCol].setFilterKW(keyword);
-  },
-
-/**
- * Create a div and give it a standardized id and class name.
- * If the div already exists, then just assign the class name.
- */
-  createDiv: function(elemName,elemParent,hidden) {
-    var id=this.tableId+"_"+elemName+"Div";
-    var newdiv=document.getElementById(id);
-    if (!newdiv) {
-      newdiv = document.createElement("div");
-      newdiv.id = id;
-      if (elemParent) elemParent.appendChild(newdiv);
-    }
-    newdiv.className = "ricoLG_"+elemName+"Div";
-    if (hidden) Rico.hide(newdiv);
-    return newdiv;
-  },
-
-/**
- * Common code used to size & position divs in both SimpleGrid & LiveGrid
- */
-  baseSizeDivs: function() {
-    this.setOtherHdrCellWidths();
-
-    if (this.options.frozenColumns) {
-      Rico.show(this.tabs[0]);
-      Rico.show(this.frozenTabs);
-      // order of next 3 lines is critical in IE6
-      this.hdrHt=Math.max(Rico.nan2zero(this.thead[0].offsetHeight),this.thead[1].offsetHeight);
-      this.dataHt=Math.max(Rico.nan2zero(this.tbody[0].offsetHeight),this.tbody[1].offsetHeight);
-      this.frzWi=this.borderWidth(this.tabs[0]);
-    } else {
-      Rico.hide(this.tabs[0]);
-      Rico.hide(this.frozenTabs);
-      this.frzWi=0;
-      this.hdrHt=this.thead[1].offsetHeight;
-      this.dataHt=this.tbody[1].offsetHeight;
-    }
-
-    var wiLimit,i;
-    var borderWi=this.borderWidth(this.columns[0].dataCell);
-    Rico.log('baseSizeDivs '+this.tableId+': hdrHt='+this.hdrHt+' dataHt='+this.dataHt);
-    Rico.log(this.tableId+' frzWi='+this.frzWi+' borderWi='+borderWi);
-    for (i=0; i<this.options.frozenColumns; i++) {
-      if (this.columns[i].visible) this.frzWi+=parseInt(this.columns[i].colWidth,10)+borderWi;
-    }
-    this.scrTabWi=this.borderWidth(this.tabs[1]);
-    this.scrTabWi0=this.scrTabWi;
-    Rico.log('scrTabWi: '+this.scrTabWi);
-    for (i=this.options.frozenColumns; i<this.columns.length; i++) {
-      if (this.columns[i].visible) this.scrTabWi+=parseInt(this.columns[i].colWidth,10)+borderWi;
-    }
-    this.scrWi=this.scrTabWi+this.options.scrollBarWidth;
-    if (this.sizeTo=='parent') {
-      if (Rico.isIE) Rico.hide(this.outerDiv);
-      wiLimit=this.outerDiv.parentNode.offsetWidth;
-      if (Rico.isIE) Rico.show(this.outerDiv);
-    }  else {
-      wiLimit=Rico.windowWidth()-this.options.scrollBarWidth-8;
-    }
-    if (this.outerDiv.parentNode.clientWidth > 0)
-      wiLimit=Math.min(this.outerDiv.parentNode.clientWidth, wiLimit);
-    var overage=this.frzWi+this.scrWi-wiLimit;
-    Rico.log('baseSizeDivs '+this.tableId+': scrWi='+this.scrWi+' wiLimit='+wiLimit+' overage='+overage+' clientWidth='+this.outerDiv.parentNode.clientWidth);
-    if (overage > 0 && this.options.frozenColumns < this.columns.length)
-      this.scrWi=Math.max(this.scrWi-overage, this.options.minScrollWidth);
-    this.scrollDiv.style.width=this.scrWi+'px';
-    //this.scrollDiv.style.top=this.hdrHt+'px';
-    //this.frozenTabs.style.width=this.scrollDiv.style[this.align[0]]=this.innerDiv.style[this.align[0]]=this.frzWi+'px';
-    this.frozenTabs.style.width=this.frzWi+'px';
-    this.outerDiv.style.width=(this.frzWi+this.scrWi)+'px';
-  },
-
-/**
- * Returns the sum of the left & right border widths of an element
- */
-  borderWidth: function(elem) {
-    var l=Rico.nan2zero(Rico.getStyle(elem,'borderLeftWidth'));
-    var r=Rico.nan2zero(Rico.getStyle(elem,'borderRightWidth'));
-    Rico.log((elem.id || elem.tagName)+' borderWidth: L='+l+', R='+r);
-    return l + r;
-//    return Rico.nan2zero(Rico.getStyle(elem,'borderLeftWidth')) + Rico.nan2zero(Rico.getStyle(elem,'borderRightWidth'));
-  },
-
-  setOtherHdrCellWidths: function() {
-    var c,i,j,r,w,hdrcell,cell,origSpan,newSpan,divs;
-    for (r=0; r<this.hdrCells.length; r++) {
-      if (r==this.headerRowIdx) continue;
-      Rico.log('setOtherHdrCellWidths: r='+r);
-      c=i=0;
-      while (i<this.headerColCnt && c<this.hdrCells[r].length) {
-        hdrcell=this.hdrCells[r][c];
-        cell=hdrcell.cell;
-        origSpan=newSpan=hdrcell.colSpan;
-        for (w=j=0; j<origSpan; j++, i++) {
-          if (this.columns[i].hdrCell.style.display=='none')
-            newSpan--;
-          else if (this.columns[i].hdrColDiv.style.display!='none')
-            w+=parseInt(this.columns[i].colWidth,10);
-        }
-        if (!hdrcell.hdrColDiv || !hdrcell.hdrCellDiv) {
-          divs=cell.getElementsByTagName('div');
-          hdrcell.hdrColDiv=(divs.length<1) ? Rico.wrapChildren(cell,'ricoLG_col') : divs[0];
-          hdrcell.hdrCellDiv=(divs.length<2) ? Rico.wrapChildren(hdrcell.hdrColDiv,'ricoLG_cell') : divs[1];
-        }
-        if (newSpan==0) {
-          cell.style.display='none';
-        } else if (w==0) {
-          hdrcell.hdrColDiv.style.display='none';
-          cell.colSpan=newSpan;
-        } else {
-          cell.style.display='';
-          hdrcell.hdrColDiv.style.display='';
-          cell.colSpan=newSpan;
-          hdrcell.hdrColDiv.style.width=w+'px';
-        }
-        c++;
-      }
-    }
-  },
-
-  initFilterImage: function(filterRowNum){
-    this.filterAnchor=document.getElementById(this.tableId+'_filterLink');
-    if (!this.filterAnchor) return;
-    this.filterRows=Rico.select('tr.'+this.tableId+'_hdg'+filterRowNum);
-    if (this.filterRows.length!=2) return;
-    for (var i=0, r=[]; i<2; i++) r[i]=Rico.select('.ricoLG_cell',this.filterRows[i]);
-    this.filterElements=r[0].concat(r[1]);
-    this.saveHeight = this.filterElements[0].offsetHeight;
-    var pt=Rico.getStyle(this.filterElements[0],'paddingTop');
-    var pb=Rico.getStyle(this.filterElements[0],'paddingBottom');
-    if (pt) this.saveHeight-=parseInt(pt,10);
-    if (pb) this.saveHeight-=parseInt(pb,10);
-    this.rowNum = filterRowNum;
-    this.setFilterImage(false);
-    //Rico.eventBind(this.filterAnchor, 'click', Rico.eventHandle(this,'toggleFilterRow'), false);
-  },
-
-  toggleFilterRow: function() {
-    if ( Rico.visible(this.filterRows[0]) )
-      this.slideFilterUp();
-    else
-      this.slideFilterDown();
-  },
-
-  setFilterImage: function(expandFlag) {
-    var altText=Rico.getPhraseById((expandFlag ? 'show' : 'hide')+'FilterRow');
-    this.filterAnchor.innerHTML = '<img src="'+Rico.imgDir+'tableFilter'+(expandFlag ? 'Expand' : 'Collapse')+'.gif" alt="'+altText+'" border="0">';
-  },
-
-/**
- * Returns a div for the cell at the specified row and column index.
- * In SimpleGrid, r can refer to any row in the grid.
- * In LiveGrid, r refers to a visible row (row 0 is the first visible row).
- */
-  cell: function(r,c) {
-    return (0<=c && c<this.columns.length && r>=0) ? this.columns[c].cell(r) : null;
-  },
-
-/**
- * Returns the screen height available for a grid
- */
-  availHt: function() {
-    var divPos=Rico.cumulativeOffset(this.outerDiv);
-    return Rico.windowHeight()-divPos.top-2*this.options.scrollBarWidth-15;  // allow for scrollbar and some margin
-  },
-
-  setHorizontalScroll: function() {
-    var newLeft=(-this.scrollDiv.scrollLeft)+'px';
-    this.hdrTabs[1].style.marginLeft=newLeft;
-  },
-
-  pluginScroll: function() {
-     if (this.scrollPluggedIn) return;
-     Rico.eventBind(this.scrollDiv,"scroll",this.scrollEventFunc, false);
-     this.scrollPluggedIn=true;
-  },
-
-  unplugScroll: function() {
-     Rico.eventUnbind(this.scrollDiv,"scroll", this.scrollEventFunc , false);
-     this.scrollPluggedIn=false;
-  },
-
-  hideMsg: function() {
-    this.messagePopup.closePopup();
-  },
-
-  showMsg: function(msg) {
-    this.messagePopup.setContent(msg);
-    this.messagePopup.centerPopup();
-    Rico.log("showMsg: "+msg);
-  },
-
-/**
- * @return array of column objects which have invisible status
- */
-  listInvisible: function(attr) {
-    var hiddenColumns=[];
-    for (var x=0;x<this.columns.length;x++) {
-      if (!this.columns[x].visible)
-        hiddenColumns.push(attr ? this.columns[x][attr] : this.columns[x]);
-    }
-    return hiddenColumns;
-  },
-
-/**
- * @return index of left-most visibile column, or -1 if there are no visible columns
- */
-  firstVisible: function() {
-    for (var x=0;x<this.columns.length;x++) {
-      if (this.columns[x].visible) return x;
-    }
-    return -1;
-  },
-
-/**
- * Show all columns
- */
-  showAll: function() {
-    var invisible=this.listInvisible();
-    for (var x=0;x<invisible.length;x++)
-      invisible[x].showColumn();
-  },
-  
-  chooseColumns: function() {
-    this.menu.cancelmenu();
-    var x,z,col,itemDiv,span,contentDiv;\r
-    if (!this.columnChooser) {
-      Rico.log('creating columnChooser');
-      z=Rico.getStyle(this.outerDiv.offsetParent,'zIndex');
-      if (typeof z!='number') z=0;
-      this.columnChooser=new Rico.Window(Rico.getPhraseById('gridChooseCols'), {zIndex:z+2, parent:this.outerDiv});
-      Rico.addClass(this.columnChooser.container, 'ricoLG_chooserDiv');
-      contentDiv=this.columnChooser.contentDiv;
-      for (x=0;x<this.columns.length;x++) {
-        col=this.columns[x];
-        itemDiv=contentDiv.appendChild(document.createElement('div'));
-        col.ChooserBox=Rico.createFormField(itemDiv,'input','checkbox');
-        span=itemDiv.appendChild(document.createElement('span'));
-        span.innerHTML=col.displayName;
-        Rico.eventBind(col.ChooserBox, 'click', Rico.eventHandle(col,'chooseColumn'), false);
-      }
-    }
-    Rico.log('opening columnChooser');
-    this.columnChooser.openPopup(1,this.hdrHt);
-    for (x=0;x<this.columns.length;x++) {
-      this.columns[x].ChooserBox.checked=this.columns[x].visible;
-      this.columns[x].ChooserBox.disabled = !this.columns[x].canHideShow();
-    }
-  },
-
-  blankRow: function(r) {
-    for (var c=0; c < this.columns.length; c++) {
-      this.columns[c].clearCell(r);
-    }
-  },
-  
-  getExportStyles: function(chkelem) {
-    var exportStyles=this.options.exportStyleList;
-    var bgImg=Rico.getStyle(chkelem,'backgroundImage');
-    if (!bgImg || bgImg=='none') return exportStyles;
-    for (var styles=[],i=0; i<exportStyles.length; i++)
-      if (exportStyles[i]!='background-color' && exportStyles[i]!='color') styles.push(exportStyles[i]);
-    return styles;
-  },
-
-/**
- * Support function for printVisible()
- */
-  exportStart: function() {
-    var r,c,i,j,hdrcell,newSpan,divs,cell;
-    var exportStyles=this.getExportStyles(this.thead[0]);
-    //alert(exportStyles.join('\n'));
-    this.exportRows=[];
-    this.exportText="<table border='1' cellspacing='0'>";\r
-    for (c=0; c<this.columns.length; c++) {\r
-      if (this.columns[c].visible) this.exportText+="<col width='"+parseInt(this.columns[c].colWidth,10)+"'>";
-    }\r
-    this.exportText+="<thead style='display: table-header-group;'>";
-    if (this.exportHeader) this.exportText+=this.exportHeader;
-    for (r=0; r<this.hdrCells.length; r++) {
-      if (this.hdrCells[r].length==0 || !Rico.visible(this.hdrCells[r][0].cell.parentNode)) continue;
-      this.exportText+="<tr>";
-      for (c=0,i=0; c<this.hdrCells[r].length; c++) {
-        hdrcell=this.hdrCells[r][c];
-        newSpan=hdrcell.colSpan;
-        for (j=0; j<hdrcell.colSpan; j++, i++) {
-          if (!this.columns[i].visible) newSpan--;
-        }
-        if (newSpan > 0) {
-          divs=Rico.select('.ricoLG_cell',hdrcell.cell);
-          cell=divs && divs.length>0 ? divs[0] : hdrcell.cell;
-          this.exportText+="<td style='"+this.exportStyle(cell,exportStyles)+"'";
-          if (hdrcell.colSpan > 1) this.exportText+=" colspan='"+newSpan+"'";
-          this.exportText+=">"+Rico.getInnerText(cell,!this.options.exportImgTags, !this.options.exportFormFields, 'NoExport')+"</td>";
-        }
-      }
-      this.exportText+="</tr>";
-    }
-    this.exportText+="</thead><tbody>";
-  },
-
-/**
- * Support function for printVisible().
- * exportType is optional and defaults 'plain'; 'owc' can be used for IE users with Office Web Components.
- */
-  exportFinish: function(exportType) {
-    if (this.hideMsg) this.hideMsg();
-    window.status=Rico.getPhraseById('exportComplete');
-    if (this.exportRows.length > 0) this.exportText+='<tr>'+this.exportRows.join('</tr><tr>')+'</tr>';
-    if (this.exportFooter) this.exportText+=this.exportFooter;
-    this.exportText+="</tbody></table>";
-    this.exportDiv.innerHTML=this.exportText;
-    this.exportText=undefined;
-    this.exportRows=undefined;
-    if (this.cancelMenu) this.cancelMenu();
-    var w=window.open(Rico.htmDir+'export_'+(exportType || 'plain')+'.html?'+this.exportDiv.id,'',this.options.exportWindow);
-    if (w == null) alert(Rico.getPhraseById('disableBlocker'));
-  },
-
-/**
- * Support function for printVisible()
- */
-  exportStyle: function(elem,styleList) {
-    for (var i=0,s=''; i < styleList.length; i++) {
-      try {
-        var curstyle=Rico.getStyle(elem,styleList[i]);
-        if (curstyle) s+=styleList[i]+':'+curstyle+';';
-      } catch(e) {};
-    }
-    return s;
-  },
-
-/**
- * Gets the value of the grid cookie and interprets the contents.
- * All information for a particular grid is stored in a single cookie.
- * This may include column widths, column hide/show status, current sort, and any column filters.
- */
-  getCookie: function() {
-    var c=Rico.getCookie(this.options.cookiePrefix+this.tableId);
-    if (!c) return;
-    var cookieVals=c.split(',');
-    for (var i=0; i<cookieVals.length; i++) {
-      var v=cookieVals[i].split(':');
-      if (v.length!=2) continue;
-      var colnum=parseInt(v[0].slice(1),10);
-      if (colnum < 0 || colnum >= this.columns.length) continue;
-      var col=this.columns[colnum];
-      switch (v[0].charAt(0)) {
-        case 'w':
-          col.setColWidth(v[1]);
-          col.customWidth=true;
-          break;
-        case 'h':
-          if (v[1].toLowerCase()=='true')
-            col.hideshow(true,true);
-          else
-            col.hideshow(false,true);
-          break;
-        case 's':
-          if (!this.options.saveColumnInfo.sort || !col.sortable) break;
-          col.setSorted(v[1]);
-          break;
-        case 'f':
-          if (!this.options.saveColumnInfo.filter || !col.filterable) break;
-          var filterTemp=v[1].split('~');
-          col.filterOp=filterTemp.shift();
-          col.filterValues = [];
-          col.filterType = Rico.ColumnConst.USERFILTER;
-          for (var j=0; j<filterTemp.length; j++)
-            col.filterValues.push(unescape(filterTemp[j]));
-          break;
-      }
-    }
-  },
-
-/**
- * Sets the grid cookie.
- * All information for a particular grid is stored in a single cookie.
- * This may include column widths, column hide/show status, current sort, and any column filters.
- */
-  setCookie: function() {
-    var cookieVals=[];
-    for (var i=0; i<this.columns.length; i++) {
-      var col=this.columns[i];
-      if (this.options.saveColumnInfo.width) {
-        if (col.customWidth) cookieVals.push('w'+i+':'+col.colWidth);
-        if (col.customVisible) cookieVals.push('h'+i+':'+col.visible);
-      }
-      if (this.options.saveColumnInfo.sort) {
-        if (col.currentSort != Rico.ColumnConst.UNSORTED)
-          cookieVals.push('s'+i+':'+col.currentSort);
-      }
-      if (this.options.saveColumnInfo.filter && col.filterType == Rico.ColumnConst.USERFILTER) {
-        var filterTemp=[col.filterOp];
-        for (var j=0; j<col.filterValues.length; j++)
-          filterTemp.push(escape(col.filterValues[j]));
-        cookieVals.push('f'+i+':'+filterTemp.join('~'));
-      }
-    }
-    Rico.setCookie(this.options.cookiePrefix+this.tableId, cookieVals.join(','), this.options.cookieDays, this.options.cookiePath, this.options.cookieDomain);
-  }
-
-};
-
-
-Rico.ColumnConst = {
-  UNFILTERED:   0,
-  SYSTEMFILTER: 1,
-  USERFILTER:   2,
-
-  UNSORTED:  0,
-  SORT_ASC:  "ASC",
-  SORT_DESC: "DESC",
-
-  MINWIDTH: 10,
-  DOLLAR:  {type:'number', prefix:'$', decPlaces:2, ClassName:'alignright'},
-  EURO:    {type:'number', prefix:'&euro;', decPlaces:2, ClassName:'alignright'},
-  PERCENT: {type:'number', suffix:'%', decPlaces:2, multiplier:100, ClassName:'alignright'},
-  QTY:     {type:'number', decPlaces:0, ClassName:'alignright'},
-  DEFAULT: {type:"showTags"}
-}
-
-
-/**
- * @class Define methods that are common to columns in both SimpleGrid and LiveGrid
- */
-Rico.TableColumnBase = function() {};
-
-Rico.TableColumnBase.prototype = {
-
-/**
- * Common code used to initialize the column in both SimpleGrid & LiveGrid
- */
-  baseInit: function(liveGrid,colIdx,hdrInfo,tabIdx) {
-    Rico.log("TableColumnBase.init index="+colIdx+" tabIdx="+tabIdx);
-    this.liveGrid  = liveGrid;
-    this.index     = colIdx;
-    this.hideWidth = Rico.isKonqueror || Rico.isWebKit || liveGrid.headerRowCnt>1 ? 5 : 2;  // column width used for "hidden" columns. Anything less than 5 causes problems with Konqueror. Best to keep this greater than padding used inside cell.
-    this.options   = liveGrid.options;
-    this.tabIdx    = tabIdx;
-    this.hdrCell   = hdrInfo.cell;
-    this.body = document.getElementsByTagName("body")[0];
-    this.displayName  = this.getDisplayName(this.hdrCell);
-    var divs=this.hdrCell.getElementsByTagName('div');
-    this.hdrColDiv=(divs.length<1) ? Rico.wrapChildren(this.hdrCell,'ricoLG_col') : divs[0];
-    this.hdrCellDiv=(divs.length<2) ? Rico.wrapChildren(this.hdrColDiv,'ricoLG_cell') : divs[1];
-    var sectionIndex= tabIdx==0 ? colIdx : colIdx-liveGrid.options.frozenColumns;
-    this.dataCell = liveGrid.tbody[tabIdx].rows[0].cells[sectionIndex];
-    divs=this.dataCell.getElementsByTagName('div');
-    this.dataColDiv=(divs.length<1) ? Rico.wrapChildren(this.dataCell,'ricoLG_col') : divs[0];
-
-    this.mouseDownHandler= Rico.eventHandle(this,'handleMouseDown');
-    this.mouseMoveHandler= Rico.eventHandle(this,'handleMouseMove');
-    this.mouseUpHandler  = Rico.eventHandle(this,'handleMouseUp');
-    this.mouseOutHandler = Rico.eventHandle(this,'handleMouseOut');
-
-    this.fieldName = 'col'+this.index;
-    var spec = liveGrid.options.columnSpecs[colIdx];
-    this.format=Rico.extend( {}, Rico.ColumnConst.DEFAULT);
-    switch (typeof spec) {
-      case 'object':
-        if (typeof spec.format=='string') Rico.extend(this.format, Rico.ColumnConst[spec.format.toUpperCase()]);
-        Rico.extend(this.format, spec);
-        break;
-      case 'string':
-        if (spec.slice(0,4)=='spec') spec=spec.slice(4).toUpperCase();  // for backwards compatibility
-        if (typeof Rico.ColumnConst[spec]=='object') Rico.extend(this.format, Rico.ColumnConst[spec]);
-        break;
-    }
-    Rico.addClass(this.dataColDiv, this.colClassName());
-    this.visible=true;
-    if (typeof this.format.visible=='boolean') this.visible=this.format.visible;
-    Rico.log("TableColumn.init index="+colIdx+" fieldName="+this.fieldName);
-    this.sortable     = typeof this.format.canSort=='boolean' ? this.format.canSort : liveGrid.options.canSortDefault;
-    this.currentSort  = Rico.ColumnConst.UNSORTED;
-    this.filterable   = typeof this.format.canFilter=='boolean' ? this.format.canFilter : liveGrid.options.canFilterDefault;
-    this.filterType   = Rico.ColumnConst.UNFILTERED;
-    this.hideable     = typeof this.format.canHide=='boolean' ? this.format.canHide : liveGrid.options.canHideDefault;
-
-    var wi=(typeof(this.format.width)=='number') ? this.format.width : hdrInfo.initWidth;
-    wi=(typeof(wi)=='number') ? Math.max(wi,Rico.ColumnConst.MINWIDTH) : liveGrid.options.defaultWidth;
-    this.setColWidth(wi);
-    if (!this.visible) this.setDisplay('none');
-  },
-  
-  colClassName: function() {
-    return this.format.ClassName ? this.format.ClassName : this.liveGrid.tableId+'_col'+this.index;
-  },
-
-  insertResizer: function() {
-    //this.hdrCell.style.width='';
-    if (this.format.noResize) return;
-    var resizer=document.createElement('div');
-    resizer.className='ricoLG_Resize';
-    resizer.style[this.liveGrid.align[1]]='0px';
-    if (this.options.resizeBackground) {
-      var resizePath=Rico.imgDir+this.liveGrid.options.resizeBackground;
-      if (Rico.isIE && Rico.ieVersion < 8)
-        resizePath=location.protocol+resizePath; // IE6+7 only
-      resizer.style.backgroundImage='url('+resizePath+')';
-    }
-    this.hdrCellDiv.appendChild(resizer);
-    Rico.eventBind(resizer,"mousedown", this.mouseDownHandler, false);
-  },
-
-/**
- * get the display name of a column
- */
-  getDisplayName: function(el) {
-    var anchors=el.getElementsByTagName("A");
-    //Check the existance of A tags
-    if (anchors.length > 0)
-      return anchors[0].innerHTML;
-    else
-      return Rico.stripTags(el.innerHTML);
-  },
-
-  _clear: function(gridCell) {
-    gridCell.innerHTML='&nbsp;';
-  },
-
-  clearCell: function(rowIndex) {
-    var gridCell=this.cell(rowIndex);
-    this._clear(gridCell,rowIndex);
-    if (!this.liveGrid.buffer) return;
-    var acceptAttr=this.liveGrid.buffer.options.acceptAttr;
-    for (var k=0; k<acceptAttr.length; k++) {
-      switch (acceptAttr[k]) {
-        case 'style': gridCell.style.cssText=''; break;
-        case 'class': gridCell.className=''; break;
-        default:      gridCell['_'+acceptAttr[k]]=''; break;
-      }
-    }
-  },
-
-  dataTable: function() {
-    return this.liveGrid.tabs[this.tabIdx];
-  },
-
-  numRows: function() {
-    return this.dataColDiv.childNodes.length;
-  },
-
-  clearColumn: function() {
-    var childCnt=this.numRows();
-    for (var r=0; r<childCnt; r++)
-      this.clearCell(r);
-  },
-
-  cell: function(r) {
-    return this.dataColDiv.childNodes[r];
-  },
-
-  getFormattedValue: function(r,xImg,xForm,xClass) {
-    return Rico.getInnerText(this.cell(r),xImg,xForm,xClass);
-  },
-
-  setColWidth: function(wi) {
-    if (typeof wi=='number') {
-      wi=parseInt(wi,10);
-      if (wi < Rico.ColumnConst.MINWIDTH) return;
-      wi=wi+'px';
-    }
-    Rico.log('setColWidth '+this.index+': '+wi);
-    this.colWidth=wi;
-    this.hdrColDiv.style.width=wi;
-    this.dataColDiv.style.width=wi;
-  },
-
-  pluginMouseEvents: function() {
-    if (this.mousePluggedIn==true) return;
-    Rico.eventBind(this.body,"mousemove", this.mouseMoveHandler, false);
-    Rico.eventBind(this.body,"mouseup",   this.mouseUpHandler  , false);
-    Rico.eventBind(this.body,"mouseout",  this.mouseOutHandler , false);
-    this.mousePluggedIn=true;
-  },
-
-  unplugMouseEvents: function() {
-    Rico.eventUnbind(this.body,"mousemove", this.mouseMoveHandler, false);
-    Rico.eventUnbind(this.body,"mouseup",   this.mouseUpHandler  , false);
-    Rico.eventUnbind(this.body,"mouseout",  this.mouseOutHandler , false);
-    this.mousePluggedIn=false;
-  },
-
-  handleMouseDown: function(e) {
-    this.resizeStart=Rico.eventClient(e).x;
-    this.origWidth=parseInt(this.colWidth,10);
-    var p=Rico.positionedOffset(this.hdrCell);
-    if (this.liveGrid.direction=='rtl') {
-      this.edge=p.left+this.liveGrid.options.scrollBarWidth;
-      switch (this.tabIdx) {
-        case 0: this.edge+=this.liveGrid.innerDiv.offsetWidth; break;
-        case 1: this.edge-=this.liveGrid.scrollDiv.scrollLeft; break;
-      }
-    } else {
-      this.edge=p.left+this.hdrCell.offsetWidth;
-      if (this.tabIdx>0) this.edge+=Rico.nan2zero(this.liveGrid.tabs[0].offsetWidth);
-    }
-    this.liveGrid.resizeDiv.style.left=this.edge+"px";
-    this.liveGrid.resizeDiv.style.display="";
-    this.liveGrid.outerDiv.style.cursor='e-resize';
-    this.tmpHighlight=this.liveGrid.highlightEnabled;
-    this.liveGrid.highlightEnabled=false;
-    this.pluginMouseEvents();
-    Rico.eventStop(e);
-  },
-
-  handleMouseMove: function(e) {
-    var delta=Rico.eventClient(e).x-this.resizeStart;
-    var newWidth=(this.liveGrid.direction=='rtl') ? this.origWidth-delta : this.origWidth+delta;
-    if (newWidth < Rico.ColumnConst.MINWIDTH) return;
-    this.liveGrid.resizeDiv.style.left=(this.edge+delta)+"px";
-    this.colWidth=newWidth;
-    Rico.eventStop(e);
-  },
-
-  handleMouseUp: function(e) {
-    this.unplugMouseEvents();
-    Rico.log('handleMouseUp '+this.liveGrid.tableId);
-    this.liveGrid.outerDiv.style.cursor='';
-    this.liveGrid.resizeDiv.style.display="none";
-    this.setColWidth(this.colWidth);
-    this.customWidth=true;
-    this.liveGrid.setCookie();
-    this.liveGrid.highlightEnabled=this.tmpHighlight;
-    this.liveGrid.sizeDivs();
-    Rico.eventStop(e);
-  },
-
-  handleMouseOut: function(e) {
-    var reltg = Rico.eventRelatedTarget(e) || e.toElement;
-    while (reltg != null && reltg.nodeName.toLowerCase() != 'body')
-      reltg=reltg.parentNode;
-    if (reltg!=null && reltg.nodeName.toLowerCase() == 'body') return true;
-    this.handleMouseUp(e);
-    return true;
-  },
-
-  setDisplay: function(d) {
-    this.hdrCell.style.display=d;
-    this.hdrColDiv.style.display=d;
-    this.dataCell.style.display=d;
-    this.dataColDiv.style.display=d;
-  },
-  
-  hideshow: function(visible,noresize) {
-    this.setDisplay(visible ? '' : 'none');
-    this.liveGrid.cancelMenu();
-    this.visible=visible;
-    this.customVisible=true;
-    if (noresize) return;
-    this.liveGrid.setCookie();
-    this.liveGrid.sizeDivs();
-  },
-
-  hideColumn: function() {
-    Rico.log('hideColumn '+this.liveGrid.tableId);
-    this.hideshow(false,false);
-  },
-
-  showColumn: function() {
-    Rico.log('showColumn '+this.liveGrid.tableId);
-    this.hideshow(true,false);
-  },
-
-  chooseColumn: function(e) {
-    var elem=Rico.eventElement(e);
-    this.hideshow(elem.checked,false);
-  },
-
-  setImage: function() {
-    if ( this.currentSort == Rico.ColumnConst.SORT_ASC ) {
-       this.imgSort.style.display='inline-block';
-       this.imgSort.className=Rico.theme.sortAsc || 'ricoLG_sortAsc';
-    } else if ( this.currentSort == Rico.ColumnConst.SORT_DESC ) {
-       this.imgSort.style.display='inline-block';
-       this.imgSort.className=Rico.theme.sortDesc || 'ricoLG_sortDesc';
-    } else {
-       this.imgSort.style.display='none';
-    }
-    if (this.filterType == Rico.ColumnConst.USERFILTER) {
-       this.imgFilter.style.display='';
-       this.imgFilter.title=this.getFilterText();
-    } else {
-       this.imgFilter.style.display='none';
-    }
-  },
-
-  canHideShow: function() {
-    return this.hideable;
-  }
-
-};
diff --git a/ricoClient/js/minsrc/ricoLiveGrid.js b/ricoClient/js/minsrc/ricoLiveGrid.js
deleted file mode 100644 (file)
index 95bc96b..0000000
+++ /dev/null
@@ -1,2439 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-if(typeof Rico=='undefined') throw("LiveGrid requires the Rico JavaScript framework");
-
-
-/** @namespace */
-if (!Rico.Buffer) Rico.Buffer = {};
-
-Rico.Buffer.Base = function(dataTable, options) {
-  this.initialize(dataTable, options);
-}
-/** @lends Rico.Buffer.Base# */
-Rico.Buffer.Base.prototype = {
-/**
- * @class Defines the static buffer class (no AJAX).
- * Loads buffer with data that already exists in the document as an HTML table or passed via javascript.
- * Also serves as a base class for AJAX-enabled buffers.
- * @constructs
- */
-  initialize: function(dataTable, options) {
-    this.clear();
-    this.updateInProgress = false;
-    this.lastOffset = 0;
-    this.rcvdRowCount = false;  // true if an eof element was included in the last xml response
-    this.foundRowCount = false; // true if an xml response is ever received with eof true
-    this.totalRows = 0;
-    this.rowcntContent = "";
-    this.rcvdOffset = -1;
-    this.options = {
-      fixedHdrRows     : 0,
-      canFilter        : true,  // does buffer object support filtering?
-      isEncoded        : true,  // is the data received via ajax html encoded?
-      acceptAttr       : []     // attributes that can be copied from original/ajax data (e.g. className, style, id)
-    };
-    Rico.extend(this.options, options || {});
-    if (dataTable) {
-      this.loadRowsFromTable(dataTable,this.options.fixedHdrRows);
-    } else {
-      this.clear();
-    }
-  },
-
-  registerGrid: function(liveGrid) {
-    this.liveGrid = liveGrid;
-  },
-
-  setTotalRows: function( newTotalRows ) {
-    if (typeof(newTotalRows)!='number') newTotalRows=this.size;
-    if (this.totalRows == newTotalRows) return;
-    this.totalRows = newTotalRows;
-    if (!this.liveGrid) return;
-    Rico.log("setTotalRows, newTotalRows="+newTotalRows);
-    switch (this.liveGrid.sizeTo) {
-      case 'data':
-        this.liveGrid.resizeWindow();
-        break;
-      case 'datamax':
-        this.liveGrid.setPageSize(newTotalRows);
-        break;
-      default:
-        this.liveGrid.updateHeightDiv();
-        break;
-    }
-  },
-
-  loadRowsFromTable: function(tableElement,firstRow) {
-    var newRows = new Array();
-    var trs = tableElement.getElementsByTagName("tr");
-    for ( var i=firstRow || 0; i < trs.length; i++ ) {
-      var row = new Array();
-      var cells = trs[i].getElementsByTagName("td");
-      for ( var j=0; j < cells.length ; j++ )
-        row[j]=cells[j].innerHTML;
-      newRows.push( row );
-    }
-    this.loadRows(newRows);
-  },
-
-  loadRowsFromArray: function(array2D) {
-    for ( var i=0; i < array2D.length; i++ ) {
-      for ( var j=0; j < array2D[i].length ; j++ ) {
-        array2D[i][j]=array2D[i][j].toString();
-      }
-    }
-    this.loadRows(array2D);
-  },
-
-  loadRows: function(jstable) {
-    this.baseRows = jstable;
-    this.startPos = 0;
-    this.size = this.baseRows.length;
-  },
-
-  dom2jstable: function(rowsElement) {
-    Rico.log('dom2jstable: encoded='+this.options.isEncoded);
-    var newRows = new Array();
-    var trs = rowsElement.getElementsByTagName("tr");
-    for ( var i=0; i < trs.length; i++ ) {
-      var row = new Array();
-      var cells = trs[i].getElementsByTagName("td");
-      for ( var j=0; j < cells.length ; j++ )
-        row[j]=Rico.getContentAsString(cells[j],this.options.isEncoded);
-      newRows.push( row );
-    }
-    return newRows;
-  },
-
-  dom2jstableAttr: function(rowsElement,firstRow) {
-    var acceptAttr=this.options.acceptAttr;
-    Rico.log("dom2jstableAttr start, # attr="+acceptAttr.length);
-    var newRows = new Array();
-    var trs = rowsElement.getElementsByTagName("tr");
-    for ( var i=firstRow || 0; i < trs.length; i++ ) {
-      var row = new Array();
-      var cells = trs[i].getElementsByTagName("td");
-      for ( var j=0; j < cells.length ; j++ ) {
-        row[j]={};
-        for (var k=0; k<acceptAttr.length; k++)
-          row[j]['_'+acceptAttr[k]]=cells[j].getAttribute(acceptAttr[k]);
-        if (Rico.isIE) row[j]._class=cells[j].getAttribute('className');
-      }
-      newRows.push( row );
-    }
-    Rico.log("dom2jstableAttr end");
-    return newRows;
-  },
-
-  _blankRow: function() {
-    var newRow=[];
-    for (var i=0; i<this.liveGrid.columns.length; i++) {
-      newRow[i]='';
-    }
-    return newRow;
-  },
-
-  deleteRows: function(rowIndex,cnt) {
-    this.baseRows.splice(rowIndex,typeof(cnt)=='number' ? cnt : 1);
-    this.liveGrid.isPartialBlank=true;
-    this.size=this.baseRows.length;
-  },
-
-  insertRow: function(beforeRowIndex) {
-    var r=this._blankRow();
-    this.baseRows.splice(beforeRowIndex,0,r);
-    this.size=this.baseRows.length;
-    this.liveGrid.isPartialBlank=true;
-    if (this.startPos < 0) this.startPos=0;
-    return r;
-  },
-
-  appendRows: function(cnt) {
-    var newRows=[];
-    for (var i=0; i<cnt; i++) {
-      var r=this._blankRow();
-      this.baseRows.push(r);
-      newRows.push(r);
-    }
-    this.size=this.baseRows.length;
-    this.liveGrid.isPartialBlank=true;
-    if (this.startPos < 0) this.startPos=0;
-    return newRows;
-  },
-  
-  sortFunc: function(coltype) {
-    var self=this;
-    switch (coltype) {
-      case 'number': return function(a,b) { return self._sortNumeric(a,b); };
-      case 'control':return function(a,b) { return self._sortControl(a,b); };
-      default:       return function(a,b) { return self._sortAlpha(a,b); };
-    }
-  },
-
-  sortBuffer: function(colnum) {
-    if (!this.baseRows) {
-      this.delayedSortCol=colnum;
-      return;
-    }
-    this.liveGrid.showMsg(Rico.getPhraseById("sorting"));
-    this.sortColumn=colnum;
-    var col=this.liveGrid.columns[colnum];
-    this.getValFunc=col._sortfunc;
-    this.baseRows.sort(this.sortFunc(col.format.type));
-    if (col.getSortDirection()=='DESC') this.baseRows.reverse();
-  },
-  
-  _sortAlpha: function(a,b) {
-    var aa = this.sortColumn<a.length ? Rico.getInnerText(a[this.sortColumn]) : '';
-    var bb = this.sortColumn<b.length ? Rico.getInnerText(b[this.sortColumn]) : '';
-    if (aa==bb) return 0;
-    if (aa<bb) return -1;
-    return 1;
-  },
-
-  _sortNumeric: function(a,b) {
-    var aa = this.sortColumn<a.length ? this.nan2zero(Rico.getInnerText(a[this.sortColumn])) : 0;
-    var bb = this.sortColumn<b.length ? this.nan2zero(Rico.getInnerText(b[this.sortColumn])) : 0;
-    return aa-bb;
-  },
-
-  nan2zero: function(n) {
-    if (typeof(n)=='string') n=parseFloat(n);
-    return isNaN(n) || typeof(n)=='undefined' ? 0 : n;
-  },
-  
-  _sortControl: function(a,b) {
-    var aa = this.sortColumn<a.length ? Rico.getInnerText(a[this.sortColumn]) : '';
-    var bb = this.sortColumn<b.length ? Rico.getInnerText(b[this.sortColumn]) : '';
-    if (this.getValFunc) {
-      aa=this.getValFunc(aa);
-      bb=this.getValFunc(bb);
-    }
-    if (aa==bb) return 0;
-    if (aa<bb) return -1;
-    return 1;
-  },
-
-  clear: function() {
-    this.baseRows = [];
-    this.rows = [];
-    this.startPos = -1;
-    this.size = 0;
-    this.windowPos = 0;
-  },
-
-  isInRange: function(position) {
-    var lastRow=Math.min(this.totalRows, position + this.liveGrid.pageSize);
-    return (position >= this.startPos) && (lastRow <= this.endPos()); // && (this.size != 0);
-  },
-
-  endPos: function() {
-    return this.startPos + this.rows.length;
-  },
-
-  fetch: function(offset) {
-    Rico.log('fetch '+this.liveGrid.tableId+': offset='+offset);
-    this.applyFilters();
-    this.setTotalRows();
-    this.rcvdRowCount = true;
-    this.foundRowCount = true;
-    if (offset < 0) offset=0;
-    this.liveGrid.refreshContents(offset);
-    return;
-  },
-
-/**
- * @return a 2D array of buffer data representing the rows that are currently visible on the grid
- */
-  visibleRows: function() {
-    return this.rows.slice(this.windowStart,this.windowEnd);
-  },
-
-  setWindow: function(startrow, endrow) {
-    this.windowStart = startrow - this.startPos;  // position in the buffer of first visible row
-    Rico.log('setWindow '+this.liveGrid.tableId+': '+startrow+', '+endrow+', newstart='+this.windowStart);
-    this.windowEnd = Math.min(endrow,this.size);  // position in the buffer of last visible row containing data+1
-    this.windowPos = startrow;                    // position in the dataset of first visible row
-  },
-
-/**
- * @return true if bufRow is currently visible in the grid
- */
-  isVisible: function(bufRow) {
-    return bufRow < this.rows.length && bufRow >= this.windowStart && bufRow < this.windowEnd;
-  },
-  
-/**
- * takes a window row index and returns the corresponding buffer row index
- */
-  bufferRow: function(windowRow) {
-    return this.windowStart+windowRow;
-  },
-
-/**
- * @return buffer cell at the specified visible row/col index
- */
-  getWindowCell: function(windowRow,col) {
-    var bufrow=this.bufferRow(windowRow);
-    return this.isVisible(bufrow) && col < this.rows[bufrow].length ? this.rows[bufrow][col] : null;
-  },
-
-  getWindowAttr: function(windowRow,col) {
-    var bufrow=this.bufferRow(windowRow);
-    return this.attr && this.isVisible(bufrow) && col < this.attr[bufrow].length ? this.attr[bufrow][col] : null;
-  },
-
-  getWindowValue: function(windowRow,col) {
-    return this.getWindowCell(windowRow,col);
-  },
-
-  setWindowValue: function(windowRow,col,newval) {
-    var bufrow=this.bufferRow(windowRow);
-    if (bufrow >= this.windowEnd) return false;
-    return this.setValue(bufrow,col,newval);
-  },
-
-  getCell: function(bufRow,col) {
-    return bufRow < this.size ? this.rows[bufRow][col] : null;
-  },
-
-  getValue: function(bufRow,col) {
-    return this.getCell(bufRow,col);
-  },
-
-  setValue: function(bufRow,col,newval,newstyle) {
-    if (bufRow>=this.size) return false;
-    if (!this.rows[bufRow][col]) this.rows[bufRow][col]={};
-    this.rows[bufRow][col]=newval;
-    if (typeof newstyle=='string') this.rows[bufRow][col]._style=newstyle;
-    this.rows[bufRow][col].modified=true;
-    return true;
-  },
-
-  getRows: function(start, count) {
-    var begPos = start - this.startPos;
-    var endPos = Math.min(begPos + count,this.size);
-    var results = new Array();
-    for ( var i=begPos; i < endPos; i++ ) {
-      results.push(this.rows[i]);
-    }
-    return results;
-  },
-
-  applyFilters: function() {
-    var newRows=[],re=[];
-    var r,c,n,i,showRow,filtercnt;
-    var cols=this.liveGrid.columns;
-    for (n=0,filtercnt=0; n<cols.length; n++) {
-      c=cols[n];
-      if (c.filterType == Rico.ColumnConst.UNFILTERED) continue;
-      filtercnt++;
-      if (c.filterOp=='LIKE') re[n]=new RegExp(c.filterValues[0],'i');
-    }
-    Rico.log('applyFilters: # of filters='+filtercnt);
-    if (filtercnt==0) {
-      this.rows = this.baseRows;
-    } else {
-      for (r=0; r<this.baseRows.length; r++) {
-        showRow=true;
-        for (n=0; n<cols.length && showRow; n++) {
-          c=cols[n];
-          if (c.filterType == Rico.ColumnConst.UNFILTERED) continue;
-          switch (c.filterOp) {
-            case 'LIKE':
-              showRow=re[n].test(this.baseRows[r][n]);
-              break;
-            case 'EQ':
-              showRow=this.baseRows[r][n]==c.filterValues[0];
-              break;
-            case 'NE':
-              for (i=0; i<c.filterValues.length && showRow; i++)
-                showRow=this.baseRows[r][n]!=c.filterValues[i];
-              break;
-            case 'LE':
-              if (c.format.type=='number')
-                showRow=this.nan2zero(this.baseRows[r][n])<=this.nan2zero(c.filterValues[0]);
-              else
-                showRow=this.baseRows[r][n]<=c.filterValues[0];
-              break;
-            case 'GE':
-              if (c.format.type=='number')
-                showRow=this.nan2zero(this.baseRows[r][n])>=this.nan2zero(c.filterValues[0]);
-              else
-                showRow=this.baseRows[r][n]>=c.filterValues[0];
-              break;
-            case 'NULL':
-              showRow=this.baseRows[r][n]=='';
-              break;
-            case 'NOTNULL':
-              showRow=this.baseRows[r][n]!='';
-              break;
-          }
-        }
-        if (showRow) newRows.push(this.baseRows[r]);
-      }
-      this.rows = newRows;
-    }
-    this.rowcntContent = this.size = this.rows.length;
-  },
-
-  printAll: function(exportType) {
-    this.liveGrid.showMsg(Rico.getPhraseById('exportInProgress'));
-    Rico.runLater(10,this,'_printAll',exportType);  // allow message to paint
-  },
-
-/**
- * Support function for printAll()
- */
-  _printAll: function(exportType) {
-    this.liveGrid.exportStart();
-    this.exportBuffer(this.getRows(0,this.totalRows));
-    this.liveGrid.exportFinish(exportType);
-  },
-
-/**
- * Copies visible rows to a new window as a simple html table.
- */
-  printVisible: function(exportType) {
-    this.liveGrid.showMsg(Rico.getPhraseById('exportInProgress'));
-    Rico.runLater(10,this,'_printVisible',exportType);  // allow message to paint
-  },
-
-  _printVisible: function(exportType) {
-    this.liveGrid.exportStart();
-    this.exportBuffer(this.visibleRows());
-    this.liveGrid.exportFinish(exportType);
-  },
-
-/**
- * Send all rows to print/export window
- */
-  exportBuffer: function(rows,startPos) {
-    var r,c,v,col,exportText;
-    Rico.log("exportBuffer: "+rows.length+" rows");
-    var exportStyles=this.liveGrid.getExportStyles(this.liveGrid.tbody[0]);
-    var tdstyle=[];
-    var totalcnt=startPos || 0;
-    var cols=this.liveGrid.columns;
-    for (c=0; c<cols.length; c++) {
-      if (cols[c].visible) tdstyle[c]=this.liveGrid.exportStyle(cols[c].cell(0),exportStyles);  // assumes row 0 style applies to all rows
-    }
-    for(r=0; r < rows.length; r++) {
-      exportText='';
-      for (c=0; c<cols.length; c++) {
-        if (!cols[c].visible) continue;
-        col=cols[c];
-        col.expStyle=tdstyle[c];
-        v=col._export(rows[r][c],rows[r]);
-        if (v=='') v='&nbsp;';
-        exportText+="<td style='"+col.expStyle+"'>"+v+"</td>";
-      }
-      this.liveGrid.exportRows.push(exportText);
-      totalcnt++;
-      if (totalcnt % 10 == 0) window.status=Rico.getPhraseById('exportStatus',totalcnt);
-    }
-  }
-
-};
-
-
-// Rico.LiveGrid -----------------------------------------------------
-
-Rico.LiveGrid = function(tableId, buffer, options) {
-  this.initialize(tableId, buffer, options);
-}
-
-/** 
- * @lends Rico.LiveGrid#
- * @property tableId id string for this grid
- * @property options the options object passed to the constructor extended with defaults
- * @property buffer the buffer object containing the data for this grid
- * @property columns array of {@link Rico.LiveGridColumn} objects
- */
-Rico.LiveGrid.prototype = {
-/**
- * @class Buffered LiveGrid component
- * @extends Rico.GridCommon
- * @constructs
- */
-  initialize: function( tableId, buffer, options ) {
-    Rico.extend(this, Rico.GridCommon);
-    Rico.extend(this, Rico.LiveGridMethods);
-    this.baseInit();
-    this.tableId = tableId;
-    this.buffer = buffer;
-    this.actionId='_action_'+tableId;
-    Rico.setDebugArea(tableId+"_debugmsgs");    // if used, this should be a textarea
-
-    Rico.extend(this.options, {
-      visibleRows      : -3,    // -1 or 'window'=size grid to client window; -2 or 'data'=size grid to min(window,data); -3 or 'body'=size so body does not have a scrollbar; -4 or 'parent'=size to parent element (e.g. if grid is inside a div)
-      frozenColumns    : 0,
-      offset           : 0,     // first row to be displayed
-      prefetchBuffer   : true,  // load table on page load?
-      minPageRows      : 2,
-      maxPageRows      : 50,
-      canSortDefault   : true,  // can be overridden in the column specs
-      canFilterDefault : buffer.options.canFilter, // can be overridden in the column specs
-      canHideDefault   : true,  // can be overridden in the column specs
-
-      // highlight & selection parameters
-      highlightElem    : 'none',// what gets highlighted/selected (cursorRow, cursorCell, menuRow, menuCell, selection, or none)
-      highlightSection : 3,     // which section gets highlighted (frozen=1, scrolling=2, all=3, none=0)
-      highlightMethod  : 'class', // outline, class, both (outline is less CPU intensive on the client)
-      highlightClass   : Rico.theme.gridHighlightClass || 'ricoLG_selection',
-
-      // export/print parameters
-      maxPrint         : 5000,  // max # of rows that can be printed/exported, 0=disable print/export feature
-
-      // heading parameters
-      headingSort      : 'link', // link: make headings a link that will sort column, hover: make headings a hoverset, none: events on headings are disabled
-      hdrIconsFirst    : true,   // true: put sort & filter icons before header text, false: after
-      filterImg        : 'filtercol.gif'
-    });
-    // other options:
-    //   sortCol: initial sort column
-
-    var self=this;
-    this.options.sortHandler = function() { self.sortHandler(); };
-    this.options.filterHandler = function() { self.filterHandler(); };
-    this.options.onRefreshComplete = function(firstrow,lastrow) { self.bookmarkHandler(firstrow,lastrow); };
-    this.options.rowOverHandler = Rico.eventHandle(this,'rowMouseOver');
-    this.options.mouseDownHandler = Rico.eventHandle(this,'selectMouseDown');
-    this.options.mouseOverHandler = Rico.eventHandle(this,'selectMouseOver');
-    this.options.mouseUpHandler  = Rico.eventHandle(this,'selectMouseUp');
-    Rico.extend(this.options, options || {});
-
-    switch (typeof this.options.visibleRows) {
-      case 'string':
-        this.sizeTo=this.options.visibleRows;
-        switch (this.options.visibleRows) {
-          case 'data':   this.options.visibleRows=-2; break;
-          case 'body':   this.options.visibleRows=-3; break;
-          case 'parent': this.options.visibleRows=-4; break;
-          case 'datamax':this.options.visibleRows=-5; break;
-          default:       this.options.visibleRows=-1; break;
-        }
-        break;
-      case 'number':
-        switch (this.options.visibleRows) {
-          case -1: this.sizeTo='window'; break;
-          case -2: this.sizeTo='data'; break;
-          case -3: this.sizeTo='body'; break;
-          case -4: this.sizeTo='parent'; break;
-          case -5: this.sizeTo='datamax'; break;
-          default: this.sizeTo='fixed'; break;
-        }
-        break;
-      default:
-        this.sizeTo='body';
-        this.options.visibleRows=-3;
-        break;
-    }
-    this.highlightEnabled=this.options.highlightSection>0;
-    this.pageSize=0;
-    this.createTables();
-    if (this.headerColCnt==0) {
-      alert('ERROR: no columns found in "'+this.tableId+'"');
-      return;
-    }
-    this.createColumnArray('LiveGridColumn');
-    if (this.options.headingSort=='hover')
-      this.createHoverSet();
-
-    this.bookmark=document.getElementById(this.tableId+"_bookmark");
-    this.sizeDivs();
-    var filterUIrow=this.buffer.options.canFilter ? this.options.FilterLocation : false;
-    if (typeof(filterUIrow)=='number' && filterUIrow<0)
-      filterUIrow=this.addHeadingRow('ricoLG_FilterRow');
-    this.createDataCells(this.options.visibleRows);
-    if (this.pageSize == 0) return;
-    this.buffer.registerGrid(this);
-    if (this.buffer.setBufferSize) this.buffer.setBufferSize(this.pageSize);
-    this.scrollTimeout = null;
-    this.lastScrollPos = 0;
-    this.attachMenuEvents();
-
-    // preload the images...
-    new Image().src = Rico.imgDir+this.options.filterImg;
-    Rico.log("images preloaded");
-
-    this.setSortUI( this.options.sortCol, this.options.sortDir );
-    this.setImages();
-    if (this.listInvisible().length==this.columns.length)
-      this.columns[0].showColumn();
-    this.sizeDivs();
-    this.scrollDiv.style.display="";
-    if (this.buffer.totalRows>0)
-      this.updateHeightDiv();
-    if (this.options.prefetchBuffer) {
-      if (this.bookmark) this.bookmark.innerHTML = Rico.getPhraseById('bookmarkLoading');
-      if (this.options.canFilterDefault && this.options.getQueryParms)
-        this.checkForFilterParms();
-      this.scrollToRow(this.options.offset);
-      this.buffer.fetch(this.options.offset);
-    }
-    if (typeof(filterUIrow)=='number')
-      this.createFilters(filterUIrow);
-    this.scrollEventFunc=Rico.eventHandle(this,'handleScroll');
-    this.wheelEventFunc=Rico.eventHandle(this,'handleWheel');
-    this.wheelEvent=(Rico.isIE || Rico.isOpera || Rico.isWebKit) ? 'mousewheel' : 'DOMMouseScroll';
-    if (this.options.offset && this.options.offset < this.buffer.totalRows)
-      Rico.runLater(50,this,'scrollToRow',this.options.offset);  // Safari requires a delay
-    this.pluginScroll();
-    this.setHorizontalScroll();
-    Rico.log("setHorizontalScroll done");
-    if (this.options.windowResize)
-      Rico.runLater(100,this,'pluginWindowResize');
-    Rico.log("initialize complete for "+this.tableId);
-  }
-};
-
-
-Rico.LiveGridMethods = {
-/** @lends Rico.LiveGrid# */
-
-  createHoverSet: function() {
-    var hdrs=[];
-    for( var c=0; c < this.headerColCnt; c++ ) {
-      if (this.columns[c].sortable) {\r
-        hdrs.push(this.columns[c].hdrCellDiv);
-      }
-    }
-    this.hoverSet = new Rico.HoverSet(hdrs);
-  },
-
-  checkForFilterParms: function() {
-    var s=window.location.search;
-    if (s.charAt(0)=='?') s=s.substring(1);
-    var pairs = s.split('&');
-    for (var i=0; i<pairs.length; i++) {
-      if (pairs[i].match(/^f\[\d+\]/)) {
-        this.buffer.options.requestParameters.push(pairs[i]);
-      }
-    }
-  },
-
-/**
- * Refreshes a detail grid from a master grid
- * @returns row index on master table on success, -1 otherwise
- */
-  drillDown: function(e,masterColNum,detailColNum) {
-    var cell=Rico.eventElement(e || window.event);
-    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
-    if (!cell) return -1;
-    var idx=this.winCellIndex(cell);
-    if (idx.row >= this.buffer.totalRows) return -1
-    this.unhighlight();
-    this.menuIdx=idx;  // ensures selection gets cleared when menu is displayed
-    this.highlight(idx);
-    var drillValue=this.buffer.getWindowCell(idx.row,masterColNum);
-    for (var i=3; i<arguments.length; i++)
-      arguments[i].setDetailFilter(detailColNum,drillValue);
-    return idx.row;
-  },
-
-/**
- * set filter on a detail grid that is in a master-detail relationship
- */
-  setDetailFilter: function(colNumber,filterValue) {
-    var c=this.columns[colNumber];
-    c.format.ColData=filterValue;
-    c.setSystemFilter('EQ',filterValue);
-  },
-
-/**
- * Create one table for frozen columns and one for scrolling columns.
- * Also create div's to contain them.
- * @returns true on success
- */
-  createTables: function() {
-    var insertloc,hdrSrc,i;
-    var table = document.getElementById(this.tableId) || document.getElementById(this.tableId+'_outerDiv');
-    if (!table) return false;
-    if (table.tagName.toLowerCase()=='table') {
-      var theads=table.getElementsByTagName("thead");
-      if (theads.length == 1) {
-        Rico.log("createTables: using thead section, id="+this.tableId);
-        if (this.options.PanelNamesOnTabHdr && this.options.panels) {
-          var r=theads[0].insertRow(0);
-          this.insertPanelNames(r, 0, this.options.frozenColumns, 'ricoFrozen');
-          this.insertPanelNames(r, this.options.frozenColumns, this.options.columnSpecs.length);
-        }
-        hdrSrc=theads[0].rows;
-      } else {
-        Rico.log("createTables: using tbody section, id="+this.tableId);
-        hdrSrc=new Array(table.rows[0]);
-      }
-      insertloc=table;
-    } else if (this.options.columnSpecs.length > 0) {
-      if (!table.id.match(/_outerDiv$/)) insertloc=table;
-      Rico.log("createTables: inserting at "+table.tagName+", id="+this.tableId);
-    } else {
-      alert("ERROR!\n\nUnable to initialize '"+this.tableId+"'\n\nLiveGrid terminated");
-      return false;
-    }
-
-    this.createDivs();
-    this.scrollContainer = this.createDiv("scrollContainer",this.structTabLR);
-    this.scrollContainer.appendChild(this.scrollDiv); // move scrollDiv
-    this.scrollTab = this.createDiv("scrollTab",this.scrollContainer);
-    this.shadowDiv  = this.createDiv("shadow",this.scrollDiv);
-    this.shadowDiv.style.direction='ltr';  // avoid FF bug
-    this.scrollDiv.style.display="none";
-    this.scrollDiv.scrollTop=0;
-    if (this.options.highlightMethod!='class') {
-      this.highlightDiv=[];
-      switch (this.options.highlightElem) {
-        case 'menuRow':
-        case 'cursorRow':
-          this.highlightDiv[0] = this.createDiv("highlight",this.outerDiv);
-          this.highlightDiv[0].style.display="none";
-          break;
-        case 'menuCell':
-        case 'cursorCell':
-          for (i=0; i<2; i++) {
-            this.highlightDiv[i] = this.createDiv("highlight",i==0 ? this.frozenTabs : this.scrollTab);
-            this.highlightDiv[i].style.display="none";
-            this.highlightDiv[i].id+=i;
-          }
-          break;
-        case 'selection':
-          // create one div for each side of the rectangle
-          var parentDiv=this.options.highlightSection==1 ? this.frozenTabs : this.scrollTab;
-          for (i=0; i<4; i++) {
-            this.highlightDiv[i] = this.createDiv("highlight",parentDiv);
-            this.highlightDiv[i].style.display="none";
-            this.highlightDiv[i].style.overflow="hidden";
-            this.highlightDiv[i].id+=i;
-            this.highlightDiv[i].style[i % 2==0 ? 'height' : 'width']="0px";
-          }
-          break;
-      }
-    }
-
-    // create new tables
-    for (i=0; i<3; i++) {
-      this.tabs[i] = document.createElement("table");
-      this.tabs[i].className = 'ricoLG_table';
-      this.tabs[i].border=0;
-      this.tabs[i].cellPadding=0;
-      this.tabs[i].cellSpacing=0;
-      this.tabs[i].id = this.tableId+"_tab"+i;
-    }
-    // set headings
-    for (i=0; i<2; i++) {
-      this.thead[i]=this.tabs[i].createTHead();
-      Rico.addClass(this.tabs[i],'ricoLG_top');
-      //this.thead[i].className='ricoLG_top';
-      if (Rico.theme.gridheader) Rico.addClass(this.thead[i],Rico.theme.gridheader);
-    }
-    // set bodies
-    for (i=0; i<2; i++) {
-      this.tbody[i]=Rico.getTBody(this.tabs[i==0?0:2]);
-      this.tbody[i].className='ricoLG_bottom';
-      if (Rico.theme.gridcontent) Rico.addClass(this.tbody[i],Rico.theme.gridcontent);
-      this.tbody[i].insertRow(-1);
-    }
-    this.frozenTabs.appendChild(this.tabs[0]);
-    this.innerDiv.appendChild(this.tabs[1]);
-    this.scrollTab.appendChild(this.tabs[2]);
-    if (insertloc) insertloc.parentNode.insertBefore(this.outerDiv,insertloc);
-    if (hdrSrc) {
-      this.headerColCnt = this.getColumnInfo(hdrSrc);
-      this.loadHdrSrc(hdrSrc);
-    } else {
-      this.createHdr(0,0,this.options.frozenColumns);
-      this.createHdr(1,this.options.frozenColumns,this.options.columnSpecs.length);
-      if (this.options.PanelNamesOnTabHdr && this.options.panels) {
-        this.insertPanelNames(this.thead[0].insertRow(0), 0, this.options.frozenColumns);
-        this.insertPanelNames(this.thead[1].insertRow(0), this.options.frozenColumns, this.options.columnSpecs.length);
-      }
-      for (i=0; i<2; i++)
-        this.headerColCnt = this.getColumnInfo(this.thead[i].rows);
-    }
-    for( var c=0; c < this.headerColCnt; c++ )
-      this.tbody[c<this.options.frozenColumns ? 0 : 1].rows[0].insertCell(-1);
-    if (insertloc) table.parentNode.removeChild(table);
-    Rico.log('createTables end');
-    return true;
-  },
-
-  createDataCells: function(visibleRows) {
-    if (visibleRows < 0) {
-      for (var i=0; i<this.options.minPageRows; i++)
-        this.appendBlankRow();
-      this.sizeDivs();
-      this.autoAppendRows(this.remainingHt());
-    } else {
-      for( var r=0; r < visibleRows; r++ )
-        this.appendBlankRow();
-    }
-    var s=this.options.highlightSection;
-    if (s & 1) this.attachHighlightEvents(this.tbody[0]);
-    if (s & 2) this.attachHighlightEvents(this.tbody[1]);
-  },
-
-/**
- * @param colnum column number
- * @return id string for a filter element
- */
-  filterId: function(colnum) {
-    return 'RicoFilter_'+this.tableId+'_'+colnum;
-  },
-
-/**
- * Create filter elements in heading
- * Reads this.columns[].filterUI to determine type of filter element for each column (t=text box, s=select list, c=custom)
- * @param r heading row where filter elements will be placed
- */
-  createFilters: function(r) {
-    for( var c=0; c < this.headerColCnt; c++ ) {
-      var col=this.columns[c];
-      var fmt=col.format;
-      if (typeof fmt.filterUI!='string') continue;
-      var cell=this.hdrCells[r][c].cell;
-      var field,name=this.filterId(c);\r
-      var divs=cell.getElementsByTagName('div');
-      // copy text alignment from data cell
-      var align=Rico.getStyle(this.cell(0,c),'textAlign');
-      divs[1].style.textAlign=align;
-      switch (fmt.filterUI.charAt(0)) {
-        case 't':
-          // text field
-          field=Rico.createFormField(divs[1],'input','text',name,name);
-          var size=fmt.filterUI.match(/\d+/);
-          field.maxLength=fmt.Length || 50;\r
-          field.size=size ? parseInt(size,10) : 10;
-          divs[1].appendChild(Rico.clearButton(Rico.eventHandle(col,'filterClear')));
-          if (col.filterType==Rico.ColumnConst.USERFILTER && col.filterOp=='LIKE') {
-            var v=col.filterValues[0];
-            if (v.charAt(0)=='*') v=v.substr(1);
-            if (v.slice(-1)=='*') v=v.slice(0,-1);
-            field.value=v;
-            col.lastKeyFilter=v;
-          }
-          Rico.eventBind(field,'keyup',Rico.eventHandle(col,'filterKeypress'),false);
-          col.filterField=field;\r
-          break;\r
-        case 'm':
-          // multi-select
-        case 's':
-          // drop-down select
-          field=Rico.createFormField(divs[1],'select',null,name);\r
-          Rico.addSelectOption(field,this.options.FilterAllToken,Rico.getPhraseById("filterAll"));\r
-          col.filterField=field;
-          var options={};\r
-          Rico.extend(options, this.buffer.ajaxOptions);
-          var colnum=typeof(fmt.filterCol)=='number' ? fmt.filterCol : c;
-          options.parameters = {id: this.tableId, distinct:colnum};
-          options.parameters[this.actionId]="query";
-          options.onComplete = this.filterValuesUpdateFunc(c);
-          new Rico.ajaxRequest(this.buffer.dataSource, options);
-          break;\r
-        case 'c':
-          // custom
-          if (typeof col._createFilters == 'function')
-            col._createFilters(divs[1], name);
-          break;
-      }
-    }
-    this.initFilterImage(r);
-  },
-  
-  filterValuesUpdateFunc: function(colnum) {
-    var self=this;
-    return function (request) { self.filterValuesUpdate(colnum,request); };
-  },
-
-/**
- * update select list filter with values in AJAX response
- * @returns true on success
- */
-  filterValuesUpdate: function(colnum,request) {
-    var response = request.responseXML.getElementsByTagName("ajax-response");
-    Rico.log("filterValuesUpdate: "+request.status);
-    if (response == null || response.length != 1) return false;
-    response=response[0];
-    var error = response.getElementsByTagName('error');
-    if (error.length > 0) {
-      Rico.log("Data provider returned an error:\n"+Rico.getContentAsString(error[0],this.buffer.isEncoded));
-      alert(Rico.getPhraseById("requestError",Rico.getContentAsString(error[0],this.buffer.isEncoded)));
-      return false;
-    }\r
-    response=response.getElementsByTagName('response')[0];\r
-    var rowsElement = response.getElementsByTagName('rows')[0];\r
-    var col=this.columns[parseInt(colnum,10)];
-    var rows = this.buffer.dom2jstable(rowsElement);\r
-    var c,opt,v;
-    if (col.filterType==Rico.ColumnConst.USERFILTER && col.filterOp=='EQ') v=col.filterValues[0];
-    Rico.log('filterValuesUpdate: col='+colnum+' rows='+rows.length);
-    switch (col.format.filterUI.charAt(0)) {
-      case 'm':
-        // multi-select
-        col.mFilter = document.body.appendChild(document.createElement("div"));
-        col.mFilter.className = 'ricoLG_mFilter'
-        Rico.hide(col.mFilter);
-        var contentDiv = col.mFilter.appendChild(document.createElement("div"));
-        contentDiv.className = 'ricoLG_mFilter_content'
-        var buttonDiv = col.mFilter.appendChild(document.createElement("div"));
-        buttonDiv.className = 'ricoLG_mFilter_button'
-        col.mFilterButton=buttonDiv.appendChild(document.createElement("button"));
-        col.mFilterButton.innerHTML=Rico.getPhraseById("ok");
-        var eventName=Rico.isWebKit ? 'mousedown' : 'click';
-        Rico.eventBind(col.filterField,eventName,Rico.eventHandle(col,'mFilterSelectClick'));
-        Rico.eventBind(col.mFilterButton,'click',Rico.eventHandle(col,'mFilterFinish'));
-        //col.filterField.options[0].text=$('AllLabel').innerHTML;
-        tab = contentDiv.appendChild(document.createElement("table"));
-        tab.border=0;
-        tab.cellPadding=2;
-        tab.cellSpacing=0;
-        //tbody=(tab.tBodies.length==0) ? tab.appendChild(document.createElement("tbody")) : tab.tBodies[0];
-        var baseId=this.filterId(colnum)+'_';
-        this.createMFilterItem(tab,this.options.FilterAllToken,Rico.getPhraseById("filterAll"),baseId+'all',Rico.eventHandle(col,'mFilterAllClick'));
-        var handle=Rico.eventHandle(col,'mFilterOtherClick')
-        for (var i=0; i<rows.length; i++) {
-          if (rows[i].length>0) {
-            c=rows[i][0];
-            this.createMFilterItem(tab,c,c || Rico.getPhraseById("filterBlank"),baseId+i,handle);
-          }
-        }
-        col.mFilterInputs=contentDiv.getElementsByTagName('input');
-        col.mFilterLabels=contentDiv.getElementsByTagName('label');
-        col.mFilterFocus=col.mFilterInputs.length ? col.mFilterInputs[0] : col.mFilterButton;
-        break;
-
-      case 's':
-        // drop-down select
-        for (var i=0; i<rows.length; i++) {
-          if (rows[i].length>0) {
-            c=rows[i][0];
-            opt=Rico.addSelectOption(col.filterField,c,c || Rico.getPhraseById("filterBlank"));
-            if (col.filterType==Rico.ColumnConst.USERFILTER && c==v) opt.selected=true;
-          }
-        }
-        Rico.eventBind(col.filterField,'change',Rico.eventHandle(col,'filterChange'));
-        break;
-    }
-    return true;\r
-  },
-  
-  createMFilterItem: function(table,code,description,id,eventHandle) {
-    var tr=table.insertRow(-1);
-    tr.vAlign='top';
-    if (tr.rowIndex % 2 == 1) tr.className='ricoLG_mFilter_oddrow';
-    var td1=tr.insertCell(-1)
-    var td2=tr.insertCell(-1)
-    var field=Rico.createFormField(td1,'input','checkbox',id);
-    field.value=code;
-    field.checked=true;
-    var label = td2.appendChild(document.createElement("label"));
-    label.htmlFor = id;
-    label.innerHTML=description;
-    Rico.eventBind(field,'click',eventHandle);
-  },
-
-  unplugHighlightEvents: function() {
-    var s=this.options.highlightSection;
-    if (s & 1) this.detachHighlightEvents(this.tbody[0]);
-    if (s & 2) this.detachHighlightEvents(this.tbody[1]);
-  },
-
-/**
- * place panel names on first row of grid header (used by LiveGridForms)
- */
-  insertPanelNames: function(r,start,limit,cellClass) {
-    Rico.log('insertPanelNames: start='+start+' limit='+limit);
-    r.className='ricoLG_hdg';
-    var lastIdx=-1, span, newCell=null, spanIdx=0;
-    for( var c=start; c < limit; c++ ) {
-      if (lastIdx == this.options.columnSpecs[c].panelIdx) {
-        span++;
-      } else {
-        if (newCell) newCell.colSpan=span;
-        newCell = r.insertCell(-1);
-        if (cellClass) newCell.className=cellClass;
-        span=1;
-        lastIdx=this.options.columnSpecs[c].panelIdx;
-        newCell.innerHTML=this.options.panels[lastIdx];
-      }
-    }
-    if (newCell) newCell.colSpan=span;
-  },
-
-/**
- * create grid header for table i (if none was provided)
- */
-  createHdr: function(i,start,limit) {
-    Rico.log('createHdr: i='+i+' start='+start+' limit='+limit);
-    var mainRow = this.thead[i].insertRow(-1);
-    mainRow.id=this.tableId+'_tab'+i+'h_main';
-    mainRow.className='ricoLG_hdg';
-    for( var c=start; c < limit; c++ ) {
-      var newCell = mainRow.insertCell(-1);
-      newCell.innerHTML=this.options.columnSpecs[c].Hdg;
-    }
-  },
-
-/**
- * move header cells in original table to grid
- */
-  loadHdrSrc: function(hdrSrc) {
-    var i,h,c,r,newrow,cells;
-    Rico.log('loadHdrSrc start');
-    for (i=0; i<2; i++) {
-      for (r=0; r<hdrSrc.length; r++) {
-        newrow = this.thead[i].insertRow(-1);
-        newrow.className='ricoLG_hdg '+this.tableId+'_hdg'+r;
-      }
-    }
-    if (hdrSrc.length==1) {
-      cells=hdrSrc[0].cells;
-      for (c=0; cells.length > 0; c++)
-        this.thead[c<this.options.frozenColumns ? 0 : 1].rows[0].appendChild(cells[0]);
-    } else {
-      for (r=0; r<hdrSrc.length; r++) {
-        cells=hdrSrc[r].cells;
-        for (c=0,h=0; cells.length > 0; c++) {
-          if (cells[0].className=='ricoFrozen') {
-            if (r==this.headerRowIdx) this.options.frozenColumns=c+1;
-          } else {
-            h=1;
-          }
-          this.thead[h].rows[r].appendChild(cells[0]);
-        }
-      }
-    }
-    Rico.log('loadHdrSrc end');
-  },
-
-/**
- * Size div elements
- */
-  sizeDivs: function() {
-    Rico.log('sizeDivs: '+this.tableId);
-    //this.cancelMenu();
-    this.unhighlight();
-    this.baseSizeDivs();
-    var firstVisible=this.firstVisible();
-    if (this.pageSize == 0 || firstVisible < 0) return;
-    var totRowHt=this.columns[firstVisible].dataColDiv.offsetHeight;
-    this.rowHeight = Math.round(totRowHt/this.pageSize);
-    var scrHt=this.dataHt;
-    if (this.scrTabWi0 == this.scrTabWi) {
-      // no scrolling columns - horizontal scroll bar not needed
-      this.innerDiv.style.height=(this.hdrHt+1)+'px';
-      this.scrollDiv.style.overflowX='hidden';
-    } else {
-      this.scrollDiv.style.overflowX='scroll';
-      scrHt+=this.options.scrollBarWidth;
-    }
-    this.scrollDiv.style.height=scrHt+'px';
-    this.innerDiv.style.width=(this.scrWi)+'px';
-    this.scrollTab.style.width=(this.scrWi-this.options.scrollBarWidth)+'px';
-    //this.resizeDiv.style.height=this.frozenTabs.style.height=this.innerDiv.style.height=(this.hdrHt+this.dataHt+1)+'px';
-    this.resizeDiv.style.height=(this.hdrHt+this.dataHt+1)+'px';
-    Rico.log('sizeDivs scrHt='+scrHt+' innerHt='+this.innerDiv.style.height+' rowHt='+this.rowHeight+' pageSize='+this.pageSize);
-    var pad=(this.scrWi-this.scrTabWi < this.options.scrollBarWidth) ? 2 : 0;
-    this.shadowDiv.style.width=(this.scrTabWi+pad)+'px';
-    this.outerDiv.style.height=(this.hdrHt+scrHt)+'px';
-    this.setHorizontalScroll();
-  },
-
-  setHorizontalScroll: function() {
-    var newLeft=(-this.scrollDiv.scrollLeft)+'px';
-    this.tabs[1].style.marginLeft=newLeft;
-    this.tabs[2].style.marginLeft=newLeft;
-  },
-
-  remainingHt: function() {
-    var tabHt;
-    var winHt=Rico.windowHeight();
-    var margin=Rico.isIE ? 15 : 10;
-    // if there is a horizontal scrollbar take it into account
-    if (!Rico.isIE && window.frameElement && window.frameElement.scrolling=='yes' && this.sizeTo!='parent') margin+=this.options.scrollBarWidth;
-    switch (this.sizeTo) {
-      case 'window':
-        var divTop=Rico.cumulativeOffset(this.outerDiv).top;
-        tabHt=Math.max(this.tabs[0].offsetHeight,this.tabs[1].offsetHeight);
-        Rico.log("remainingHt, winHt="+winHt+' tabHt='+tabHt+' gridY='+divTop);
-        return winHt-divTop-tabHt-this.options.scrollBarWidth-margin;  // allow for scrollbar and some margin
-      case 'parent':
-        var offset=this.offsetFromParent(this.outerDiv);
-        tabHt=Math.max(this.tabs[0].offsetHeight,this.tabs[1].offsetHeight);
-        if (Rico.isIE) Rico.hide(this.outerDiv);
-        var parentHt=this.outerDiv.parentNode.offsetHeight;
-        if (Rico.isIE) Rico.show(this.outerDiv);
-        Rico.log("remainingHt, parentHt="+parentHt+' gridY='+offset+' winHt='+winHt+' tabHt='+tabHt);
-        return parentHt - tabHt - offset - this.options.scrollBarWidth;
-      case 'data':
-      case 'body':
-        var bodyHt=Rico.isIE ? document.body.scrollHeight : document.body.offsetHeight;
-        //alert("remainingHt\n document.height="+document.height+"\n body.offsetHeight="+document.body.offsetHeight+"\n body.scrollHeight="+document.body.scrollHeight+"\n documentElement.scrollHeight="+document.documentElement.scrollHeight);
-        var remHt=winHt-bodyHt-margin;
-        if (!Rico.isWebKit) remHt-=this.options.scrollBarWidth;
-        Rico.log("remainingHt, winHt="+winHt+' pageHt='+bodyHt+' remHt='+remHt);
-        return remHt;
-      default:
-        tabHt=Math.max(this.tabs[0].offsetHeight,this.tabs[1].offsetHeight);
-        Rico.log("remainingHt, winHt="+winHt+' tabHt='+tabHt);
-        if (this.sizeTo.slice(-1)=='%') winHt*=parseFloat(this.sizeTo)/100.0;
-        else if (this.sizeTo.slice(-2)=='px') winHt=parseInt(this.sizeTo,10);
-        return winHt-tabHt-this.options.scrollBarWidth-margin;  // allow for scrollbar and some margin
-    }
-  },
-
-  offsetFromParent: function(element) {
-    var valueT = 0;
-    var elParent=element.parentNode;
-    do {
-      //Rico.log("offsetFromParent: "+element.tagName+' id='+element.id+' otop='+element.offsetTop);
-      valueT += element.offsetTop  || 0;
-      element = element.offsetParent;
-      if (!element || element==null) break;
-      var p = Rico.getStyle(element, 'position');
-      if (element.tagName=='BODY' || element.tagName=='HTML' || p=='absolute') return valueT-elParent.offsetTop;
-    } while (element != elParent);
-    return valueT;
-  },
-
-  adjustPageSize: function() {
-    var remHt=this.remainingHt();
-    Rico.log('adjustPageSize remHt='+remHt+' lastRow='+this.lastRowPos);
-    if (remHt > this.rowHeight)
-      this.autoAppendRows(remHt);
-    else if (remHt < 0 || this.sizeTo=='data')
-      this.autoRemoveRows(-remHt);
-  },
-  
-  setPageSize: function(newRowCount) {
-    newRowCount=Math.min(newRowCount,this.options.maxPageRows);
-    newRowCount=Math.max(newRowCount,this.options.minPageRows);
-    this.sizeTo='fixed';
-    var oldSize=this.pageSize;
-    while (this.pageSize > newRowCount) {
-      this.removeRow();
-    }
-    while (this.pageSize < newRowCount) {
-      this.appendBlankRow();
-    }
-    this.finishResize(oldSize);
-  },
-
-  pluginWindowResize: function() {
-    Rico.log("pluginWindowResize");
-    this.resizeWindowHandler=Rico.eventHandle(this,'resizeWindow');
-    Rico.eventBind(window, "resize", this.resizeWindowHandler, false);
-  },
-
-  unplugWindowResize: function() {
-    if (!this.resizeWindowHandler) return;
-    Rico.eventUnbind(window,"resize", this.resizeWindowHandler, false);
-    this.resizeWindowHandler=null;
-  },
-
-  resizeWindow: function() {
-    Rico.log('resizeWindow '+this.tableId+' lastRow='+this.lastRowPos);
-    if (this.resizeState=='finish') {
-      Rico.log('resizeWindow postponed');
-      this.resizeState='resize';
-      return;
-    }
-    if (!this.sizeTo || this.sizeTo=='fixed') {
-      this.sizeDivs();
-      return;
-    }
-    if (this.sizeTo=='parent' && Rico.getStyle(this.outerDiv.parentNode,'display') == 'none') return;
-    var oldSize=this.pageSize;
-    this.adjustPageSize();
-    this.finishResize(oldSize);
-  },
-
-  finishResize: function(oldSize) {
-    if (this.pageSize > oldSize && this.buffer.totalRows>0) {
-      this.isPartialBlank=true;
-      var adjStart=this.adjustRow(this.lastRowPos);
-      this.buffer.fetch(adjStart);
-    } else if (this.pageSize < oldSize) {
-      if (this.options.onRefreshComplete) this.options.onRefreshComplete(this.contentStartPos,this.contentStartPos+this.pageSize-1);  // update bookmark
-    }
-    this.resizeState='finish';
-    Rico.runLater(20,this,'finishResize2');
-    Rico.log('Resize '+this.tableId+' complete. old size='+oldSize+' new size='+this.pageSize);
-  },
-
-  finishResize2: function() {
-    this.sizeDivs();
-    this.updateHeightDiv();
-    if (this.resizeState=='resize') {
-      this.resizeWindow();
-    } else {
-      this.resizeState='';
-    }
-  },
-
-  topOfLastPage: function() {
-    return Math.max(this.buffer.totalRows-this.pageSize,0);
-  },
-
-  updateHeightDiv: function() {
-    var notdisp=this.topOfLastPage();
-    var ht = notdisp ? this.scrollDiv.clientHeight + Math.floor(this.rowHeight * (notdisp + 0.4)) : 1;
-    Rico.log("updateHeightDiv, ht="+ht+' scrollDiv.clientHeight='+this.scrollDiv.clientHeight+' rowsNotDisplayed='+notdisp);
-    this.shadowDiv.style.height=ht+'px';
-  },
-
-  autoRemoveRows: function(overage) {
-    if (!this.rowHeight) return;
-    var removeCnt=Math.ceil(overage / this.rowHeight);
-    if (this.sizeTo=='data')
-      removeCnt=Math.max(removeCnt,this.pageSize-this.buffer.totalRows);
-    Rico.log("autoRemoveRows overage="+overage+" removeCnt="+removeCnt);
-    for (var i=0; i<removeCnt; i++)
-      this.removeRow();
-  },
-
-  removeRow: function() {
-    if (this.pageSize <= this.options.minPageRows) return;
-    this.pageSize--;
-    for( var c=0; c < this.headerColCnt; c++ ) {
-      var cell=this.columns[c].cell(this.pageSize);
-      this.columns[c].dataColDiv.removeChild(cell);
-    }
-  },
-
-  autoAppendRows: function(overage) {
-    if (!this.rowHeight) return;
-    var addCnt=Math.floor(overage / this.rowHeight);
-    Rico.log("autoAppendRows overage="+overage+" cnt="+addCnt+" rowHt="+this.rowHeight);
-    for (var i=0; i<addCnt; i++) {
-      if (this.sizeTo=='data' && this.pageSize>=this.buffer.totalRows) break;
-      this.appendBlankRow();
-    }
-  },
-
-/**
- * on older systems, this can be fairly slow
- */
-  appendBlankRow: function() {
-    if (this.pageSize >= this.options.maxPageRows) return;
-    Rico.log("appendBlankRow #"+this.pageSize);
-    var cls=this.defaultRowClass(this.pageSize);
-    for( var c=0; c < this.headerColCnt; c++ ) {
-      var newdiv = document.createElement("div");
-      newdiv.className = 'ricoLG_cell '+cls;
-      newdiv.id=this.tableId+'_'+this.pageSize+'_'+c;
-      this.columns[c].dataColDiv.appendChild(newdiv);
-      if (this.columns[c].format.canDrag && Rico.registerDraggable)
-        Rico.registerDraggable( new Rico.LiveGridDraggable(this, this.pageSize, c), this.options.dndMgrIdx );
-      newdiv.innerHTML='&nbsp;';   // this seems to be required by IE
-      if (this.columns[c]._create) {
-        this.columns[c]._create(newdiv,this.pageSize);
-      }
-    }
-    this.pageSize++;
-  },
-
-  defaultRowClass: function(rownum) {
-    var cls
-    if (rownum % 2==0) {
-      cls='ricoLG_evenRow';
-      //if (Rico.theme.primary) cls+=' '+Rico.theme.primary;
-    } else {
-      cls='ricoLG_oddRow';
-      //if (Rico.theme.secondary) cls+=' '+Rico.theme.secondary;
-    }
-    return cls;
-  },
-
-  handleMenuClick: function(e) {
-    if (!this.menu) return;
-    this.cancelMenu();
-    this.unhighlight(); // in case highlighting was invoked externally
-    var idx;
-    var cell=Rico.eventElement(e);
-    if (cell.className=='ricoLG_highlightDiv') {
-      idx=this.highlightIdx;
-    } else {
-      cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
-      if (!cell) return;
-      idx=this.winCellIndex(cell);
-      if ((this.options.highlightSection & (idx.tabIdx+1))==0) return;
-    }
-    this.highlight(idx);
-    this.highlightEnabled=false;
-    if (this.hideScroll) this.scrollDiv.style.overflow="hidden";
-    this.menuIdx=idx;
-    if (!this.menu.div) this.menu.createDiv();
-    this.menu.liveGrid=this;
-    if (this.menu.buildGridMenu) {
-      var showMenu=this.menu.buildGridMenu(idx.row, idx.column, idx.tabIdx);
-      if (!showMenu) return;
-    }
-    if (this.options.highlightElem=='selection' && !this.isSelected(idx.cell)) {
-      this.selectCell(idx.cell);
-    }
-    var self=this;
-    this.menu.showmenu(e,function() { self.closeMenu(); });
-  },
-
-  closeMenu: function() {
-    if (!this.menuIdx) return;
-    if (this.hideScroll) this.scrollDiv.style.overflow="";
-    //this.unhighlight();
-    this.highlightEnabled=true;
-    this.menuIdx=null;
-  },
-
-/**
- * @return index of cell within the window
- */
-  winCellIndex: function(cell) {
-    var l=cell.id.lastIndexOf('_',cell.id.length);
-    var l2=cell.id.lastIndexOf('_',l-1)+1;
-    var c=parseInt(cell.id.substr(l+1));
-    var r=parseInt(cell.id.substr(l2,l));
-    return {row:r, column:c, tabIdx:this.columns[c].tabIdx, cell:cell};
-  },
-
-/**
- * @return index of cell within the dataset
- */
-  datasetIndex: function(cell) {
-    var idx=this.winCellIndex(cell);
-    idx.row+=this.buffer.windowPos;
-    idx.onBlankRow=(idx.row >= this.buffer.endPos());
-    return idx;
-  },
-
-  attachHighlightEvents: function(tBody) {
-    switch (this.options.highlightElem) {
-      case 'selection':
-        Rico.eventBind(tBody,"mousedown", this.options.mouseDownHandler, false);
-        /** @ignore */
-        tBody.ondrag = function () { return false; };
-        /** @ignore */
-        tBody.onselectstart = function () { return false; };
-        break;
-      case 'cursorRow':
-      case 'cursorCell':
-        Rico.eventBind(tBody,"mouseover", this.options.rowOverHandler, false);
-        break;
-    }
-  },
-
-  detachHighlightEvents: function(tBody) {
-    switch (this.options.highlightElem) {
-      case 'selection':
-        Rico.eventUnbind(tBody,"mousedown", this.options.mouseDownHandler, false);
-        tBody.ondrag = null;
-        tBody.onselectstart = null;
-        break;
-      case 'cursorRow':
-      case 'cursorCell':
-        Rico.eventUnbind(tBody,"mouseover", this.options.rowOverHandler, false);
-        break;
-    }
-  },
-
-/**
- * @return array of objects containing row/col indexes (index values are relative to the start of the window)
- */
-  getVisibleSelection: function() {
-    var cellList=[];
-    if (this.SelectIdxStart && this.SelectIdxEnd) {
-      var r1=Math.max(Math.min(this.SelectIdxEnd.row,this.SelectIdxStart.row)-this.buffer.startPos,this.buffer.windowStart);
-      var r2=Math.min(Math.max(this.SelectIdxEnd.row,this.SelectIdxStart.row)-this.buffer.startPos,this.buffer.windowEnd-1);
-      var c1=Math.min(this.SelectIdxEnd.column,this.SelectIdxStart.column);
-      var c2=Math.max(this.SelectIdxEnd.column,this.SelectIdxStart.column);
-      //Rico.log("getVisibleSelection "+r1+','+c1+' to '+r2+','+c2+' ('+this.SelectIdxStart.row+',startPos='+this.buffer.startPos+',windowPos='+this.buffer.windowPos+',windowEnd='+this.buffer.windowEnd+')');
-      for (var r=r1; r<=r2; r++) {
-        for (var c=c1; c<=c2; c++)
-          cellList.push({row:r-this.buffer.windowStart,column:c});
-      }
-    }
-    if (this.SelectCtrl) {
-      for (var i=0; i<this.SelectCtrl.length; i++) {
-        if (this.SelectCtrl[i].row>=this.buffer.windowStart && this.SelectCtrl[i].row<this.buffer.windowEnd)
-          cellList.push({row:this.SelectCtrl[i].row-this.buffer.windowStart,column:this.SelectCtrl[i].column});
-      }
-    }
-    return cellList;
-  },
-
-  updateSelectOutline: function() {
-    if (!this.SelectIdxStart || !this.SelectIdxEnd) return;
-    var r1=Math.max(Math.min(this.SelectIdxEnd.row,this.SelectIdxStart.row), this.buffer.windowStart);
-    var r2=Math.min(Math.max(this.SelectIdxEnd.row,this.SelectIdxStart.row), this.buffer.windowEnd-1);
-    if (r1 > r2) {
-      this.HideSelection();
-      return;
-    }
-    var c1=Math.min(this.SelectIdxEnd.column,this.SelectIdxStart.column);
-    var c2=Math.max(this.SelectIdxEnd.column,this.SelectIdxStart.column);
-    var top1=this.columns[c1].cell(r1-this.buffer.windowStart).offsetTop;
-    var cell2=this.columns[c1].cell(r2-this.buffer.windowStart);
-    var bottom2=cell2.offsetTop+cell2.offsetHeight;
-    var left1=this.columns[c1].dataCell.offsetLeft;
-    var left2=this.columns[c2].dataCell.offsetLeft;
-    var right2=left2+this.columns[c2].dataCell.offsetWidth;
-    //window.status='updateSelectOutline: '+r1+' '+r2+' top='+top1+' bot='+bottom2;
-    this.highlightDiv[0].style.top=this.highlightDiv[3].style.top=this.highlightDiv[1].style.top=(this.hdrHt+top1-1) + 'px';
-    this.highlightDiv[2].style.top=(this.hdrHt+bottom2-1)+'px';
-    this.highlightDiv[3].style.left=(left1-2)+'px';
-    this.highlightDiv[0].style.left=this.highlightDiv[2].style.left=(left1-1)+'px';
-    this.highlightDiv[1].style.left=(right2-1)+'px';
-    this.highlightDiv[0].style.width=this.highlightDiv[2].style.width=(right2-left1-1) + 'px';
-    this.highlightDiv[1].style.height=this.highlightDiv[3].style.height=(bottom2-top1) + 'px';
-    //this.highlightDiv[0].style.right=this.highlightDiv[2].style.right=this.highlightDiv[1].style.right=()+'px';
-    //this.highlightDiv[2].style.bottom=this.highlightDiv[3].style.bottom=this.highlightDiv[1].style.bottom=(this.hdrHt+bottom2) + 'px';
-    for (var i=0; i<4; i++)
-      this.highlightDiv[i].style.display='';
-  },
-
-  HideSelection: function() {
-    var i;
-    if (this.options.highlightMethod!='class') {
-      for (i=0; i<this.highlightDiv.length; i++)
-        this.highlightDiv[i].style.display='none';
-    }
-    if (this.options.highlightMethod!='outline') {
-      var cellList=this.getVisibleSelection();
-      Rico.log("HideSelection "+cellList.length);
-      for (i=0; i<cellList.length; i++)
-        this.unhighlightCell(this.columns[cellList[i].column].cell(cellList[i].row));
-    }
-  },
-
-  ShowSelection: function() {
-    if (this.options.highlightMethod!='class')
-      this.updateSelectOutline();
-    if (this.options.highlightMethod!='outline') {
-      var cellList=this.getVisibleSelection();
-      for (var i=0; i<cellList.length; i++)
-        this.highlightCell(this.columns[cellList[i].column].cell(cellList[i].row));
-    }
-  },
-
-  ClearSelection: function() {
-    Rico.log("ClearSelection");
-    this.HideSelection();
-    this.SelectIdxStart=null;
-    this.SelectIdxEnd=null;
-    this.SelectCtrl=[];
-  },
-
-  selectCell: function(cell) {
-    this.ClearSelection();
-    this.SelectIdxStart=this.SelectIdxEnd=this.datasetIndex(cell);
-    this.ShowSelection();
-  },
-
-  AdjustSelection: function(cell) {
-    var newIdx=this.datasetIndex(cell);
-    if (this.SelectIdxStart.tabIdx != newIdx.tabIdx) return;
-    this.HideSelection();
-    this.SelectIdxEnd=newIdx;
-    this.ShowSelection();
-  },
-
-  RefreshSelection: function() {
-    var cellList=this.getVisibleSelection();
-    for (var i=0; i<cellList.length; i++) {
-      this.columns[cellList[i].column].displayValue(cellList[i].row);
-    }
-  },
-
-  FillSelection: function(newVal,newStyle) {
-    if (this.SelectIdxStart && this.SelectIdxEnd) {
-      var r1=Math.min(this.SelectIdxEnd.row,this.SelectIdxStart.row);
-      var r2=Math.max(this.SelectIdxEnd.row,this.SelectIdxStart.row);
-      var c1=Math.min(this.SelectIdxEnd.column,this.SelectIdxStart.column);
-      var c2=Math.max(this.SelectIdxEnd.column,this.SelectIdxStart.column);
-      for (var r=r1; r<=r2; r++) {
-        for (var c=c1; c<=c2; c++) {
-          this.buffer.setValue(r,c,newVal,newStyle);
-        }
-      }
-    }
-    if (this.SelectCtrl) {
-      for (var i=0; i<this.SelectCtrl.length; i++) {
-        this.buffer.setValue(this.SelectCtrl[i].row,this.SelectCtrl[i].column,newVal,newStyle);
-      }
-    }
-    this.RefreshSelection();
-  },
-
-/**
- * Process mouse down event
- * @param e event object
- */
-  selectMouseDown: function(e) {
-    if (this.highlightEnabled==false) return true;
-    this.cancelMenu();
-    var cell=Rico.eventElement(e);
-    if (!Rico.eventLeftClick(e)) return true;
-    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
-    if (!cell) return true;
-    Rico.eventStop(e);
-    var newIdx=this.datasetIndex(cell);
-    if (newIdx.onBlankRow) return true;
-    Rico.log("selectMouseDown @"+newIdx.row+','+newIdx.column);
-    if (e.ctrlKey) {
-      if (!this.SelectIdxStart || this.options.highlightMethod!='class') return true;
-      if (!this.isSelected(cell)) {
-        this.highlightCell(cell);
-        this.SelectCtrl.push(this.datasetIndex(cell));
-      } else {
-        for (var i=0; i<this.SelectCtrl.length; i++) {
-          if (this.SelectCtrl[i].row==newIdx.row && this.SelectCtrl[i].column==newIdx.column) {
-            this.unhighlightCell(cell);
-            this.SelectCtrl.splice(i,1);
-            break;
-          }
-        }
-      }
-    } else if (e.shiftKey) {
-      if (!this.SelectIdxStart) return true;
-      this.AdjustSelection(cell);
-    } else {
-      this.selectCell(cell);
-      this.pluginSelect();
-    }
-    return false;
-  },
-
-  pluginSelect: function() {
-    if (this.selectPluggedIn) return;
-    var tBody=this.tbody[this.SelectIdxStart.tabIdx];
-    Rico.eventBind(tBody,"mouseover", this.options.mouseOverHandler, false);
-    Rico.eventBind(this.outerDiv,"mouseup",  this.options.mouseUpHandler,  false);
-    this.selectPluggedIn=true;
-  },
-
-  unplugSelect: function() {
-    if (!this.selectPluggedIn) return;
-    var tBody=this.tbody[this.SelectIdxStart.tabIdx];
-    Rico.eventUnbind(tBody,"mouseover", this.options.mouseOverHandler , false);
-    Rico.eventUnbind(this.outerDiv,"mouseup", this.options.mouseUpHandler , false);
-    this.selectPluggedIn=false;
-  },
-
-  selectMouseUp: function(e) {
-    this.unplugSelect();
-    var cell=Rico.eventElement(e);
-    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
-    if (!cell) return;
-    if (this.SelectIdxStart && this.SelectIdxEnd)
-      this.AdjustSelection(cell);
-    else
-      this.ClearSelection();
-  },
-
-  selectMouseOver: function(e) {
-    var cell=Rico.eventElement(e);
-    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
-    if (!cell) return;
-    this.AdjustSelection(cell);
-    Rico.eventStop(e);
-  },
-
-  isSelected: function(cell) {
-    if (this.options.highlightMethod!='outline') return Rico.hasClass(cell,this.options.highlightClass);
-    if (!this.SelectIdxStart || !this.SelectIdxEnd) return false;
-    var r1=Math.max(Math.min(this.SelectIdxEnd.row,this.SelectIdxStart.row), this.buffer.windowStart);
-    var r2=Math.min(Math.max(this.SelectIdxEnd.row,this.SelectIdxStart.row), this.buffer.windowEnd-1);
-    if (r1 > r2) return false;
-    var c1=Math.min(this.SelectIdxEnd.column,this.SelectIdxStart.column);
-    var c2=Math.max(this.SelectIdxEnd.column,this.SelectIdxStart.column);
-    var curIdx=this.datasetIndex(cell);
-    return (r1<=curIdx.row && curIdx.row<=r2 && c1<=curIdx.column && curIdx.column<=c2);
-  },
-
-  highlightCell: function(cell) {
-    Rico.addClass(cell,this.options.highlightClass);
-  },
-
-  unhighlightCell: function(cell) {
-    if (cell==null) return;
-    Rico.removeClass(cell,this.options.highlightClass);
-  },
-
-  selectRow: function(r) {
-    for (var c=0; c<this.columns.length; c++)
-      this.highlightCell(this.columns[c].cell(r));
-  },
-
-  unselectRow: function(r) {
-    for (var c=0; c<this.columns.length; c++)
-      this.unhighlightCell(this.columns[c].cell(r));
-  },
-
-  rowMouseOver: function(e) {
-    if (!this.highlightEnabled) return;
-    var cell=Rico.eventElement(e);
-    cell=Rico.getParentByTagName(cell,'div','ricoLG_cell');
-    if (!cell) return;
-    var newIdx=this.winCellIndex(cell);
-    if ((this.options.highlightSection & (newIdx.tabIdx+1))==0) return;
-    this.highlight(newIdx);
-  },
-
-  highlight: function(newIdx) {
-    if (this.options.highlightMethod!='outline') this.cursorSetClass(newIdx);
-    if (this.options.highlightMethod!='class') this.cursorOutline(newIdx);
-    this.highlightIdx=newIdx;
-  },
-
-  cursorSetClass: function(newIdx) {
-    switch (this.options.highlightElem) {
-      case 'menuCell':
-      case 'cursorCell':
-        if (this.highlightIdx) this.unhighlightCell(this.highlightIdx.cell);
-        this.highlightCell(newIdx.cell);
-        break;
-      case 'menuRow':
-      case 'cursorRow':
-        if (this.highlightIdx) this.unselectRow(this.highlightIdx.row);
-        var s1=this.options.highlightSection & 1;
-        var s2=this.options.highlightSection & 2;
-        var c0=s1 ? 0 : this.options.frozenColumns;
-        var c1=s2 ? this.columns.length : this.options.frozenColumns;
-        for (var c=c0; c<c1; c++)
-          this.highlightCell(this.columns[c].cell(newIdx.row));
-        break;
-      default: return;
-    }
-  },
-
-  cursorOutline: function(newIdx) {
-    var div;
-    switch (this.options.highlightElem) {
-      case 'menuCell':
-      case 'cursorCell':
-        div=this.highlightDiv[newIdx.tabIdx];
-        div.style.left=(this.columns[newIdx.column].dataCell.offsetLeft-1)+'px';
-        div.style.width=this.columns[newIdx.column].colWidth;
-        this.highlightDiv[1-newIdx.tabIdx].style.display='none';
-        break;
-      case 'menuRow':
-      case 'cursorRow':
-        div=this.highlightDiv[0];
-        var s1=this.options.highlightSection & 1;
-        var s2=this.options.highlightSection & 2;
-        div.style.left=s1 ? '0px' : this.frozenTabs.style.width;
-        div.style.width=((s1 ? this.frozenTabs.offsetWidth : 0) + (s2 ? this.innerDiv.offsetWidth : 0) - 4)+'px';
-        break;
-      default: return;
-    }
-    div.style.top=(this.hdrHt+newIdx.row*this.rowHeight-1)+'px';
-    div.style.height=(this.rowHeight-1)+'px';
-    div.style.display='';
-  },
-
-  unhighlight: function() {
-    switch (this.options.highlightElem) {
-      case 'menuCell':
-        //this.highlightIdx=this.menuIdx;
-        /*jsl:fallthru*/
-      case 'cursorCell':
-        if (this.highlightIdx) this.unhighlightCell(this.highlightIdx.cell);
-        if (!this.highlightDiv) return;
-        for (var i=0; i<2; i++)
-          this.highlightDiv[i].style.display='none';
-        break;
-      case 'menuRow':
-        //this.highlightIdx=this.menuIdx;
-        /*jsl:fallthru*/
-      case 'cursorRow':
-        if (this.highlightIdx) this.unselectRow(this.highlightIdx.row);
-        if (this.highlightDiv) this.highlightDiv[0].style.display='none';
-        break;
-    }
-  },
-
-  resetContents: function() {
-    Rico.log("resetContents");
-    this.ClearSelection();
-    this.buffer.clear();
-    this.clearRows();
-    this.clearBookmark();
-  },
-
-  setImages: function() {
-    for (var n=0; n<this.columns.length; n++)
-      this.columns[n].setImage();
-  },
-
-/**
- * @return column index, or -1 if there are no sorted columns
- */
-  findSortedColumn: function() {
-    for (var n=0; n<this.columns.length; n++) {
-      if (this.columns[n].isSorted()) return n;
-    }
-    return -1;
-  },
-
-  findColumnName: function(name) {
-    for (var n=0; n<this.columns.length; n++) {
-      if (this.columns[n].fieldName == name) return n;
-    }
-    return -1;
-  },
-  
-/**
- * Searches options.columnSpecs colAttr for matching colValue
- * @return array of matching column indexes
- */
-  findColumnsBySpec: function(colAttr, colValue) {
-    var result=[]
-    for (var n=0; n<this.options.columnSpecs.length; n++) {
-      if (this.options.columnSpecs[n][colAttr] == colValue) result.push(n);
-    }
-    return result;
-  },
-
-/**
- * Set initial sort
- */
-  setSortUI: function( columnNameOrNum, sortDirection ) {
-    Rico.log("setSortUI: "+columnNameOrNum+' '+sortDirection);
-    var colnum=this.findSortedColumn();
-    if (colnum >= 0) {
-      sortDirection=this.columns[colnum].getSortDirection();
-    } else {
-      if (typeof sortDirection!='string') {
-        sortDirection=Rico.ColumnConst.SORT_ASC;
-      } else {
-        sortDirection=sortDirection.toUpperCase();
-        if (sortDirection != Rico.ColumnConst.SORT_DESC) sortDirection=Rico.ColumnConst.SORT_ASC;
-      }
-      switch (typeof columnNameOrNum) {
-        case 'string':
-          colnum=this.findColumnName(columnNameOrNum);
-          break;
-        case 'number':
-          colnum=columnNameOrNum;
-          break;
-      }
-    }
-    if (typeof(colnum)!='number' || colnum < 0) return;
-    this.clearSort();
-    this.columns[colnum].setSorted(sortDirection);
-    this.buffer.sortBuffer(colnum);
-  },
-
-/**
- * clear sort flag on all columns
- */
-  clearSort: function() {
-    for (var x=0;x<this.columns.length;x++)
-      this.columns[x].setUnsorted();
-  },
-
-/**
- * clear filters on all columns
- */
-  clearFilters: function() {
-    for (var x=0;x<this.columns.length;x++) {
-      this.columns[x].setUnfiltered(true);
-    }
-    if (this.options.filterHandler) {
-      this.options.filterHandler();
-    }
-  },
-
-/**
- * returns number of columns with a user filter set
- */
-  filterCount: function() {
-    for (var x=0,cnt=0;x<this.columns.length;x++) {
-      if (this.columns[x].isFiltered()) cnt++;
-    }
-    return cnt;
-  },
-
-  sortHandler: function() {
-    this.cancelMenu();
-    this.ClearSelection();
-    this.setImages();
-    var n=this.findSortedColumn();
-    if (n < 0) return;
-    Rico.log("sortHandler: sorting column "+n);
-    this.buffer.sortBuffer(n);
-    this.clearRows();
-    this.scrollDiv.scrollTop = 0;
-    this.buffer.fetch(0);
-  },
-
-  filterHandler: function() {
-    Rico.log("filterHandler");
-    this.cancelMenu();
-    if (this.buffer.processingRequest) {
-      this.queueFilter=true;
-      return;
-    }
-    this.unplugScroll();
-    this.ClearSelection();
-    this.setImages();
-    this.clearBookmark();
-    this.clearRows();
-    this.buffer.fetch(-1);
-    Rico.runLater(10,this,'pluginScroll'); // resetting ht div can cause a scroll event, triggering an extra fetch
-  },
-
-  clearBookmark: function() {
-    if (this.bookmark) this.bookmark.innerHTML="&nbsp;";
-  },
-
-  bookmarkHandler: function(firstrow,lastrow) {
-    var newhtml;
-    if (isNaN(firstrow) || !this.bookmark) return;
-    var totrows=this.buffer.totalRows;
-    if (totrows < lastrow) lastrow=totrows;
-    if (totrows<=0) {
-      newhtml = Rico.getPhraseById('bookmarkNoMatch');
-    } else if (lastrow<0) {
-      newhtml = Rico.getPhraseById('bookmarkNoRec');
-    } else if (this.buffer.foundRowCount) {
-      newhtml = Rico.getPhraseById('bookmarkExact',firstrow,lastrow,totrows);
-    } else {
-      newhtml = Rico.getPhraseById('bookmarkAbout',firstrow,lastrow,totrows);
-    }
-    this.bookmark.innerHTML = newhtml;
-  },
-
-  clearRows: function() {
-    if (this.isBlank==true) return;
-    for (var c=0; c < this.columns.length; c++)
-      this.columns[c].clearColumn();
-    this.isBlank = true;
-  },
-
-  refreshContents: function(startPos) {
-    Rico.log("refreshContents1 "+this.tableId+": startPos="+startPos+" lastRow="+this.lastRowPos+" PartBlank="+this.isPartialBlank+" pageSize="+this.pageSize);
-    this.hideMsg();
-    this.cancelMenu();
-    this.unhighlight(); // in case highlighting was manually invoked
-    if (this.queueFilter) {
-      Rico.log("refreshContents: cancelling refresh because filter has changed");
-      this.queueFilter=false;
-      this.filterHandler();
-      return;
-    }
-    this.highlightEnabled=this.options.highlightSection!='none';
-    var viewPrecedesBuffer = this.buffer.startPos > startPos;
-    var contentStartPos = viewPrecedesBuffer ? this.buffer.startPos: startPos;
-    this.contentStartPos = contentStartPos+1;
-    var contentEndPos = Math.min(this.buffer.startPos + this.buffer.size, startPos + this.pageSize);
-    this.buffer.setWindow(contentStartPos, contentEndPos);
-    Rico.log('refreshContents2 '+this.tableId+': cStartPos='+contentStartPos+' cEndPos='+contentEndPos+' vPrecedesBuf='+viewPrecedesBuffer+' b.startPos='+this.buffer.startPos);
-    if (startPos == this.lastRowPos && !this.isPartialBlank && !this.isBlank) return;
-    this.isBlank = false;
-    var onRefreshComplete = this.options.onRefreshComplete;
-
-    if ((startPos + this.pageSize < this.buffer.startPos) ||
-        (this.buffer.startPos + this.buffer.size < startPos) ||
-        (this.buffer.size == 0)) {
-      this.clearRows();
-      if (onRefreshComplete) onRefreshComplete(this.contentStartPos,contentEndPos);  // update bookmark
-      return;
-    }
-
-    Rico.log('refreshContents: contentStartPos='+contentStartPos+' contentEndPos='+contentEndPos+' viewPrecedesBuffer='+viewPrecedesBuffer);
-    var rowSize = contentEndPos - contentStartPos;
-    var blankSize = this.pageSize - rowSize;
-    var blankOffset = viewPrecedesBuffer ? 0: rowSize;
-    var contentOffset = viewPrecedesBuffer ? blankSize: 0;
-
-    for (var r=0; r < rowSize; r++) { //initialize what we have
-      for (var c=0; c < this.columns.length; c++)
-        this.columns[c].displayValue(r + contentOffset);
-    }
-    for (var i=0; i < blankSize; i++)     // blank out the rest
-      this.blankRow(i + blankOffset);
-    if (this.options.highlightElem=='selection') this.ShowSelection();
-    this.isPartialBlank = blankSize > 0;
-    this.lastRowPos = startPos;
-    Rico.log("refreshContents complete, startPos="+startPos);
-    if (onRefreshComplete) onRefreshComplete(this.contentStartPos,contentEndPos);  // update bookmark
-  },
-
-  scrollToRow: function(rowOffset) {
-     var p=this.rowToPixel(rowOffset);
-     Rico.log("scrollToRow, rowOffset="+rowOffset+" pixel="+p);
-     this.scrollDiv.scrollTop = p; // this causes a scroll event
-     if ( this.options.onscroll )
-        this.options.onscroll( this, rowOffset );
-  },
-
-  scrollUp: function() {
-     this.moveRelative(-1);
-  },
-
-  scrollDown: function() {
-     this.moveRelative(1);
-  },
-
-  pageUp: function() {
-     this.moveRelative(-this.pageSize);
-  },
-
-  pageDown: function() {
-     this.moveRelative(this.pageSize);
-  },
-
-  adjustRow: function(rowOffset) {
-     var notdisp=this.topOfLastPage();
-     if (notdisp == 0 || !rowOffset) return 0;
-     return Math.min(notdisp,rowOffset);
-  },
-
-  rowToPixel: function(rowOffset) {
-     return this.adjustRow(rowOffset) * this.rowHeight;
-  },
-
-/**
- * @returns row to display at top of scroll div
- */
-  pixeltorow: function(p) {
-     var notdisp=this.topOfLastPage();
-     if (notdisp == 0) return 0;
-     var prow=parseInt(p/this.rowHeight,10);
-     return Math.min(notdisp,prow);
-  },
-
-  moveRelative: function(relOffset) {
-     var newoffset=Math.max(this.scrollDiv.scrollTop+relOffset*this.rowHeight,0);
-     newoffset=Math.min(newoffset,this.scrollDiv.scrollHeight);
-     //Rico.log("moveRelative, newoffset="+newoffset);
-     this.scrollDiv.scrollTop=newoffset;
-  },
-
-  pluginScroll: function() {
-     if (this.scrollPluggedIn) return;
-     Rico.log("pluginScroll: wheelEvent="+this.wheelEvent);
-     Rico.eventBind(this.scrollDiv,"scroll",this.scrollEventFunc, false);
-     for (var t=0; t<2; t++)
-       Rico.eventBind(this.tabs[t],this.wheelEvent,this.wheelEventFunc, false);
-     this.scrollPluggedIn=true;
-  },
-
-  unplugScroll: function() {
-     if (!this.scrollPluggedIn) return;
-     Rico.log("unplugScroll");
-     Rico.eventUnbind(this.scrollDiv,"scroll", this.scrollEventFunc , false);
-     for (var t=0; t<2; t++)
-       Rico.eventUnbind(this.tabs[t],this.wheelEvent,this.wheelEventFunc, false);
-     this.scrollPluggedIn=false;
-  },
-
-  handleWheel: function(e) {
-    var delta = 0;
-    if (e.wheelDelta) {
-      if (Rico.isOpera)
-        delta = e.wheelDelta/120;
-      else if (Rico.isWebKit)
-        delta = -e.wheelDelta/12;
-      else
-        delta = -e.wheelDelta/120;
-    } else if (e.detail) {
-      delta = e.detail/3; /* Mozilla/Gecko */
-    }
-    if (delta) this.moveRelative(delta);
-    Rico.eventStop(e);
-    return false;
-  },
-
-  handleScroll: function(e) {
-     if ( this.scrollTimeout )
-       clearTimeout( this.scrollTimeout );
-     this.setHorizontalScroll();
-     var scrtop=this.scrollDiv.scrollTop;
-     var vscrollDiff = this.lastScrollPos-scrtop;
-     if (vscrollDiff == 0.00) return;
-     var newrow=this.pixeltorow(scrtop);
-     if (newrow == this.lastRowPos && !this.isPartialBlank && !this.isBlank) return;
-     var stamp1 = new Date();
-     Rico.log("handleScroll, newrow="+newrow+" scrtop="+scrtop);
-     if (this.options.highlightElem=='selection') this.HideSelection();
-     this.buffer.fetch(newrow);
-     if (this.options.onscroll) this.options.onscroll(this, newrow);
-     this.scrollTimeout = Rico.runLater(1200,this,'scrollIdle');
-     this.lastScrollPos = this.scrollDiv.scrollTop;
-     var stamp2 = new Date();
-     //Rico.log("handleScroll, time="+(stamp2.getTime()-stamp1.getTime()));
-  },
-
-  scrollIdle: function() {
-     if ( this.options.onscrollidle )
-        this.options.onscrollidle();
-  }
-
-};
-
-
-Rico.LiveGridColumn = function(grid,colIdx,hdrInfo,tabIdx) {
-  this.initialize(grid,colIdx,hdrInfo,tabIdx);
-};
-
-Rico.LiveGridColumn.prototype = 
-/** @lends Rico.LiveGridColumn# */
-{
-/**
- * Implements a LiveGrid column. Also contains static properties used by SimpleGrid columns.
- * @extends Rico.TableColumnBase
- * @constructs
- */
-initialize: function(liveGrid,colIdx,hdrInfo,tabIdx) {
-  Rico.extend(this, new Rico.TableColumnBase());
-  this.baseInit(liveGrid,colIdx,hdrInfo,tabIdx);
-  this.buffer=liveGrid.buffer;
-  if (typeof(this.format.type)!='string' || this.format.EntryType=='tinyMCE') this.format.type='raw';
-  if (typeof this.isNullable!='boolean') this.isNullable = /number|date/.test(this.format.type);
-  this.isText = /raw|text|showTags/.test(this.format.type);
-  Rico.log(" sortable="+this.sortable+" filterable="+this.filterable+" hideable="+this.hideable+" isNullable="+this.isNullable+' isText='+this.isText);
-  this.fixHeaders(this.liveGrid.tableId, this.options.hdrIconsFirst);
-  if (this['format_'+this.format.type]) {
-    this._format=this['format_'+this.format.type];
-  }
-  if (this.format.control) {
-    // copy all properties/methods that start with '_'
-    if (typeof this.format.control=='string') {
-      this.format.control=eval(this.format.control);
-    }
-    for (var property in this.format.control) {
-      if (property.charAt(0)=='_') {
-        Rico.log("Copying control property "+property+ ' to ' + this);
-        this[property] = this.format.control[property];
-      }
-    }
-  }
-},
-
-/**
- * Sorts the column in ascending order
- */
-sortAsc: function() {
-  this.setColumnSort(Rico.ColumnConst.SORT_ASC);
-},
-
-/**
- * Sorts the column in descending order
- */
-sortDesc: function() {
-  this.setColumnSort(Rico.ColumnConst.SORT_DESC);
-},
-
-/**
- * Sorts the column in the specified direction
- * @param direction must be one of Rico.ColumnConst.UNSORTED, .SORT_ASC, or .SORT_DESC
- */
-setColumnSort: function(direction) {
-  this.liveGrid.clearSort();
-  this.setSorted(direction);
-  if (this.liveGrid.options.saveColumnInfo.sort)
-    this.liveGrid.setCookie();
-  if (this.options.sortHandler)
-    this.options.sortHandler();
-},
-
-/**
- * @returns true if this column is allowed to be sorted
- */
-isSortable: function() {
-  return this.sortable;
-},
-
-/**
- * @returns true if this column is currently sorted
- */
-isSorted: function() {
-  return this.currentSort != Rico.ColumnConst.UNSORTED;
-},
-
-/**
- * @returns Rico.ColumnConst.UNSORTED, .SORT_ASC, or .SORT_DESC
- */
-getSortDirection: function() {
-  return this.currentSort;
-},
-
-/**
- * toggle the sort sequence for this column
- */
-toggleSort: function() {
-  if (this.buffer && this.buffer.totalRows==0) return;
-  if (this.currentSort == Rico.ColumnConst.SORT_ASC)
-    this.sortDesc();
-  else
-    this.sortAsc();
-},
-
-/**
- * Flags that this column is not sorted
- */
-setUnsorted: function() {
-  this.setSorted(Rico.ColumnConst.UNSORTED);
-},
-
-/**
- * Flags that this column is sorted, but doesn't actually carry out the sort
- * @param direction must be one of Rico.ColumnConst.UNSORTED, .SORT_ASC, or .SORT_DESC
- */
-setSorted: function(direction) {
-  this.currentSort = direction;
-},
-
-/**
- * @returns true if this column is allowed to be filtered
- */
-canFilter: function() {
-  return this.filterable;
-},
-
-/**
- * @returns a textual representation of how this column is filtered
- */
-getFilterText: function() {
-  var vals=[];
-  for (var i=0; i<this.filterValues.length; i++) {
-    var v=this.filterValues[i];
-    vals.push(v=='' ? Rico.getPhraseById('filterBlank') : v);
-  }
-  switch (this.filterOp) {
-    case 'EQ':   return '= '+vals.join(', ');
-    case 'NE':   return Rico.getPhraseById('filterNot',vals.join(', '));
-    case 'LE':   return '<= '+vals[0];
-    case 'GE':   return '>= '+vals[0];
-    case 'LIKE': return Rico.getPhraseById('filterLike',vals[0]);
-    case 'NULL': return Rico.getPhraseById('filterEmpty');
-    case 'NOTNULL': return Rico.getPhraseById('filterNotEmpty');
-  }
-  return '?';
-},
-
-/**
- * @returns returns the query string representation of the filter
- */
-getFilterQueryParm: function() {
-  if (this.filterType == Rico.ColumnConst.UNFILTERED) return '';
-  var retval='&f['+this.index+'][op]='+this.filterOp;
-  retval+='&f['+this.index+'][len]='+this.filterValues.length;
-  for (var i=0; i<this.filterValues.length; i++) {
-    retval+='&f['+this.index+']['+i+']='+escape(this.filterValues[i]);
-  }
-  return retval;
-},
-
-/**
- * removes the filter from this column
- */
-setUnfiltered: function(skipHandler) {
-  this.filterType = Rico.ColumnConst.UNFILTERED;
-  if (this.liveGrid.options.saveColumnInfo.filter)
-    this.liveGrid.setCookie();
-  if (this.removeFilterFunc)
-    this.removeFilterFunc();
-  if (this.options.filterHandler && !skipHandler)
-    this.options.filterHandler();
-},
-
-setFilterEQ: function() {
-  this.setUserFilter('EQ');
-},
-setFilterNE: function() {
-  this.setUserFilter('NE');
-},
-addFilterNE: function() {
-  this.filterValues.push(this.userFilter);
-  if (this.liveGrid.options.saveColumnInfo.filter)
-    this.liveGrid.setCookie();
-  if (this.options.filterHandler)
-    this.options.filterHandler();
-},
-setFilterGE: function() { this.setUserFilter('GE'); },
-setFilterLE: function() { this.setUserFilter('LE'); },
-setFilterKW: function(keyword) {
-  if (keyword!='' && keyword!=null) {
-    this.setFilter('LIKE',keyword,Rico.ColumnConst.USERFILTER);
-  } else {
-    this.setUnfiltered(false);
-  }
-},
-
-setUserFilter: function(relop) {
-  this.setFilter(relop,this.userFilter,Rico.ColumnConst.USERFILTER);
-},
-
-setSystemFilter: function(relop,filter) {
-  this.setFilter(relop,filter,Rico.ColumnConst.SYSTEMFILTER);
-},
-
-setFilter: function(relop,filter,type,removeFilterFunc) {
-  this.filterValues = typeof(filter)=='object' ? filter : [filter];
-  this.filterType = type;
-  this.filterOp = relop;
-  if (type == Rico.ColumnConst.USERFILTER && this.liveGrid.options.saveColumnInfo.filter)
-    this.liveGrid.setCookie();
-  this.removeFilterFunc=removeFilterFunc;
-  if (this.options.filterHandler)
-    this.options.filterHandler();
-},
-
-isFiltered: function() {
-  return this.filterType == Rico.ColumnConst.USERFILTER;
-},
-
-filterChange: function(e) {\r
-  var selbox=Rico.eventElement(e);
-  if (selbox.value==this.liveGrid.options.FilterAllToken)\r
-    this.setUnfiltered();\r
-  else
-    this.setFilter('EQ',selbox.value,Rico.ColumnConst.USERFILTER,function() {selbox.selectedIndex=0;});\r
-},
-
-filterClear: function(e) {\r
-  this.filterField.value='';
-  this.setUnfiltered();\r
-},
-
-filterKeypress: function(e) {\r
-  var txtbox=Rico.eventElement(e);
-  if (typeof this.lastKeyFilter != 'string') this.lastKeyFilter='';\r
-  if (this.lastKeyFilter==txtbox.value) return;\r
-  var v=txtbox.value;\r
-  Rico.log("filterKeypress: "+this.index+' '+v);\r
-  this.lastKeyFilter=v;
-  if (v=='' || v=='*')\r
-    this.setUnfiltered();\r
-  else {
-    this.setFilter('LIKE', v, Rico.ColumnConst.USERFILTER, function() {txtbox.value='';});
-  }\r
-},\r
-
-mFilterSelectClick: function(e) {
-  Rico.eventStop(e);
-  if (this.mFilter.style.display!='none') {
-    this.mFilterFinish(e);
-    if (Rico.isIE && Rico.ieVersion <= 6) {
-      this.filterField.focus();
-    } else {
-      this.filterField.blur();
-    }
-  } else {
-    var offset=Rico.cumulativeOffset(this.filterField);
-    this.mFilter.style.top=(offset.top+this.filterField.offsetHeight)+'px';
-    this.mFilter.style.left=offset.left+'px';
-    this.mFilter.style.width=Math.min(this.filterField.offsetWidth,parseInt(this.colWidth,10))+'px';
-    Rico.show(this.mFilter);
-    this.mFilterFocus.focus();
-  }
-},
-
-mFilterFinish: function(e) {
-  if (!this.mFilterChange) {
-    Rico.hide(this.mFilter);
-    return;
-  }
-  if (this.mFilterInputs[0].checked) {
-    this.mFilterReset();
-    Rico.hide(this.mFilter);
-    this.setUnfiltered();
-    return;
-  }
-  var newValues=[];
-  var newLabels=[];
-  for (var i=1; i<this.mFilterInputs.length; i++) {
-    if (this.mFilterInputs[i].checked) {
-      newValues.push(this.mFilterInputs[i].value)
-      newLabels.push(this.mFilterLabels[i].innerHTML)
-    }
-  }
-  if (newValues.length > 0) {
-    var newText=newLabels.join(', ');
-    this.filterField.options[0].text=newText;
-    this.filterField.title=newText;
-    Rico.hide(this.mFilter);
-    this.mFilterChange=false;
-    var self=this;
-    this.setFilter('EQ',newValues,Rico.ColumnConst.USERFILTER,function() { self.mFilterReset(); });
-  } else {
-    alert('Please select at least one value');
-  }
-},
-
-mFilterReset: function() {
-  var newText=this.mFilterLabels[0].innerHTML;  // all
-  this.filterField.options[0].text=newText;
-  this.filterField.title=newText;
-},
-
-mFilterAllClick: function(e) {
-  var allChecked=this.mFilterInputs[0].checked;
-  for (var i=1; i<this.mFilterInputs.length; i++) {
-    this.mFilterInputs[i].checked=allChecked;
-  }
-  this.mFilterChange=true;
-},
-
-mFilterOtherClick: function(e) {
-  this.mFilterInputs[0].checked=false;
-  this.mFilterChange=true;
-},
-
-format_text: function(v) {
-  if (typeof v!='string')
-    return '&nbsp;';
-  else
-    return Rico.stripTags(v);
-},
-
-format_showTags: function(v) {
-  if (typeof v!='string')
-    return '&nbsp;';
-  else
-    return v.replace(/&/g, '&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
-},
-
-format_number: function(v) {
-  if (typeof v=='undefined' || v=='' || v==null)
-    return '&nbsp;';
-  else
-    return Rico.formatNumber(v,this.format);
-},
-
-format_datetime: function(v) {
-  if (typeof v=='undefined' || v=='' || v==null)
-    return '&nbsp;';
-  else {
-    var d=Rico.setISO8601(v);
-    if (!d) return v;
-    return (this.format.prefix || '')+Rico.formatDate(d,this.format.dateFmt || 'translateDateTime')+(this.format.suffix || '');
-  }
-},
-
-// converts GMT/UTC to local time
-format_UTCasLocalTime: function(v) {
-  if (typeof v=='undefined' || v=='' || v==null)
-    return '&nbsp;';
-  else {
-    var tz=new Date();
-    var d=Rico.setISO8601(v,-tz.getTimezoneOffset());
-    if (!d) return v;
-    return (this.format.prefix || '')+Rico.formatDate(d,this.format.dateFmt || 'translateDateTime')+(this.format.suffix || '');
-  }
-},
-
-format_date: function(v) {
-  if (typeof v=='undefined' || v==null || v=='')
-    return '&nbsp;';
-  else {
-    var d=Rico.setISO8601(v);
-    if (!d) return v;
-    return (this.format.prefix || '')+Rico.formatDate(d,this.format.dateFmt || 'translateDate')+(this.format.suffix || '');
-  }
-},
-
-fixHeaders: function(prefix, iconsfirst) {
-  if (this.sortable) {
-    var handler=Rico.eventHandle(this,'toggleSort');
-    switch (this.options.headingSort) {
-      case 'link':
-        var a=Rico.wrapChildren(this.hdrCellDiv,'ricoSort',undefined,'a');
-        a.href = "javascript:void(0)";
-        Rico.eventBind(a,"click", handler);
-        break;
-      case 'hover':
-        Rico.eventBind(this.hdrCellDiv,"click", handler);
-        break;
-    }
-  }
-  this.imgFilter = document.createElement('img');
-  this.imgFilter.style.display='none';
-  this.imgFilter.src=Rico.imgDir+this.options.filterImg;
-  this.imgFilter.className='ricoLG_HdrIcon';
-  this.imgSort = document.createElement('span');
-  this.imgSort.style.display='none';
-  if (iconsfirst) {
-    this.hdrCellDiv.insertBefore(this.imgSort,this.hdrCellDiv.firstChild);
-    this.hdrCellDiv.insertBefore(this.imgFilter,this.hdrCellDiv.firstChild);
-  } else {
-    this.hdrCellDiv.appendChild(this.imgFilter);
-    this.hdrCellDiv.appendChild(this.imgSort);
-  }
-  if (!this.format.filterUI) {
-    Rico.eventBind(this.imgFilter, 'click', Rico.eventHandle(this,'filterClick'), false);
-  }
-},
-
-filterClick: function(e) {
-  if (this.filterType==Rico.ColumnConst.USERFILTER && this.filterOp=='LIKE') {
-    this.liveGrid.openKeyword(this.index);
-  }
-},
-
-getValue: function(windowRow) {
-  return this.buffer.getWindowCell(windowRow,this.index);
-},
-
-getBufferAttr: function(windowRow) {
-  return this.buffer.getWindowAttr(windowRow,this.index);
-},
-
-setValue: function(windowRow,newval) {
-  this.buffer.setWindowValue(windowRow,this.index,newval);
-},
-
-_format: function(v) {
-  return v;
-},
-
-_display: function(v,gridCell) {
-  gridCell.innerHTML=this._format(v);
-},
-
-_export: function(v) {
-  return this._format(v);
-},
-
-exportBuffer: function(bufRow) {
-  return this._export(this.buffer.getValue(bufRow,this.index));
-},
-
-displayValue: function(windowRow) {
-  var bufval=this.getValue(windowRow);
-  if (bufval==null) {
-    this.clearCell(windowRow);
-    return;
-  }
-  var gridCell=this.cell(windowRow);
-  this._display(bufval,gridCell,windowRow);
-  var acceptAttr=this.buffer.options.acceptAttr;
-  if (acceptAttr.length==0) return;
-  var bufAttr=this.getBufferAttr(windowRow);
-  if (bufAttr==null) return;
-  for (var k=0; k<acceptAttr.length; k++) {
-    bufAttr=bufAttr['_'+acceptAttr[k]] || '';
-    switch (acceptAttr[k]) {
-      case 'style': gridCell.style.cssText=bufAttr; break;
-      case 'class': gridCell.className=bufAttr; break;
-      default:      gridCell['_'+acceptAttr[k]]=bufAttr; break;
-    }
-  }
-}
-
-};
diff --git a/ricoClient/js/minsrc/ricoLiveGridAjax.js b/ricoClient/js/minsrc/ricoLiveGridAjax.js
deleted file mode 100644 (file)
index cab820f..0000000
+++ /dev/null
@@ -1,577 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-if(typeof Rico=='undefined') throw("LiveGridAjax requires the Rico JavaScript framework");
-
-if (!Rico.Buffer) Rico.Buffer = {};
-
-Rico.Buffer.AjaxXML = function(url,options,ajaxOptions) {
-  this.initialize(url,options,ajaxOptions);
-}
-
-Rico.Buffer.AjaxXML.prototype = {
-/**
- * @class Implements buffer for LiveGrid. Loads data from server via a single AJAX call.
- * @extends Rico.Buffer.Base
- * @constructs
- */
-  initialize: function(url,options,ajaxOptions) {
-    Rico.extend(this, new Rico.Buffer.Base());
-    Rico.extend(this, Rico.Buffer.AjaxXMLMethods);
-    this.dataSource=url;
-    this.options.bufferTimeout=20000;            // time to wait for ajax response (milliseconds)
-    this.options.requestParameters=[];
-    this.options.waitMsg=Rico.getPhraseById("waitForData");  // replace this with an image tag if you prefer
-    this.options.canFilter=true;
-    this.options.fmt='xml';
-    Rico.extend(this.options, options || {});
-    this.ajaxOptions = { parameters: null, method : 'get' };
-    Rico.extend(this.ajaxOptions, ajaxOptions || {});
-    this.requestCount=0;
-    this.processingRequest=false;
-    this.pendingRequest=-2;
-    this.fetchData=true;
-    this.sortParm={};
-  }
-}
-
-Rico.Buffer.AjaxXMLMethods = {
-
-/** @lends Rico.Buffer.AjaxXML# */
-  fetch: function(offset) {
-    if (this.fetchData) {
-      this.foundRowCount=true;
-      this.fetchData=false;
-      this.processingRequest=true;
-      this.liveGrid.showMsg(this.options.waitMsg);
-      this.timeoutHandler = Rico.runLater(this.options.bufferTimeout,this,'handleTimedOut');
-      this.ajaxOptions.parameters = this.formQueryHashXML(0,-1);
-      Rico.log('sending request');
-      var self=this;
-      if (typeof this.dataSource=='string') {
-        this.ajaxOptions.onComplete = function(xhr) { self.ajaxUpdate(offset,xhr); };
-        new Rico.ajaxRequest(this.dataSource, this.ajaxOptions);
-      } else {
-        this.ajaxOptions.onComplete = function(newRows, newAttr, totalRows, errMsg) { self.jsUpdate(offset, newRows, newAttr, totalRows, errMsg); };
-        this.dataSource(this.ajaxOptions);
-      }
-    } else {
-      if (offset < 0) {
-        this.applyFilters();
-        this.setTotalRows(this.size);
-        offset=0;
-      }
-      this.liveGrid.refreshContents(offset);
-    }
-  },
-
-/**
- * Server did not respond in time... assume that there could have been
- * an error, and allow requests to be processed again.
- */
-  handleTimedOut: function() {
-    Rico.log("Request Timed Out");
-    this.liveGrid.showMsg(Rico.getPhraseById("requestTimedOut"));
-  },
-
-  formQueryHashXML: function(startPos,fetchSize) {
-    var queryHash= {
-      id: this.liveGrid.tableId,
-      page_size: (typeof fetchSize=='number') ? fetchSize : this.totalRows,
-      offset: startPos.toString()
-    };
-    if (!this.foundRowCount) queryHash['get_total']='true';
-    if (this.options.requestParameters) {
-      for ( var i=0; i < this.options.requestParameters.length; i++ ) {
-        var anArg = this.options.requestParameters[i];
-        if ( anArg.name != undefined && anArg.value != undefined ) {
-          queryHash[anArg.name]=anArg.value;
-        } else {
-          var ePos  = anArg.indexOf('=');
-          var argName  = anArg.substring( 0, ePos );
-          var argValue = anArg.substring( ePos + 1 );
-          queryHash[argName]=argValue;
-        }
-      }
-    }
-    return queryHash;
-  },
-
-  clearTimer: function() {
-    if(typeof this.timeoutHandler != "number") return;
-    window.clearTimeout(this.timeoutHandler);
-    delete this.timeoutHandler;
-  },
-
-  // used by both XML and SQL buffers
-  jsUpdate: function(startPos, newRows, newAttr, totalRows, errMsg) {
-    this.clearTimer();
-    this.processingRequest=false;
-    Rico.log("jsUpdate: "+arguments.length);
-    if (errMsg) {
-      Rico.log("jsUpdate: received error="+errMsg);
-      this.liveGrid.showMsg(Rico.getPhraseById("requestError",errMsg));
-      return;
-    }
-    this.rcvdRows = newRows.length;
-    if (typeof totalRows=='number') {
-      this.rowcntContent = totalRows.toString();
-      this.rcvdRowCount = true;
-      this.foundRowCount = true;
-      Rico.log("jsUpdate: found RowCount="+this.rowcntContent);
-    }
-    this.updateBuffer(startPos, newRows, newAttr);
-    if (this.options.onAjaxUpdate)
-      this.options.onAjaxUpdate();
-    this.updateGrid(startPos);
-    if (this.options.TimeOut && this.timerMsg)
-      this.restartSessionTimer();
-    if (this.pendingRequest>=-1) {
-      var offset=this.pendingRequest;
-      Rico.log("jsUpdate: found pending request for offset="+offset);
-      this.pendingRequest=-2;
-      this.fetch(offset);
-    }
-  },
-
-  // used by both XML and SQL buffers
-  ajaxUpdate: function(startPos,xhr) {
-    this.clearTimer();
-    this.processingRequest=false;
-    if (xhr.status != 200) {
-      Rico.log("ajaxUpdate: received http error="+xhr.status);
-      this.liveGrid.showMsg(Rico.getPhraseById("httpError",xhr.status));
-      return;
-    }
-    this._responseHandler=this['processResponse'+this.options.fmt.toUpperCase()];
-    if (!this._responseHandler(startPos,xhr)) return;
-    if (this.options.onAjaxUpdate)
-      this.options.onAjaxUpdate();
-    this.updateGrid(startPos);
-    if (this.options.TimeOut && this.timerMsg)
-      this.restartSessionTimer();
-    if (this.pendingRequest>=-1) {
-      var offset=this.pendingRequest;
-      Rico.log("ajaxUpdate: found pending request for offset="+offset);
-      this.pendingRequest=-2;
-      this.fetch(offset);
-    }
-  },
-  
-  // used by both XML and SQL buffers
-  processResponseXML: function(startPos,request) {
-    // The response text may contain META DATA for debugging if client side debugging is enabled in VS\r
-    var xmlDoc = request.responseXML;\r
-    if (request.responseText.substring(0, 4) == "<!--") {\r
-      var nEnd = request.responseText.indexOf("-->");\r
-      if (nEnd == -1) {\r
-        this.liveGrid.showMsg('Web server error - client side debugging may be enabled');\r
-        return false;\r
-      }\r
-      xmlDoc = Rico.createXmlDocument();\r
-      xmlDoc.loadXML(request.responseText.substring(nEnd+3));\r
-    }
-    
-    if (!xmlDoc) {
-      alert("Data provider returned an invalid XML response");
-      Rico.log("Data provider returned an invalid XML response");
-      return false;
-    }
-
-    // process children of <ajax-response>
-    var response = xmlDoc.getElementsByTagName("ajax-response");
-    if (response == null || response.length != 1) return false;
-    this.rcvdRows = 0;
-    this.rcvdRowCount = false;
-    var ajaxResponse=response[0];
-    var debugtags = ajaxResponse.getElementsByTagName('debug');
-    for (var i=0; i<debugtags.length; i++)
-      Rico.log("ajaxUpdate: debug msg "+i+": "+Rico.getContentAsString(debugtags[i],this.options.isEncoded));
-    var error = ajaxResponse.getElementsByTagName('error');
-    if (error.length > 0) {
-      var msg=Rico.getContentAsString(error[0],this.options.isEncoded);
-      alert("Data provider returned an error:\n"+msg);
-      Rico.log("Data provider returned an error:\n"+msg);
-      return false;
-    }
-    var rowsElement = ajaxResponse.getElementsByTagName('rows')[0];
-    if (!rowsElement) {
-      Rico.log("ajaxUpdate: invalid response");
-      this.liveGrid.showMsg(Rico.getPhraseById("invalidResponse"));
-      return false;
-    }
-    var rowcnttags = ajaxResponse.getElementsByTagName('rowcount');
-    if (rowcnttags && rowcnttags.length==1) {
-      this.rowcntContent = Rico.getContentAsString(rowcnttags[0],this.options.isEncoded);
-      this.rcvdRowCount = true;
-      this.foundRowCount = true;
-      Rico.log("ajaxUpdate: found RowCount="+this.rowcntContent);
-    }
-
-    // process <rows>
-    this.updateUI = rowsElement.getAttribute("update_ui") == "true";
-    this.rcvdOffset = rowsElement.getAttribute("offset");
-    Rico.log("ajaxUpdate: rcvdOffset="+this.rcvdOffset);
-    var newRows = this.dom2jstable(rowsElement);
-    var newAttr = (this.options.acceptAttr.length > 0) ? this.dom2jstableAttr(rowsElement) : false;
-    this.rcvdRows = newRows.length;
-    this.updateBuffer(startPos, newRows, newAttr);
-    return true;
-  },
-
-  processResponseJSON: function(startPos,request) {
-    var json = Rico.getJSON(request);
-    if (!json || json == null) {
-      alert("Data provider returned an invalid JSON response");
-      Rico.log("Data provider returned an invalid JSON response");
-      return false;
-    }
-
-    if (json.debug) {
-      for (var i=0; i<json.debug.length; i++)
-        Rico.writeDebugMsg("debug msg "+i+": "+json.debug[i]);
-    }
-    if (json.error) {
-      alert("Data provider returned an error:\n"+json.error);
-      Rico.writeDebugMsg("Data provider returned an error:\n"+json.error);
-      return false;
-    }
-
-    if (json.rowcount) {
-      this.rowcntContent = json.rowcount;
-      this.rcvdRowCount = true;
-      this.foundRowCount = true;
-      Rico.writeDebugMsg("loadRows, found RowCount="+json.rowcount);
-    }
-
-    this.rcvdRows = json.rows.length;
-    this.updateBuffer(startPos, json.rows);
-    return true;
-  },
-
-  // specific to XML buffer
-  updateBuffer: function(start, newRows, newAttr) {
-    this.baseRows = newRows;
-    this.attr = newAttr;
-    Rico.log("updateBuffer: # of rows="+this.rcvdRows);
-    this.rcvdRowCount=true;
-    this.rowcntContent=this.rcvdRows;
-    if (typeof this.delayedSortCol=='number')
-      this.sortBuffer(this.delayedSortCol);
-    this.applyFilters();
-    this.startPos = 0;
-  },
-
-  // used by both XML and SQL buffers
-  updateGrid: function(offset) {
-    Rico.log("updateGrid, size="+this.size+' rcv cnt type='+typeof(this.rowcntContent));
-    var newpos;
-    if (this.rcvdRowCount==true) {
-      Rico.log("found row cnt: "+this.rowcntContent);
-      var eofrow=parseInt(this.rowcntContent,10);
-      var lastTotalRows=this.totalRows;
-      if (!isNaN(eofrow) && eofrow!=lastTotalRows) {
-        this.setTotalRows(eofrow);
-        newpos=Math.min(this.liveGrid.topOfLastPage(),offset);
-        Rico.log("updateGrid: new rowcnt="+eofrow+" newpos="+newpos);
-        if (lastTotalRows==0 && this.liveGrid.sizeTo=='data')
-          Rico.runLater(100,this.liveGrid,'adjustPageSize');  // FF takes a long time to calc initial size
-        this.liveGrid.scrollToRow(newpos);
-        if ( this.isInRange(newpos) ) {
-          this.liveGrid.refreshContents(newpos);
-        } else {
-          this.fetch(newpos);
-        }
-        return;
-      }
-    } else {
-      var lastbufrow=offset+this.rcvdRows;
-      if (lastbufrow>this.totalRows) {
-        var newcnt=lastbufrow;
-        Rico.log("extending totrows to "+newcnt);
-        this.setTotalRows(newcnt);
-      }
-    }
-    newpos=this.liveGrid.pixeltorow(this.liveGrid.scrollDiv.scrollTop);
-    Rico.log("updateGrid: newpos="+newpos);
-    this.liveGrid.refreshContents(newpos);
-  }
-
-};
-
-
-
-Rico.Buffer.AjaxSQL = function(url,options,ajaxOptions) {
-  this.initialize(url,options,ajaxOptions);
-}
-
-Rico.Buffer.AjaxSQL.prototype = {
-/**
- * @class Implements buffer for LiveGrid. Loads data from server in chunks as user scrolls through the grid.
- * @extends Rico.Buffer.AjaxXML
- * @constructs
- */
-  initialize: function(url,options,ajaxOptions) {
-    Rico.extend(this, new Rico.Buffer.AjaxXML());
-    Rico.extend(this, Rico.Buffer.AjaxSQLMethods);
-    this.dataSource=url;
-    this.options.canFilter=true;
-    this.options.largeBufferSize  = 7.0;   // 7 pages
-    this.options.nearLimitFactor  = 1.0;   // 1 page
-    Rico.extend(this.options, options || {});
-    Rico.extend(this.ajaxOptions, ajaxOptions || {});
-  }
-}
-
-Rico.Buffer.AjaxSQLMethods = {
-/** @lends Rico.Buffer.AjaxSQL# */
-
-  registerGrid: function(liveGrid) {
-    this.liveGrid = liveGrid;
-    this.sessionExpired=false;
-    this.timerMsg=document.getElementById(liveGrid.tableId+'_timer');
-    if (this.options.TimeOut && this.timerMsg) {
-      if (!this.timerMsg.title) this.timerMsg.title=Rico.getPhraseById("sessionExpireMinutes");
-      this.restartSessionTimer();
-    }
-  },
-
-  setBufferSize: function(pageSize) {
-    this.maxFetchSize = Math.max(50,parseInt(this.options.largeBufferSize * pageSize,10));
-    this.nearLimit = parseInt(this.options.nearLimitFactor * pageSize,10);
-    this.maxBufferSize = this.maxFetchSize * 3;
-  },
-
-  restartSessionTimer: function() {
-    if (this.sessionExpired==true) return;
-    this.sessionEndTime = (new Date()).getTime() + this.options.TimeOut*60000;
-    if (this.sessionTimer) clearTimeout(this.sessionTimer);
-    this.updateSessionTimer();
-  },
-
-  updateSessionTimer: function() {
-    var now=(new Date()).getTime();
-    if (now > this.sessionEndTime) {
-      this.displaySessionTimer(Rico.getPhraseById("sessionExpired"));
-      this.timerMsg.style.backgroundColor="red";
-      this.sessionExpired=true;
-    } else {
-      var timeRemaining=Math.ceil((this.sessionEndTime - now) / 60000);
-      this.displaySessionTimer(timeRemaining);
-      this.sessionTimer=Rico.runLater(10000,this,'updateSessionTimer');
-    }
-  },
-
-  displaySessionTimer: function(msg) {
-    this.timerMsg.innerHTML='&nbsp;'+msg+'&nbsp;';
-  },
-
-  /**
-   * Update the grid with fresh data from the database, maintaining scroll position.
-   * @param resetRowCount indicates whether the total row count should be refreshed as well
-   */
-  refresh: function(resetRowCount) {
-    var lastGridPos=this.liveGrid.lastRowPos;\r
-    this.clear();
-    if (resetRowCount) {
-      this.setTotalRows(0);
-      this.foundRowCount = false;
-    }
-    this.liveGrid.clearBookmark();
-    this.liveGrid.clearRows();
-    this.fetch(lastGridPos);
-  },
-
-  /**
-   * Fetch data from database.
-   * @param offset position (row) within the dataset (-1=clear existing buffer before issuing request)
-   */
-  fetch: function(offset) {
-    Rico.log("AjaxSQL fetch: offset="+offset+', lastOffset='+this.lastOffset);
-    if (this.processingRequest) {
-      Rico.log("AjaxSQL fetch: queue request");
-      this.pendingRequest=offset;
-      return;
-    }
-    if (offset < 0) {
-      this.clear();
-      this.setTotalRows(0);
-      this.foundRowCount = false;
-      offset=0;
-    }
-    var lastOffset = this.lastOffset;
-    this.lastOffset = offset;
-    if (this.isInRange(offset)) {
-      Rico.log("AjaxSQL fetch: in buffer");
-      this.liveGrid.refreshContents(offset);
-      if (offset > lastOffset) {
-        if (offset+this.liveGrid.pageSize < this.endPos()-this.nearLimit) return;
-        if (this.endPos()==this.totalRows && this.foundRowCount) return;
-      } else if (offset < lastOffset) {
-        if (offset > this.startPos+this.nearLimit) return;
-        if (this.startPos==0) return;
-      } else return;
-    }
-    if (offset >= this.totalRows && this.foundRowCount) return;
-
-    this.processingRequest=true;
-    Rico.log("AjaxSQL fetch: processing offset="+offset);
-    var bufferStartPos = this.getFetchOffset(offset);
-    var fetchSize = this.getFetchSize(bufferStartPos);
-    var partialLoaded = false;
-
-    this.liveGrid.showMsg(this.options.waitMsg);
-    this.timeoutHandler = Rico.runLater(this.options.bufferTimeout, this, 'handleTimedOut');
-    this.ajaxOptions.parameters = this.formQueryHashSQL(bufferStartPos,fetchSize,this.options.fmt);
-    this.requestCount++;
-    Rico.log('sending req #'+this.requestCount);
-    var self=this;
-    if (typeof this.dataSource=='string') {
-      this.ajaxOptions.onComplete = function(xhr) { self.ajaxUpdate(bufferStartPos, xhr); };
-      new Rico.ajaxRequest(this.dataSource, this.ajaxOptions);
-    } else {
-      this.ajaxOptions.onComplete = function(newRows, newAttr, totalRows, errMsg) { self.jsUpdate(bufferStartPos, newRows, newAttr, totalRows, errMsg); };
-      this.dataSource(this.ajaxOptions);
-    }
-  },
-
-  formQueryHashSQL: function(startPos,fetchSize,fmt) {
-    var queryHash=this.formQueryHashXML(startPos,fetchSize);
-    queryHash[this.liveGrid.actionId]="query";
-    if (fmt) queryHash._fmt=fmt;
-
-    // sort
-    Rico.extend(queryHash,this.sortParm);
-
-    // filters
-    for (var n=0; n<this.liveGrid.columns.length; n++) {
-      var c=this.liveGrid.columns[n];
-      if (c.filterType == Rico.ColumnConst.UNFILTERED) continue;
-      var colnum=typeof(c.format.filterCol)=='number' ? c.format.filterCol : c.index;
-      queryHash['f['+colnum+'][op]']=c.filterOp;
-      queryHash['f['+colnum+'][len]']=c.filterValues.length;
-      for (var i=0; i<c.filterValues.length; i++) {
-        var fval=c.filterValues[i];
-        if (c.filterOp=='LIKE' && fval.indexOf('*')==-1) fval='*'+fval+'*';
-        queryHash['f['+colnum+']['+i+']']=fval;
-      }
-    }
-    return queryHash;
-  },
-
-  getFetchSize: function(adjustedOffset) {
-    var adjustedSize = 0;
-    if (adjustedOffset >= this.startPos) { //appending
-      var endFetchOffset = this.maxFetchSize + adjustedOffset;
-      adjustedSize = endFetchOffset - adjustedOffset;
-      if(adjustedOffset == 0 && adjustedSize < this.maxFetchSize)
-        adjustedSize = this.maxFetchSize;
-      Rico.log("getFetchSize/append, adjustedSize="+adjustedSize+" adjustedOffset="+adjustedOffset+' endFetchOffset='+endFetchOffset);
-    } else { //prepending
-      adjustedSize = Math.min(this.startPos - adjustedOffset,this.maxFetchSize);
-    }
-    return adjustedSize;
-  },
-
-  getFetchOffset: function(offset) {
-    var adjustedOffset = offset;
-    if (offset > this.startPos)
-      adjustedOffset = Math.max(offset, this.endPos());  //appending
-    else if (offset + this.maxFetchSize >= this.startPos)
-      adjustedOffset = Math.max(this.startPos - this.maxFetchSize, 0);  //prepending
-    return adjustedOffset;
-  },
-
-  updateBuffer: function(start, newRows, newAttr) {
-    Rico.log("updateBuffer: start="+start+", # of rows="+this.rcvdRows);
-    if (this.rows.length == 0) { // initial load
-      this.rows = newRows;
-      this.attr = newAttr;
-      this.startPos = start;
-    } else if (start > this.startPos) { //appending
-      if (this.startPos + this.rows.length < start) {
-        this.rows =  newRows;
-        this.attr = newAttr;
-        this.startPos = start;
-      } else {
-        this.rows = this.rows.concat( newRows.slice(0, newRows.length));
-        if (this.attr) this.attr = this.attr.concat( newAttr.slice(0, newAttr.length));
-        if (this.rows.length > this.maxBufferSize) {
-          var fullSize = this.rows.length;
-          this.rows = this.rows.slice(this.rows.length - this.maxBufferSize, this.rows.length);
-          if (this.attr) this.attr = this.attr.slice(this.attr.length - this.maxBufferSize, this.attr.length);
-          this.startPos = this.startPos +  (fullSize - this.rows.length);
-        }
-      }
-    } else { //prepending
-      if (start + newRows.length < this.startPos) {
-        this.rows =  newRows;
-      } else {
-        this.rows = newRows.slice(0, this.startPos).concat(this.rows);
-        if (this.maxBufferSize && this.rows.length > this.maxBufferSize)
-          this.rows = this.rows.slice(0, this.maxBufferSize);
-      }
-      this.startPos =  start;
-    }
-    this.size = this.rows.length;
-  },
-
-  sortBuffer: function(colnum) {
-    this.sortParm={};
-    var col=this.liveGrid.columns[colnum];
-    if (this.options.sortParmFmt) {
-      this.sortParm['sort_col']=col[this.options.sortParmFmt];
-      this.sortParm['sort_dir']=col.getSortDirection();
-    } else {
-      this.sortParm['s'+colnum]=col.getSortDirection();
-    }
-    this.clear();
-  },
-
-  printAllSQL: function(exportType) {
-    var parms=this.formQueryHashSQL(0,this.liveGrid.options.maxPrint,exportType);
-    parms.hidden=this.liveGrid.listInvisible('index').join(',');
-    var url=this.dataSource+'?'+Rico.toQueryString(parms);
-    window.open(url,'',this.liveGrid.options.exportWindow);
-  },
-
-  printVisibleSQL: function(exportType) {
-    var parms=this.formQueryHashSQL(this.liveGrid.contentStartPos-1, this.liveGrid.pageSize, exportType);
-    parms.hidden=this.liveGrid.listInvisible('index').join(',');
-    var url=this.dataSource+'?'+Rico.toQueryString(parms);
-    window.open(url,'',this.liveGrid.options.exportWindow);
-  },
-
-  // for datasource that is a javascript function
-  _printAll: function(exportType) {
-    this.liveGrid.exportStart();
-    this.ajaxOptions.parameters = this.formQueryHashSQL(0,this.liveGrid.options.maxPrint);
-    var self=this;
-    this.ajaxOptions.onComplete = function() { self._jsExport(exportType); };
-    this.dataSource(this.ajaxOptions);
-  },
-
-  _jsExport: function(exportType, newRows, newAttr, totalRows, errMsg) {
-    Rico.log("_jsExport: "+arguments.length);
-    if (errMsg) {
-      Rico.log("_jsExport: received error="+errMsg);
-      this.liveGrid.showMsg(Rico.getPhraseById("requestError",errMsg));
-      return;
-    }
-    this.exportBuffer(newRows,0);
-    this.liveGrid.exportFinish(exportType);
-  }
-
-};
diff --git a/ricoClient/js/minsrc/ricoLiveGridControls.js b/ricoClient/js/minsrc/ricoLiveGridControls.js
deleted file mode 100644 (file)
index cd2989c..0000000
+++ /dev/null
@@ -1,423 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-\r
-// -----------------------------------------------------\r
-//\r
-// Custom formatting for LiveGrid columns\r
-//\r
-// columnSpecs Usage: { type:'control', control:new Rico.TableColumn.CONTROLNAME() }\r
-//\r
-// -----------------------------------------------------\r
-
-Rico.TableColumn = {};
-\r
-Rico.TableColumn.checkboxKey = function(showKey) {
-  this.initialize(showKey);
-}
-
-Rico.TableColumn.checkboxKey.prototype = {
-/**
- * @class Custom formatting for a LiveGrid column.
- * Display unique key column as: &lt;checkbox&gt; &lt;key value&gt;
- * and keep track of which keys the user selects
- * Key values should not contain &lt;, &gt;, or &amp;
- * @constructs
- */
-  initialize: function(showKey) {
-    this._checkboxes=[];
-    this._spans=[];
-    this._KeyHash={};
-    this._showKey=showKey;\r
-  },
-
-  _create: function(gridCell,windowRow) {
-    this._checkboxes[windowRow]=Rico.createFormField(gridCell,'input','checkbox',this.liveGrid.tableId+'_chkbox_'+this.index+'_'+windowRow);
-    this._spans[windowRow]=Rico.createFormField(gridCell,'span',null,this.liveGrid.tableId+'_desc_'+this.index+'_'+windowRow);
-    this._clear(gridCell,windowRow);
-    Rico.eventBind(this._checkboxes[windowRow], 'click', Rico.eventHandle(this,'_onclick'));
-  },
-
-  _onclick: function(e) {
-    var elem=Event.element(e);
-    var windowRow=parseInt(elem.id.substr((elem.id.lastIndexOf('_',elem.id.length)+1)));  //faster than split
-    var v=this.getValue(windowRow);
-    if (elem.checked)
-      this._addChecked(v);
-    else
-      this._remChecked(v);
-  },
-
-  _clear: function(gridCell,windowRow) {
-    var box=this._checkboxes[windowRow];
-    box.checked=false;
-    box.style.display='none';
-    this._spans[windowRow].innerHTML='';
-  },
-
-  _display: function(v,gridCell,windowRow) {
-    var box=this._checkboxes[windowRow];
-    box.style.display='';
-    box.checked=this._KeyHash[v];
-    if (this._showKey) this._spans[windowRow].innerHTML=v;
-  },
-
-  _SelectedKeys: function() {
-    return Rico.keys(this._KeyHash);
-  },
-
-  _addChecked: function(k){\r
-    this._KeyHash[k]=1;\r
-  },\r
-\r
-  _remChecked: function(k){\r
-    delete this._KeyHash[k];\r
-  }\r
-}
-
-
-Rico.TableColumn.checkbox = function(checkedValue, uncheckedValue, defaultValue, readOnly)
-{
-  this.initialize(checkedValue, uncheckedValue, defaultValue, readOnly);
-}
-
-Rico.TableColumn.checkbox.prototype = {
-/**
- * @class display checkboxes for two-valued column (e.g. yes/no)
- * @constructs
- */
-  initialize: function(checkedValue, uncheckedValue, defaultValue, readOnly) {
-    this._checkedValue=checkedValue;
-    this._uncheckedValue=uncheckedValue;
-    this._defaultValue=defaultValue || false;
-    this._readOnly=readOnly || false;
-    this._checkboxes=[];
-  },
-
-  _create: function(gridCell,windowRow) {
-    this._checkboxes[windowRow]=Rico.createFormField(gridCell,'input','checkbox',this.liveGrid.tableId+'_chkbox_'+this.index+'_'+windowRow);
-    this._clear(gridCell,windowRow);
-    if (this._readOnly)
-      this._checkboxes[windowRow].disabled=true;
-    else
-      Rico.eventBind(this._checkboxes[windowRow], 'click', Rico.eventHandle(this,'_onclick'));
-  },
-
-  _onclick: function(e) {
-    var elem=Event.element(e);
-    var windowRow=parseInt(elem.id.substr((elem.id.lastIndexOf('_',elem.id.length)+1)));  //faster than split
-    var newval=elem.checked ? this._checkedValue : this._uncheckedValue;
-    this.setValue(windowRow,newval);
-  },
-
-  _clear: function(gridCell,windowRow) {
-    var box=this._checkboxes[windowRow];
-    box.checked=this._defaultValue;
-    box.style.display='none';
-  },
-
-  _display: function(v,gridCell,windowRow) {
-    var box=this._checkboxes[windowRow];
-    box.style.display='';
-    box.checked=(v==this._checkedValue);
-  }
-
-}
-
-
-Rico.TableColumn.textbox = function(boxSize, boxMaxLen, readOnly) {
-  this.initialize(boxSize, boxMaxLen, readOnly);
-}
-
-Rico.TableColumn.textbox.prototype = {
-/**
- * @class display value in a text box
- * @constructs
- */
-  initialize: function(boxSize, boxMaxLen, readOnly) {
-    this._boxSize=boxSize;
-    this._boxMaxLen=boxMaxLen;
-    this._readOnly=readOnly || false;
-    this._textboxes=[];
-  },
-
-  _create: function(gridCell,windowRow) {
-    var box=Rico.createFormField(gridCell,'input','text',this.liveGrid.tableId+'_txtbox_'+this.index+'_'+windowRow);
-    box.size=this._boxSize;
-    box.maxLength=this._boxMaxLen;
-    this._textboxes[windowRow]=box;
-    this._clear(gridCell,windowRow);
-    if (this._readOnly)
-      box.disabled=true;
-    else
-      Rico.eventBind(box, 'change', Rico.eventHandle(this,'_onchange'));
-  },
-
-  _onchange: function(e) {
-    var elem=Event.element(e);
-    var windowRow=parseInt(elem.id.substr((elem.id.lastIndexOf('_',elem.id.length)+1)));  //faster than split
-    this.setValue(windowRow,elem.value);
-  },
-
-  _clear: function(gridCell,windowRow) {
-    var box=this._textboxes[windowRow];
-    box.value='';
-    box.style.display='none';
-  },
-
-  _display: function(v,gridCell,windowRow) {
-    var box=this._textboxes[windowRow];
-    box.style.display='';
-    box.value=v;
-  }
-
-}
-
-
-Rico.TableColumn.HighlightCell = function(chkcol,chkval,highlightColor,highlightBackground,chkop) {
-  this.initialize(chkcol,chkval,highlightColor,highlightBackground,chkop);
-}
-
-Rico.TableColumn.HighlightCell.prototype = {
-/**
- * @class highlight a grid cell when a particular value is present in the specified column
- * @constructs
- */
-  initialize: function(chkcol,chkval,highlightColor,highlightBackground,chkop) {\r
-    this._chkcol=chkcol;\r
-    this._chkval=chkval;
-    this._chkop=chkop;\r
-    this._highlightColor=highlightColor;\r
-    this._highlightBackground=highlightBackground;\r
-  },\r
-\r
-  _clear: function(gridCell,windowRow) {\r
-    gridCell.style.color='';\r
-    gridCell.style.backgroundColor='';\r
-    gridCell.innerHTML='&nbsp;';\r
-  },\r
-\r
-  _display: function(v,gridCell,windowRow) {\r
-    var gridval=this.liveGrid.buffer.getWindowValue(windowRow,this._chkcol);\r
-    var match;\r
-    switch(this._chkop){
-        case '!=':
-          match=(gridval!=this._chkval);
-          break;
-        case '>':
-          match=(gridval>this._chkval);
-          break;
-        case '<':
-          match=(gridval<this._chkval);
-          break;
-        case '>=':
-          match=(gridval>=this._chkval);
-          break;
-        case '<=':
-          match=(gridval<=this._chkval);
-          break;
-        case 'abs>':
-          match=(Math.abs(gridval)>this._chkval);
-          break;
-        case 'abs<':
-          match=(Math.abs(gridval)<this._chkval);
-          break;
-        case 'abs>=':
-          match=(Math.abs(gridval)>=this._chkval);
-          break;
-        case 'abs<=':
-          match=(Math.abs(gridval)<=this._chkval);
-          break;
-        default:
-          match=(gridval==this._chkval);
-          break;
-    }
-    gridCell.style.color=match ? this._highlightColor : '';\r
-    gridCell.style.backgroundColor=match ? this._highlightBackground : '';\r
-    gridCell.innerHTML=this._format(v);\r
-  }\r
-}
-
-
-Rico.TableColumn.bgColor = function() {
-}
-
-Rico.TableColumn.bgColor.prototype = {
-/**
- * @class database value contains a css color name/value
- */
- _clear: function(gridCell,windowRow) {
-    gridCell.style.backgroundColor='';
-  },
-
-  _display: function(v,gridCell,windowRow) {
-    gridCell.style.backgroundColor=v;
-  }
-
-}
-
-
-Rico.TableColumn.link = function(href,target,linktext) {
-  this.initialize(href,target,linktext);
-}
-
-Rico.TableColumn.link.prototype = {
-/**
- * @class database value contains a url to another page
- * @constructs
- */
-  initialize: function(href,target,linktext) {
-    this._href=href;
-    this._target=target;
-    this._linktext=linktext;
-    this._anchors=[];
-  },
-
-  _create: function(gridCell,windowRow) {
-    this._anchors[windowRow]=Rico.createFormField(gridCell,'a',null,this.liveGrid.tableId+'_a_'+this.index+'_'+windowRow);
-    if (this._target) this._anchors[windowRow].target=this._target;
-    this._clear(gridCell,windowRow);
-  },
-
-  _clear: function(gridCell,windowRow) {
-    this._anchors[windowRow].href='';
-    this._anchors[windowRow].innerHTML='';
-  },
-
-  _display: function(v,gridCell,windowRow) {
-    var buf=this.liveGrid.buffer;
-    var href=this._href=='self' ? v : this._href.replace(/\{\d+\}/g,
-      function ($1) {
-        var colIdx=parseInt($1.substr(1),10);
-        return encodeURIComponent(buf.getWindowValue(windowRow,colIdx));
-      }
-    );
-    var desc=this._linktext || v;
-    if (href && desc) {
-      this._anchors[windowRow].href=href;
-      this._anchors[windowRow].innerHTML=desc;
-    } else {
-      this._clear(gridCell,windowRow);
-    }
-  }
-
-};
-
-
-Rico.TableColumn.image = function(prefix,suffix) {
-  this.initialize(prefix,suffix);
-};
-
-Rico.TableColumn.image.prototype = {
-/**
- * @class database value contains a url to an image
- * @constructs
- */
-  initialize: function(prefix,suffix) {
-    this._img=[];
-    this._prefix=prefix || '';
-    this._suffix=suffix || '';
-  },
-
-  _create: function(gridCell,windowRow) {
-    this._img[windowRow]=Rico.createFormField(gridCell,'img',null,this.liveGrid.tableId+'_img_'+this.index+'_'+windowRow);
-    this._clear(gridCell,windowRow);
-  },
-
-  _clear: function(gridCell,windowRow) {
-    var img=this._img[windowRow];
-    img.style.display='none';
-    img.src='';
-  },
-
-  _display: function(v,gridCell,windowRow) {
-    var img=this._img[windowRow];
-    this._img[windowRow].src=this._prefix+v+this._suffix;
-    img.style.display='';
-  }
-
-};
-
-
-Rico.TableColumn.lookup = function(map, defaultCode, defaultDesc) {
-  this.initialize(map, defaultCode, defaultDesc);
-};
-
-Rico.TableColumn.lookup.prototype = {
-/**
- * @class map a database value to a display value
- * @constructs
- */
-  initialize: function(map, defaultCode, defaultDesc) {
-    this._map=map;
-    this._defaultCode=defaultCode || '';
-    this._defaultDesc=defaultDesc || '&nbsp;';
-    var self=this;
-    this._sortfunc=function(v) { return self._sortvalue(v); };
-    this._codes=[];
-    this._descriptions=[];
-  },
-
-  _create: function(gridCell,windowRow) {
-    this._descriptions[windowRow]=Rico.createFormField(gridCell,'span',null,this.liveGrid.tableId+'_desc_'+this.index+'_'+windowRow);
-    this._codes[windowRow]=Rico.createFormField(gridCell,'input','hidden',this.liveGrid.tableId+'_code_'+this.index+'_'+windowRow);
-    this._clear(gridCell,windowRow);
-  },
-
-  _clear: function(gridCell,windowRow) {
-    this._codes[windowRow].value=this._defaultCode;
-    this._descriptions[windowRow].innerHTML=this._defaultDesc;
-  },
-
-  _sortvalue: function(v) {
-    return this._getdesc(v).replace(/&amp;/g, '&').replace(/&lt;/g,'<').replace(/&gt;/g,'>').replace(/&nbsp;/g,' ');
-  },
-
-  _getdesc: function(v) {
-    var desc=this._map[v];
-    return (typeof desc=='string') ? desc : this._defaultDesc;
-  },
-
-  _export: function(v) {
-    return this._getdesc(v);
-  },
-
-  _display: function(v,gridCell,windowRow) {
-    this._codes[windowRow].value=v;
-    this._descriptions[windowRow].innerHTML=this._getdesc(v);
-  }
-
-};
-
-
-
-Rico.TableColumn.MultiLine = function() {
-};
-
-Rico.TableColumn.MultiLine.prototype = {
-/**
- * @class Fix issues with multiline content in IE
- */
-  _display: function(v,gridCell,windowRow) {\r
-    var newdiv = document.createElement("div");\r
-    newdiv.innerHTML = this._format(v);\r
-    newdiv.style.height='100%';\r
-    if (gridCell.firstChild)\r
-      gridCell.replaceChild(newdiv, gridCell.firstChild);\r
-    else\r
-      gridCell.appendChild(newdiv);\r
-  }\r
-
-};
diff --git a/ricoClient/js/minsrc/ricoLiveGridForms.js b/ricoClient/js/minsrc/ricoLiveGridForms.js
deleted file mode 100644 (file)
index 017214c..0000000
+++ /dev/null
@@ -1,1130 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-if(typeof Rico=='undefined') throw("LiveGridForms requires the Rico JavaScript framework");
-
-
-Rico.TableEdit = function(liveGrid) {
-  this.initialize(liveGrid);
-}
-
-Rico.TableEdit.prototype = {
-/**
- * @class Supports editing LiveGrid data.
- * @constructs
- */
-  initialize: function(liveGrid) {
-    Rico.log('Rico.TableEdit initialize: '+liveGrid.tableId);
-    this.grid=liveGrid;
-    this.options = {
-      maxDisplayLen    : 20,    // max displayed text field length
-      panelHeight      : 200,   // size of tabbed panels
-      panelWidth       : 500,
-      compact          : false,    // compact corners
-      RecordName       : Rico.getPhraseById("record"),
-      updateURL        : window.location.href, // default is that updates post back to the generating page
-      showSaveMsg      : 'errors'  // disposition of database update responses (full - show full response, errors - show full response for errors and short response otherwise)
-    };
-    Rico.extend(this.options, liveGrid.options);
-    var self=this;
-    this.hasWF2=(document.implementation && document.implementation.hasFeature && document.implementation.hasFeature('WebForms', '2.0'));
-    this.menu=liveGrid.menu;
-    this.menu.options.dataMenuHandler=function(grid,r,c,onBlankRow) { return self.editMenu(grid,r,c,onBlankRow); };
-    this.menu.ignoreClicks();
-    this.editText=Rico.getPhraseById("editRecord",this.options.RecordName);
-    this.cloneText=Rico.getPhraseById("cloneRecord",this.options.RecordName);
-    this.delText=Rico.getPhraseById("deleteRecord",this.options.RecordName);
-    this.addText=Rico.getPhraseById("addRecord",this.options.RecordName);
-    this.buttonHover=new Rico.HoverSet();
-    this.dateRegExp=/^\s*(\w+)(\W)(\w+)(\W)(\w+)/i;
-    Rico.EditControls.atLoad();
-    this.createKeyArray();
-    this.createEditDiv();
-    this.saveMsg=Rico.$(liveGrid.tableId+'_savemsg');
-    Rico.eventBind(document,"click", Rico.eventHandle(this,'clearSaveMsg'));
-    this.extraMenuItems=[];
-    this.responseHandler=function() { self.processResponse(); };
-    Rico.log("Rico.TableEdit.initialize complete, hasWF2="+this.hasWF2);
-  },
-
-  createKeyArray: function() {
-    this.keys=[];
-    for (var i=0; i<this.grid.columns.length; i++) {
-      if (this.grid.columns[i].format && this.grid.columns[i].format.isKey)
-        this.keys.push({colidx:i});
-    }
-  },
-
-  createEditDiv: function() {
-
-    // create popup form
-
-    this.requestCount=1;
-    this.formPopup=new Rico.Window('', {overflow: this.options.panels ? 'hidden' : 'auto'});
-    Rico.addClass(this.formPopup.content,'ricoLG_editDiv');
-    if (this.options.canEdit || this.options.canAdd) {
-      this.startForm();
-      this.createForm(this.form);
-    } else {
-      var buttonClose=this.createButton(Rico.getPhraseById("close"));
-      Rico.eventBind(buttonClose,"click", Rico.eventHandle(this,'cancelEdit'), false);
-      this.createForm(this.formPopup.contentDiv);
-    }
-    this.editDivCreated=true;
-
-    // create responseDialog
-
-    this.responseDialog = this.grid.createDiv('editResponse',document.body);
-    this.responseDialog.style.display='none';
-
-    var buttonOK = document.createElement('button');
-    buttonOK.appendChild(document.createTextNode('OK'));
-    Rico.eventBind(buttonOK,"click", Rico.eventHandle(this,'ackResponse'));
-    this.responseDialog.appendChild(buttonOK);
-
-    this.responseDiv = this.grid.createDiv('editResponseText',this.responseDialog);
-
-    if (this.panelGroup) {
-      Rico.log("createEditDiv complete, requestCount="+this.requestCount);
-      Rico.runLater(50,this,'initPanelGroup');
-    }
-  },
-
-  initPanelGroup: function() {
-    this.requestCount--;
-    Rico.log("initPanelGroup: "+this.requestCount);
-    if (this.requestCount>0) return;
-    var wi=parseInt(this.options.panelWidth,10);
-    if (this.form) {
-      //this.form.style.width=(wi+10)+'px';
-      if (Rico.isWebKit) this.formPopup.container.style.display='block';  // this causes display to flash briefly
-      this.options.bgColor = Rico.Color.createColorFromBackground(this.form).toString();
-    }
-    this.formPopup.container.style.display='none';
-    //this.options.panelHdrWidth=(Math.floor(wi / this.options.panels.length)-20)+'px';
-    this.formPanels=new Rico.TabbedPanel(this.panelGroup, this.options);
-  },
-
-  notEmpty: function(v) {
-    return typeof(v)!='undefined';
-  },
-
-  startForm: function() {
-    this.form = document.createElement('form');
-    /** @ignore */
-    this.form.onsubmit=function() {return false;};
-    this.form.autocomplete="off"; // seems to fix "Permission denied..." errors in FF
-    this.formPopup.contentDiv.appendChild(this.form);
-
-    var tab = document.createElement('div');
-    tab.className='ButtonBar';
-    var button=tab.appendChild(this.createButton(Rico.getPhraseById("saveRecord",this.options.RecordName)));
-    Rico.eventBind(button,"click", Rico.eventHandle(this,'TESubmit'), false);
-    button=tab.appendChild(this.createButton(Rico.getPhraseById("cancel")));
-    Rico.eventBind(button,"click", Rico.eventHandle(this,'cancelEdit'), false);
-    this.form.appendChild(tab);
-
-    // hidden fields
-    this.hiddenFields = document.createElement('div');
-    this.hiddenFields.style.display='none';
-    this.action = this.appendHiddenField(this.grid.actionId,'');
-    var i,fldSpec;
-    for (i=0; i<this.grid.columns.length; i++) {
-      fldSpec=this.grid.columns[i].format;
-      if (fldSpec && fldSpec.FormView && fldSpec.FormView=="hidden")
-        this.appendHiddenField(fldSpec.FieldName,fldSpec.ColData);
-    }
-    for (var k=0; k<this.keys.length; k++) {
-      this.keys[k].keyField = this.appendHiddenField('_k'+this.keys[k].colidx,'');
-    }
-    this.form.appendChild(this.hiddenFields);
-  },
-
-  createButton: function(buttonLabel) {
-    var button = document.createElement('a');
-    button.href='javascript:void(0)';
-    button.innerHTML=buttonLabel;
-    button.className='RicoButton';
-    if (Rico.theme.button) Rico.addClass(button,Rico.theme.button);
-    this.buttonHover.add(button);
-    return button;
-  },
-
-  createPanel: function(i) {
-    var hasFields=false;
-    for (var j=0; j<this.grid.columns.length; j++) {
-      var fldSpec=this.grid.columns[j].format;
-      if (!fldSpec) continue;
-      if (!fldSpec.EntryType) continue;
-      if (fldSpec.EntryType=='H') continue;
-      if (fldSpec.FormView && fldSpec.FormView=="hidden") continue;
-      var panelIdx=fldSpec.panelIdx || 0;
-      if (panelIdx==i) {
-        hasFields=true;
-        break;
-      }
-    }
-    if (!hasFields) return null;
-    this.panelHdr[i] = document.createElement('li');
-    this.panelHdr[i].innerHTML=this.options.panels[i];
-    this.panelHdrs.appendChild(this.panelHdr[i]);
-    this.panelContent[i] = document.createElement('div');
-    this.panelContents.appendChild(this.panelContent[i]);
-    this.panelActualIdx[i]=this.panelCnt++;
-    return this.createFormTable(this.panelContent[i],'tabContent');
-  },
-
-  createForm: function(parentDiv) {
-    var i,div,fldSpec,panelIdx,tables=[];
-    this.panelCnt=0;
-    this.panelHdr=[];
-    this.panelContent=[];
-    if (this.options.panels) {
-      this.panelGroup = document.createElement('div');
-      this.panelGroup.className='tabPanelGroup';
-      this.panelHdrs = document.createElement('ul');
-      this.panelGroup.appendChild(this.panelHdrs);
-      this.panelContents = document.createElement('div');
-      this.panelContents.className='tabContentContainer';
-      this.panelGroup.appendChild(this.panelContents);
-      this.panelActualIdx=[];
-      parentDiv.appendChild(this.panelGroup);
-      if (this.grid.direction=='rtl') {
-        for (i=this.options.panels.length-1; i>=0; i--) {
-          tables[i]=this.createPanel(i);
-        }
-      } else {
-        for (i=0; i<this.options.panels.length; i++) {
-          tables[i]=this.createPanel(i);
-        }
-      }
-      parentDiv.appendChild(this.panelGroup);
-    } else {
-      div=document.createElement('div');
-      div.className='noTabContent';
-      tables[0]=this.createFormTable(div);
-      parentDiv.appendChild(div);
-    }
-    for (i=0; i<this.grid.columns.length; i++) {
-      fldSpec=this.grid.columns[i].format;
-      if (!fldSpec) continue;
-      panelIdx=fldSpec.panelIdx || 0;
-      if (tables[panelIdx]) this.appendFormField(this.grid.columns[i],tables[panelIdx]);
-      if (typeof fldSpec.pattern=='string') {
-        switch (fldSpec.pattern) {
-          case 'email':
-            fldSpec.regexp=/^[_a-zA-Z0-9-]+(\.[_a-zA-Z0-9-]+)*@[a-zA-Z0-9-]+(\.[a-zA-Z0-9-]+)*\.(([0-9]{1,3})|([a-zA-Z]{2,3})|(aero|coop|info|museum|name))$/;
-            break;
-          case 'float-unsigned':
-            fldSpec.regexp=/^\d+(\.\d+)?$/;
-            break;
-          case 'float-signed':
-            fldSpec.regexp=/^[-+]?\d+(\.\d+)?$/;
-            break;
-          case 'int-unsigned':
-            fldSpec.regexp=/^\d+$/;
-            break;
-          case 'int-signed':
-            fldSpec.regexp=/^[-+]?\d+$/;
-            break;
-          default:
-            fldSpec.regexp=new RegExp(fldSpec.pattern);
-            break;
-        }
-      }
-    }
-  },
-
-  createFormTable: function(div) {
-    var tab=document.createElement('table');
-    tab.border=0;
-    div.appendChild(tab);
-    return tab;
-  },
-
-  appendHiddenField: function(name,value) {
-    var field=Rico.createFormField(this.hiddenFields,'input','hidden',name,name);
-    field.value=value;
-    return field;
-  },
-
-  appendFormField: function(column, table) {
-    var fmt=column.format;
-    if (!fmt.EntryType) return;
-    if (fmt.EntryType=="H") return;
-    if (fmt.FormView) return;
-    Rico.log('appendFormField: '+column.displayName+' - '+fmt.EntryType);
-    var row = fmt.noFormBreak && table.rows.length > 0 ? table.rows[table.rows.length-1] : table.insertRow(-1);
-    var hdr = row.insertCell(-1);
-    column.formLabel=hdr;
-    if (hdr.noWrap) hdr.noWrap=true;
-    var entry = row.insertCell(-1);
-    if (entry.noWrap) entry.noWrap=true;
-    hdr.id='lbl_'+fmt.FieldName;
-    var field, name=fmt.FieldName;
-    switch (fmt.EntryType) {
-      case 'TA':
-      case 'tinyMCE':
-        field=Rico.createFormField(entry,'textarea',null,name);
-        field.cols=fmt.TxtAreaCols;
-        field.rows=fmt.TxtAreaRows;
-        field.innerHTML=fmt.ColData;
-        hdr.style.verticalAlign='top';
-        break;
-      case 'R':
-      case 'RL':
-        field=Rico.createFormField(entry,'div',null,name);
-        if (fmt.DescriptionField) field.RicoUpdate=fmt.DescriptionField;
-        if (fmt.MultiSelect) Rico.addClass(field, 'MultiSelect');
-        if (fmt.isNullable && !fmt.MultiSelect) this.addSelectNone(field);
-        this.selectValuesRequest(field,column);
-        break;
-      case 'N':
-        field=Rico.createFormField(entry,'select',null,name);
-        if (fmt.isNullable) this.addSelectNone(field);
-        Rico.eventBind(field,"change", Rico.eventHandle(this,'checkSelectNew'));
-        this.selectValuesRequest(field,column);
-        field=document.createElement('span');
-        field.className='ricoEditLabel';
-        field.id='labelnew__'+fmt.FieldName;
-        field.innerHTML='&nbsp;&nbsp;&nbsp;'+Rico.getPhraseById('formNewValue').replace(' ','&nbsp;');
-        entry.appendChild(field);
-        name='textnew__'+fmt.FieldName;
-        field=Rico.createFormField(entry,'input','text',name,name);
-        break;
-      case 'S':
-      case 'SL':
-        if (fmt.ReadOnly) {
-          field=Rico.createFormField(entry,'input','text',name,name);
-          this.initField(field,fmt);
-        } else {
-          field=Rico.createFormField(entry,'select',null,name);
-          if (fmt.MultiSelect) field.multiple=true;
-          if (fmt.SelectRows) field.size=parseInt(fmt.SelectRows,10);
-          if (fmt.isNullable && !fmt.MultiSelect) this.addSelectNone(field);
-          if (fmt.DescriptionField) {
-            field.RicoUpdate=fmt.DescriptionField;
-            Rico.eventBind(field,"change", Rico.eventHandle(this,'selectClick'), false);
-          }
-          this.selectValuesRequest(field,column);
-        }
-        break;
-      case 'D':
-        if (!fmt.isNullable) fmt.required=true;
-        if (!fmt.dateFmt) fmt.dateFmt=Rico.dateFmt;
-        if (!fmt.Help) fmt.Help=fmt.dateFmt;
-        if (typeof fmt.min=='string') fmt.min=Rico.setISO8601(fmt.min) || new Date(fmt.min);
-        if (typeof fmt.max=='string') fmt.max=Rico.setISO8601(fmt.max) || new Date(fmt.max);
-        if (this.hasWF2) {
-          field=Rico.createFormField(entry,'input','date',name,name);
-          field.required=fmt.required;
-          if (fmt.min) field.min=Rico.toISO8601String(fmt.min,3);
-          if (fmt.max) field.max=Rico.toISO8601String(fmt.max,3);
-          field.required=fmt.required;
-          fmt.SelectCtl=null;  // use the WebForms calendar instead of the Rico calendar
-        } else {
-          field=Rico.createFormField(entry,'input','text',name,name);
-        }
-        this.initField(field,fmt);
-        break;
-      case 'I':
-        if (!fmt.isNullable) fmt.required=true;
-        if (!fmt.pattern) fmt.pattern='int-signed';
-        if (this.hasWF2) {
-          field=Rico.createFormField(entry,'input','number',name,name);
-          field.required=fmt.required;
-          field.min=fmt.min;
-          field.max=fmt.max;
-          field.step=1;
-        } else {
-          field=Rico.createFormField(entry,'input','text',name,name);
-        }
-        if (typeof fmt.min=='string') fmt.min=parseInt(fmt.min,10);
-        if (typeof fmt.max=='string') fmt.max=parseInt(fmt.max,10);
-        this.initField(field,fmt);
-        break;
-      case 'F':
-        if (!fmt.isNullable) fmt.required=true;
-        if (!fmt.pattern) fmt.pattern='float-signed';
-        field=Rico.createFormField(entry,'input','text',name,name);
-        this.initField(field,fmt);
-        if (typeof fmt.min=='string') fmt.min=parseFloat(fmt.min);
-        if (typeof fmt.max=='string') fmt.max=parseFloat(fmt.max);
-        break;
-      default:
-        field=Rico.createFormField(entry,'input','text',name,name);
-        if (!fmt.isNullable && fmt.EntryType!='T') fmt.required=true;
-        this.initField(field,fmt);
-        break;
-    }
-    if (field) {
-      if (fmt.SelectCtl)
-        Rico.EditControls.applyTo(column,field);
-    }
-    var hdrSuffix='';
-    hdr.className='ricoEditLabel';
-    if (fmt.Help) {
-      hdr.title=fmt.Help;
-      hdrSuffix="&nbsp;<img src='"+Rico.imgDir+"info_icon.gif'>";
-      Rico.addClass(hdr,'ricoHelp');
-    }
-    var hdrText=fmt.EntryType.length>1 && fmt.EntryType.charAt(1)=='L' ? column.next.displayName : column.displayName;
-    hdr.innerHTML=hdrText+hdrSuffix;
-  },
-
-  addSelectNone: function(field) {
-    this.addSelectOption(field,this.options.TableSelectNone,Rico.getPhraseById("selectNone"));
-  },
-
-  initField: function(field,fmt) {
-    if (fmt.Length) {
-      field.maxLength=fmt.Length;
-      field.size=Math.min(fmt.Length, this.options.maxDisplayLen);
-    }
-    field.value=fmt.ColData;
-  },
-  
-  selectClick: function(e) {
-    var SelObj=Rico.eventElement(e);
-    if (SelObj.readOnly) {
-      Rico.eventStop(e);
-      return false;
-    }
-    if (SelObj.RicoUpdate) {
-      var opt=SelObj.options[SelObj.selectedIndex];
-      Rico.$(SelObj.RicoUpdate).value=opt.innerHTML;
-    }
-  },
-  
-  radioClick: function(e) {
-    var ChkBoxObj=Rico.eventElement(e);
-    if (ChkBoxObj.readOnly) {
-      Rico.eventStop(e);
-      return false;
-    }
-    var container=Rico.getParentByTagName(ChkBoxObj,'div');
-    if (container.RicoUpdate) {
-      Rico.$(container.RicoUpdate).value=ChkBoxObj.nextSibling.innerHTML;
-    }
-  },
-
-  checkSelectNew: function(e) {
-    this.updateSelectNew(Rico.eventElement(e));
-  },
-
-  updateSelectNew: function(SelObj) {
-    var vis=(SelObj.value==this.options.TableSelectNew) ? "" : "hidden";
-    Rico.$("labelnew__" + SelObj.id).style.visibility=vis;
-    Rico.$("textnew__" + SelObj.id).style.visibility=vis;
-  },
-
-  selectValuesRequest: function(elem,column) {
-    var fldSpec=column.format;
-    if (fldSpec.SelectValues) {
-      var valueList=fldSpec.SelectValues.split(',');
-      for (var i=0; i<valueList.length; i++)
-        this.addSelectOption(elem,valueList[i],valueList[i],i);
-    } else {
-      this.requestCount++;
-      var options={}, self=this;
-      Rico.extend(options, this.grid.buffer.ajaxOptions);
-      options.parameters = {id: this.grid.tableId, offset: '0', page_size: '-1', edit: column.index};
-      options.parameters[this.grid.actionId]="query";
-      options.onComplete = function(request) { self.selectValuesUpdate(elem,request); };
-      new Rico.ajaxRequest(this.grid.buffer.dataSource, options);
-      Rico.log("selectValuesRequest: "+fldSpec.FieldName);
-    }
-  },
-
-  selectValuesUpdate: function(elem,request) {
-    var response = request.responseXML.getElementsByTagName("ajax-response");
-    Rico.log("selectValuesUpdate: "+request.status);
-    if (response == null || response.length != 1) return;
-    response=response[0];
-    var error = response.getElementsByTagName('error');
-    if (error.length > 0) {
-      var errmsg=Rico.getContentAsString(error[0],this.grid.buffer.isEncoded);
-      Rico.log("Data provider returned an error:\n"+errmsg);
-      alert(Rico.getPhraseById("requestError",errmsg));
-      return;
-    }
-    response=response.getElementsByTagName('response')[0];
-    var rowsElement = response.getElementsByTagName('rows')[0];
-    var rows = this.grid.buffer.dom2jstable(rowsElement);
-    Rico.log("selectValuesUpdate: id="+elem.id+' rows='+rows.length);
-    for (var i=0; i<rows.length; i++) {
-      if (rows[i].length>0) {
-        var c0=rows[i][0];
-        var c1=(rows[i].length>1) ? rows[i][1] : c0;
-        this.addSelectOption(elem,c0,c1,i);
-      }
-    }
-    if (Rico.$('textnew__'+elem.id))
-      this.addSelectOption(elem,this.options.TableSelectNew,Rico.getPhraseById("selectNewVal"));
-    if (this.panelGroup)
-      Rico.runLater(50,this,'initPanelGroup');
-  },
-
-  addSelectOption: function(elem,value,text,idx) {
-    switch (elem.tagName.toLowerCase()) {
-      case 'div':
-        var opt=Rico.createFormField(elem,'input', Rico.hasClass(elem, 'MultiSelect') ? 'checkbox' : 'radio', elem.id+'_'+idx, elem.id);
-        opt.value=value;
-        var lbl=document.createElement('label');
-        lbl.innerHTML=text;
-        lbl.htmlFor=opt.id;
-        elem.appendChild(lbl);
-        Rico.eventBind(opt,"click", Rico.eventHandle(this,'radioClick'), false);
-        break;
-      case 'select':
-        Rico.addSelectOption(elem,value,text);
-        break;
-    }
-  },
-
-  clearSaveMsg: function() {
-    if (this.saveMsg) this.saveMsg.innerHTML="";
-  },
-
-  addMenuItem: function(menuText,menuAction,enabled) {
-    this.extraMenuItems.push({menuText:menuText,menuAction:menuAction,enabled:enabled});
-  },
-
-  editMenu: function(grid,r,c,onBlankRow) {
-    this.clearSaveMsg();
-    if (this.grid.buffer.sessionExpired==true || this.grid.buffer.startPos<0) return false;
-    this.rowIdx=r;
-    var elemTitle=Rico.$('pageTitle');
-    var pageTitle=elemTitle ? elemTitle.innerHTML : document.title;
-    this.menu.addMenuHeading(pageTitle);
-    var self=this;
-    if (onBlankRow==false) {
-      for (var i=0; i<this.extraMenuItems.length; i++) {
-        this.menu.addMenuItem(this.extraMenuItems[i].menuText,this.extraMenuItems[i].menuAction,this.extraMenuItems[i].enabled);
-      }
-      this.menu.addMenuItem(this.editText, function() { self.editRecord(); },this.canEdit(r));
-      this.menu.addMenuItem(this.delText, function() { self.deleteRecord(); },this.canDelete(r));
-      if (this.options.canClone) {
-        this.menu.addMenuItem(this.cloneText, function() { self.cloneRecord(); },this.canAdd(r) && this.canEdit(r));
-      }
-    }
-    this.menu.addMenuItem(this.addText, function() { self.addRecord(); },this.canAdd(r));
-    return true;
-  },
-  
-  canAdd: function(r) {
-    return (typeof this.options.canAdd=='function') ? this.options.canAdd(r) : this.options.canAdd;
-  },
-
-  canEdit: function(r) {
-    return (typeof this.options.canEdit=='function') ? this.options.canEdit(r) : this.options.canEdit;
-  },
-
-  canDelete: function(r) {
-    return (typeof this.options.canDelete=='function') ? this.options.canDelete(r) : this.options.canDelete;
-  },
-
-  cancelEdit: function(e) {
-    Rico.eventStop(e);
-    for (var i=0; i<this.grid.columns.length; i++) {
-      if (this.grid.columns[i].format && this.grid.columns[i].format.SelectCtl)
-        Rico.EditControls.close(this.grid.columns[i].format.SelectCtl);
-    }
-    this.makeFormInvisible();
-    this.grid.highlightEnabled=true;
-    this.menu.cancelmenu();
-    return false;
-  },
-
-  setField: function(fldnum,fldvalue) {
-    var fldSpec=this.grid.columns[fldnum].format;
-    var e=Rico.$(fldSpec.FieldName);
-    var a,i,o,elems,opts,txt;
-    if (!e) return;
-    Rico.log('setField: '+fldSpec.FieldName+'='+fldvalue);
-    switch (e.tagName.toUpperCase()) {
-      case 'DIV':
-        elems=e.getElementsByTagName('INPUT');
-        o={}
-        if (fldSpec.MultiSelect && fldvalue) {
-          a=fldvalue.split(',');
-          for (var i=0; i<a.length; i++) o[a[i]]=1;
-        } else {
-          o[fldvalue]=1;
-        }
-        for (i=0; i<elems.length; i++)
-          elems[i].checked=o[elems[i].value]==1;
-        break;
-      case 'INPUT':
-        if (fldSpec.EntryType=='D' && fldvalue!=fldSpec.ColData) {
-          // remove time data if it exists
-          a=fldvalue.split(/\s|T/);
-          fldvalue=a[0];
-          if (this.isTextInput(e)) {
-            var d=Rico.setISO8601(fldvalue);
-            if (d) fldvalue=Rico.formatDate(d,fldSpec.dateFmt);
-          }
-        }
-        e.value=fldvalue;
-        break;
-      case 'SELECT':
-        opts=e.options;
-        //alert('setField SELECT: id='+e.id+'\nvalue='+fldvalue+'\nopt cnt='+opts.length)
-        o={}
-        if (fldSpec.MultiSelect && fldvalue) {
-          a=fldvalue.split(',');
-          for (var i=0; i<a.length; i++) o[a[i]]=1;
-          for (i=0; i<opts.length; i++)
-            opts[i].selected=o[opts[i].value]==1;
-        } else {
-          for (i=0; i<opts.length; i++) {
-            if (opts[i].value==fldvalue) {
-              e.selectedIndex=i;
-              break;
-            }
-          }
-        }
-        if (fldSpec.EntryType=='N') {
-          txt=Rico.$('textnew__'+e.id);
-          if (!txt) alert('Warning: unable to find id "textnew__'+e.id+'"');
-          txt.value=fldvalue;
-          if (e.selectedIndex!=i) e.selectedIndex=opts.length-1;
-          this.updateSelectNew(e);
-        }
-        return;
-      case 'TEXTAREA':
-        e.value=fldvalue;
-        if (fldSpec.EntryType=='tinyMCE' && typeof(tinyMCE)!='undefined' && this.initialized) {
-          if (tinyMCE.updateContent) {
-            tinyMCE.updateContent(e.id);  // version 2.x
-          } else {
-            tinyMCE.execInstanceCommand(e.id, 'mceSetContent', false, fldvalue);  // version 3.x
-          }
-        }
-        return;
-    }
-  },
-
-  setReadOnly: function(action) {
-    for (var ro,i=0; i<this.grid.columns.length; i++) {
-      var fldSpec=this.grid.columns[i].format;
-      if (!fldSpec) continue;
-      var e=Rico.$(fldSpec.FieldName);
-      if (!e) continue;
-      switch (action) {
-        case 'ins': ro=!fldSpec.Writeable || fldSpec.ReadOnly || fldSpec.UpdateOnly; break;
-        case 'upd': ro=!fldSpec.Writeable || fldSpec.ReadOnly || fldSpec.InsertOnly; break;
-        default:    ro=false; break;
-      }
-      switch (e.tagName.toUpperCase()) {
-        case 'DIV':
-          var elems=e.getElementsByTagName('INPUT');
-          for (var j=0; j<elems.length; j++) {
-            elems[j].disabled=ro;
-          }
-          break;
-        case 'SELECT':
-          if (fldSpec.EntryType=='N') {
-            var txt=Rico.$('textnew__'+e.id);
-            txt.disabled=ro;
-          }
-          e.disabled=ro;
-          break;
-        case 'TEXTAREA':
-        case 'INPUT':
-          e.disabled=ro;
-          if (fldSpec.selectIcon) fldSpec.selectIcon.style.display=ro ? 'none' : '';
-          break;
-      }
-    }
-  },
-
-  hideResponse: function(msg) {
-    this.responseDiv.innerHTML=msg;
-    this.responseDialog.style.display='none';
-  },
-
-  showResponse: function() {
-    var offset=Rico.cumulativeOffset(this.grid.outerDiv);
-    offset.top+=Rico.docScrollTop();
-    this.responseDialog.style.top=offset.top+"px";
-    this.responseDialog.style.left=offset.left+"px";
-    this.responseDialog.style.display='';
-  },
-
-  processResponse: function(xhr) {
-    var responseText,success=true;
-    Rico.log('Processing response from form submittal');
-    this.responseDiv.innerHTML=xhr.responseText;
-    var respNodes=Rico.select('.ricoFormResponse',this.responseDiv);
-    if (respNodes) {
-      // generate a translated response
-      var phraseId=Rico.trim(respNodes[0].className).split(/\s+/)[1];
-      responseText=Rico.getPhraseById(phraseId,this.options.RecordName);
-    } else {
-      // present the response as sent from the server (untranslated)
-      var ch=this.responseDiv.childNodes;
-      for (var i=ch.length-1; i>=0; i--) {
-        if (ch[i].nodeType==1 && ch[i].nodeName!='P' && ch[i].nodeName!='DIV' && ch[i].nodeName!='BR')
-          this.responseDiv.removeChild(ch[i]);
-      }
-      responseText=Rico.stripTags(this.responseDiv.innerHTML);
-      success=(responseText.toLowerCase().indexOf('error')==-1);
-    }
-    if (success && this.options.showSaveMsg!='full') {
-      this.hideResponse('');
-      this.grid.resetContents();
-      this.grid.buffer.foundRowCount = false;
-      this.grid.buffer.fetch(this.grid.lastRowPos || 0);
-      if (this.saveMsg) this.saveMsg.innerHTML='&nbsp;'+responseText+'&nbsp;';
-    }
-    this.processCallback(this.options.onSubmitResponse);
-    Rico.log('Processing response completed');
-  },
-
-  processCallback: function(callback) {
-    switch (typeof callback) {
-      case 'string': return eval(callback);
-      case 'function': return callback();
-    }
-  },
-
-  // called when ok pressed on error response message
-  ackResponse: function(e) {
-    this.hideResponse('');
-    this.grid.highlightEnabled=true;
-  },
-
-  cloneRecord: function() {
-    this.formPopup.setTitle(this.cloneText);
-    this.displayEditForm("ins");
-  },
-
-  editRecord: function() {
-    this.formPopup.setTitle(this.editText);
-    this.displayEditForm("upd");
-  },
-
-  displayEditForm: function(action) {
-    this.grid.highlightEnabled=false;
-    this.menu.cancelmenu();
-    this.hideResponse(Rico.getPhraseById('saving'));
-    this.grid.outerDiv.style.cursor = 'auto';
-    this.action.value=action;
-    for (var i=0; i<this.grid.columns.length; i++) {
-      var c=this.grid.columns[i];
-      if (c.format) {
-        var v=c.getValue(this.rowIdx);
-        this.setField(i,v);
-        if (c.format.selectDesc) {
-          if (c.format.EntryType.length>1 && c.format.EntryType.charAt(1)=='L')
-            v=this.grid.columns[i+1].getValue(this.rowIdx);
-          v=c._format(v);
-          if (v==='') v='&nbsp;';
-          c.format.selectDesc.innerHTML=v;
-        }
-        if (c.format.SelectCtl)
-          Rico.EditControls.displayClrImg(c, !c.format.InsertOnly);
-      }
-    }
-    this.setReadOnly(action);
-    for (var k=0; k<this.keys.length; k++) {
-      this.keys[k].keyField.value = this.grid.buffer.getWindowValue(this.rowIdx,this.keys[k].colidx);
-    }
-    this.makeFormVisible(this.rowIdx);
-  },
-  
-  addPrepare: function() {
-    this.hideResponse(Rico.getPhraseById('saving'));
-    this.form.reset();
-    this.setReadOnly("ins");
-    this.action.value="ins";
-    for (var i=0; i<this.grid.columns.length; i++) {
-      var c=this.grid.columns[i];
-      if (c.format) {
-        this.setField(i,c.format.ColData);
-        if (c.format.SelectCtl) {
-          Rico.EditControls.resetValue(c);
-          Rico.EditControls.displayClrImg(c, !c.format.UpdateOnly);
-        }
-      }
-    }
-  },
-
-  addRecord: function() {
-    this.menu.cancelmenu();
-    this.formPopup.setTitle(this.addText);
-    this.addPrepare();
-    this.makeFormVisible(-1);
-    if (this.formPanels) this.formPanels.select(0);
-  },
-
-  drillDown: function(e,masterColNum,detailColNum) {
-    return this.grid.drillDown.apply(this.grid, arguments);
-  },
-
-  // set filter on a detail grid that is in a master-detail relationship
-  setDetailFilter: function(colNumber,filterValue) {
-    this.grid.setDetailFilter(colNumber,filterValue);
-  },
-
-  makeFormVisible: function(row) {
-    this.formPopup.container.style.display='block';
-
-    // set left position
-    var editWi=this.formPopup.container.offsetWidth;
-    var odOffset=Rico.cumulativeOffset(this.grid.outerDiv);
-    var winWi=Rico.windowWidth();
-    this.formPopup.container.style.left=editWi+odOffset.left > winWi ? (winWi-editWi)+'px' : (odOffset.left+1)+'px';
-
-    // set top position
-    var scrTop=Rico.docScrollTop();
-    var editHt=this.formPopup.container.offsetHeight;
-    var newTop=odOffset.top+this.grid.hdrHt+scrTop;
-    var bottom=Rico.windowHeight()+scrTop;
-    if (row >= 0) {
-      newTop+=(row+1)*this.grid.rowHeight;
-      if (newTop+editHt>bottom) newTop-=(editHt+this.grid.rowHeight);
-    } else {
-      if (newTop+editHt>bottom) newTop=bottom-editHt-2;
-    }
-
-    if (this.processCallback(this.options.formOpen) === false) return;
-    this.formPopup.openPopup(null,Math.max(newTop,scrTop));
-    this.formPopup.container.style.visibility='visible';
-    Rico.EditControls.setZ(Rico.getStyle(this.formPopup.container,'zIndex'));
-    if (this.initialized) return;
-
-    var i, spec;
-    for (i = 0; i < this.grid.columns.length; i++) {
-      spec=this.grid.columns[i].format;
-      if (!spec || !spec.EntryType || !spec.FieldName) continue;
-      switch (spec.EntryType) {
-        case 'tinyMCE':
-          if (typeof tinyMCE!='undefined') tinyMCE.execCommand('mceAddControl', true, spec.FieldName);
-          break;
-      }
-    }
-    this.initialized=true;
-  },
-
-  makeFormInvisible: function() {
-    this.formPopup.container.style.visibility='hidden';
-    this.formPopup.closePopup();
-    this.processCallback(this.options.formClose);
-  },
-
-  getConfirmDesc: function(rowIdx) {
-    return Rico.stripTags(this.grid.cell(rowIdx,this.options.ConfirmDeleteCol).innerHTML).replace('&nbsp;',' '); //.unescapeHTML();
-  },
-
-  deleteRecord: function() {
-    this.menu.cancelmenu();
-    var desc;
-    switch(this.options.ConfirmDeleteCol){
-      case -1 :
-        desc=Rico.getPhraseById("thisRecord",this.options.RecordName);
-        break;
-      case -2 : // Use key/column header to identify the row
-        for (var k=0; k<this.keys.length; k++) {
-          var i=this.keys[k].colidx;
-          var fmt=this.grid.columns[i].format;
-          if (fmt.EntryType.length>1 && fmt.EntryType.charAt(1)=='L') i++;
-          var value=Rico.stripTags(this.grid.cell(rowIdx,i).innerHTML).replace(/&nbsp;/g,' ');
-          if (desc) desc+=', ';
-          desc+=this.grid.columns[i].displayName+" "+value;
-        }
-        break;
-      default   :
-        desc='\"' + Rico.truncate(this.getConfirmDesc(this.rowIdx),50) + '\"';
-        break;
-    }
-    if (!this.options.ConfirmDelete.valueOf || confirm(Rico.getPhraseById("confirmDelete",desc))) {
-      this.hideResponse(Rico.getPhraseById('deleting'));
-      this.showResponse();
-      var parms={};
-      parms[this.grid.actionId]="del";
-      for (var k=0; k<this.keys.length; k++) {
-        var i=this.keys[k].colidx;
-        var value=this.grid.columns[i].getValue(this.rowIdx);
-        parms['_k'+i]=value;  // prototype does the encoding automatically
-        //parms['_k'+i]=encodeURIComponent(value);
-      }
-      new Rico.ajaxRequest(this.options.updateURL, {parameters:parms,method:'post',onComplete:this.responseHandler});
-    }
-    this.menu.cancelmenu();
-  },
-
-  validationMsg: function(elem,colnum,phraseId) {
-    var col=this.grid.columns[colnum];
-    if (this.formPanels) this.formPanels.select(this.panelActualIdx[col.format.panelIdx]);
-    var label=Rico.stripTags(col.formLabel.innerHTML).replace(/&nbsp;/g,' ');
-    var msg=Rico.getPhraseById(phraseId," \"" + label + "\"");
-    Rico.log(' Validation error: '+msg);
-    if (col.format.Help) msg+="\n\n"+col.format.Help;
-    alert(msg);
-    setTimeout(function() { try { elem.focus(); elem.select(); } catch(e) {}; }, 10);
-    return false;
-  },
-  
-  isTextInput: function(elem) {
-    if (!elem) return false;
-    if (elem.tagName.toLowerCase()!='input') return false;
-    if (elem.type.toLowerCase()!='text') return false;
-    if (elem.readOnly) return false;
-    if (!Rico.visible(elem)) return false;
-    return true;
-  },
-  
-  parseDate: function(v, dateFmt) {
-    dateParts={};
-    if (!this.dateRegExp.exec(dateFmt)) return NaN;
-    dateParts[RegExp.$1]=0;
-    dateParts[RegExp.$3]=1;
-    dateParts[RegExp.$5]=2;
-    var aDate = v.split(/\D/);
-    var d=new Date();
-    var curyr=d.getFullYear();
-    if (aDate.length==2 && dateParts.yyyy==2) aDate.push(curyr);
-    if (aDate.length!=3) return NaN;
-    var dd=parseInt(aDate[dateParts.dd], 10);
-    if (dd==0 || dd>31) return NaN;
-    var mm=parseInt(aDate[dateParts.mm], 10) - 1;
-    if (mm > 11) return NaN;
-    var yy=parseInt(aDate[dateParts.yyyy], 10);
-    if (yy < 100) {
-      // apply a century to 2-digit years
-      yy+=curyr - (curyr % 100);
-    }
-    d.setFullYear(yy,mm,dd);
-    return d;
-  },
-
-  TESubmit: function(e) {
-    var i,ro,lbl,spec,elem,n,dateValues=[];
-
-    Rico.eventStop(e);
-    Rico.log('Event: TESubmit called to validate input');
-
-    // check fields that are supposed to be non-blank
-
-    for (i = 0; i < this.grid.columns.length; i++) {
-      spec=this.grid.columns[i].format;
-      if (!spec || !spec.EntryType || !spec.FieldName) continue;
-      elem=Rico.$(spec.FieldName);
-      if (!this.isTextInput(elem)) continue;
-      switch (this.action.value) {
-        case 'ins': ro=!spec.Writeable || spec.ReadOnly || spec.UpdateOnly; break;
-        case 'upd': ro=!spec.Writeable || spec.ReadOnly || spec.InsertOnly; break;
-        default:    ro=false; break;
-      }
-      if (ro) continue;  // readonly, so don't validate
-      Rico.log(' Validating field #'+i+' EntryType='+spec.EntryType+' ('+spec.FieldName+')');
-
-      // check for blanks
-      if (elem.value.length == 0) {
-        if (spec.required)
-          return this.validationMsg(elem,i,"formPleaseEnter");
-        else
-          continue;
-      }
-
-      // check pattern
-      if (elem.value.length > 0 && spec.regexp && !spec.regexp.test(elem.value))
-        return this.validationMsg(elem,i,"formInvalidFmt");
-
-      // check min/max and date values
-      switch (spec.EntryType.charAt(0)) {
-        case 'I': n=parseInt(elem.value,10); break;
-        case 'F': n=parseFloat(elem.value); break;
-        case 'D': 
-          n=this.parseDate(elem.value,spec.dateFmt);
-          if (isNaN(n)) return this.validationMsg(elem,i,"formInvalidFmt");
-          dateValues.push({e:elem,v:n});
-          break;
-        default:  n=NaN; break;
-      }
-      if (typeof spec.min!='undefined' && !isNaN(n) && n < spec.min)
-        return this.validationMsg(elem,i,"formOutOfRange");
-      if (typeof spec.max!='undefined' && !isNaN(n) && n > spec.max)
-        return this.validationMsg(elem,i,"formOutOfRange");
-    }
-    if (this.processCallback(this.options.formSubmit) === false) return false;
-
-    // update drop-down for any columns with entry type of N
-
-    for (i = 0; i < this.grid.columns.length; i++) {
-      spec=this.grid.columns[i].format;
-      if (!spec || !spec.EntryType || !spec.FieldName) continue;
-      if (spec.EntryType.charAt(0) != 'N') continue;
-      var SelObj=Rico.$(spec.FieldName);
-      if (!SelObj || SelObj.value!=this.options.TableSelectNew) continue;
-      var newtext=Rico.$("textnew__" + SelObj.id).value;
-      this.addSelectOption(SelObj,newtext,newtext);
-    }
-    
-    // set date values to ISO format
-    for (i = 0; i < dateValues.length; i++) {
-      dateValues[i].e.value = Rico.formatDate(dateValues[i].v,'yyyy-mm-dd');
-    }
-
-    if (typeof tinyMCE!='undefined') tinyMCE.triggerSave();
-    this.makeFormInvisible();
-    this.sendForm();
-    this.menu.cancelmenu();
-    return false;
-  },
-  
-  sendForm: function() {
-    this.setReadOnly("reset");  // reset disabled flag so that all fields are sent to server
-    this.showResponse();
-    Rico.log("sendForm: "+this.grid.tableId);
-    Rico.ajaxSubmit(this.form, this.options.updateURL, {method:'post',onComplete:this.responseHandler});
-  }
-};
-
-
-/**
- * @namespace Registers custom popup widgets to fill in a text box (e.g. ricoCalendar and ricoTree)
- * <pre>
- * Custom widget must implement:
- *   open() method (make control visible)
- *   close() method (hide control)
- *   container property (div element that contains the control)
- *   id property (uniquely identifies the widget class)
- *
- * widget calls returnValue method to return a value to the caller
- *
- * this object handles clicks on the control's icon and positions the control appropriately.
- * </pre>
- */
-Rico.EditControls = {
-  widgetList : {},
-  elemList   : {},
-  zIndex     : 0,
-
-  register: function(widget, imgsrc) {
-    this.widgetList[widget.id] = {imgsrc:imgsrc, widget:widget, currentEl:''};
-    var self=this;
-    widget.returnValue=function(newVal,newDesc) { self.setValue(widget,newVal,newDesc); };
-    Rico.log("Rico.EditControls.register:"+widget.id);
-  },
-  
-  setZ: function(z) {
-    this.zIndex=Math.max(this.zIndex,z+10);
-  },
-
-  atLoad: function() {
-    for (var id in this.widgetList) {
-      var widget=this.widgetList[id].widget;
-      if (widget.atLoad && !widget.atLoadComplete) {
-        Rico.log("Rico.EditControls.atLoad: "+id);
-        widget.atLoad();
-        widget.atLoadComplete=true;
-      }
-    }
-  },
-
-  applyTo: function(column,inputCtl) {
-    var wInfo=this.widgetList[column.format.SelectCtl];
-    if (!wInfo) return;
-    Rico.log('Rico.EditControls.applyTo: '+column.displayName+' : '+column.format.SelectCtl);
-    var descSpan = document.createElement('span');
-    var newimg = document.createElement('img');
-    newimg.style.paddingLeft='4px';
-    newimg.style.cursor='pointer';
-    newimg.align='top';
-    newimg.src=wInfo.imgsrc;
-    newimg.id=this.imgId(column.format.FieldName);
-    Rico.eventBind(newimg,"click", Rico.eventHandle(this,'processClick'));
-    inputCtl.parentNode.appendChild(descSpan);
-    inputCtl.parentNode.appendChild(newimg);
-    inputCtl.style.display='none';    // comment out this line for debugging
-    var clr;
-    if (column.format.isNullable) {
-      clr=Rico.clearButton(Rico.eventHandle(this,'processClear'));
-      clr.id=newimg.id+'_clear';
-      inputCtl.parentNode.appendChild(clr);
-    }
-    this.elemList[newimg.id] = {descSpan:descSpan, inputCtl:inputCtl, widget:wInfo.widget, listObj:wInfo, column:column, clrimg:clr};
-    column.format.selectIcon=newimg;
-    column.format.selectDesc=descSpan;
-  },
-
-  displayClrImg: function(column,bShow) {
-    var el=this.elemList[this.imgId(column.format.FieldName)];
-    //alert(column.format.FieldName+': '+bShow+' '+typeof(el.clrimg));
-    if (el && el.clrimg) el.clrimg.style.display=bShow ? 'inline-block' : 'none';
-  },
-
-  processClear: function(e) {
-    var elem=Rico.eventElement(e);
-    var el=this.elemList[elem.id.slice(0,-6)];
-    if (!el) return;
-    el.inputCtl.value='';
-    el.descSpan.innerHTML=el.column._format('');
-  },
-
-  processClick: function(e) {
-    var elem=Rico.eventElement(e);
-    var el=this.elemList[elem.id];
-    if (!el) return;
-    if (el.listObj.currentEl==elem.id && el.widget.container.style.display!='none') {
-      el.widget.close();
-      el.listObj.currentEl='';
-    } else {
-      el.listObj.currentEl=elem.id;
-      Rico.log('Rico.EditControls.processClick: '+el.widget.id+' : '+el.inputCtl.value);
-      el.widget.container.style.zIndex=this.zIndex;
-      el.widget.open(el.inputCtl.value,el.column);     // this may change the size of the widget
-      Rico.positionCtlOverIcon(el.widget.container,elem);
-    }
-  },
-
-  imgId: function(fieldname) {
-    return 'icon_'+fieldname;
-  },
-
-  resetValue: function(column) {
-    var el=this.elemList[this.imgId(column.format.FieldName)];
-    if (!el) return;
-    el.inputCtl.value=column.format.ColData;
-    var v=column._format(column.format.ColData);
-    if (v==='') v='&nbsp;';
-    el.descSpan.innerHTML=v;
-  },
-
-  setValue: function(widget,newVal,newDesc) {
-    var wInfo=this.widgetList[widget.id];
-    if (!wInfo) return null;
-    var id=wInfo.currentEl;
-    if (!id) return null;
-    var el=this.elemList[id];
-    if (!el) return null;
-    el.inputCtl.value=newVal;
-    if (!newDesc) newDesc=el.column._format(newVal);
-    el.descSpan.innerHTML=newDesc;
-    if (el.column.format.DescriptionField)
-      Rico.$(el.column.format.DescriptionField).value = newDesc;
-    //alert(widget.id+':'+id+':'+el.inputCtl.id+':'+el.inputCtl.value+':'+newDesc);
-  },
-
-  close: function(id) {
-    var wInfo=this.widgetList[id];
-    if (!wInfo) return;
-    if (wInfo.widget.container.style.display!='none')
-      wInfo.widget.close();
-  }
-};
diff --git a/ricoClient/js/minsrc/ricoLiveGridMenu.js b/ricoClient/js/minsrc/ricoLiveGridMenu.js
deleted file mode 100644 (file)
index bb85181..0000000
+++ /dev/null
@@ -1,136 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
- if(typeof Rico=='undefined')
-  throw("GridMenu requires the Rico JavaScript framework");
-
-/**
- * Standard menu for LiveGrid
- */
-Rico.GridMenu = function(options) {
-  this.initialize(options);
-};
-
-Rico.GridMenu.prototype = {
-
-initialize: function(options) {
-  this.options = {
-    width           : '18em',
-    dataMenuHandler : null          // put custom items on the menu
-  };
-  Rico.extend(this.options, options || {});
-  Rico.extend(this, new Rico.Menu(this.options));
-  this.sortmenu = new Rico.Menu({ width: '15em' });
-  this.filtermenu = new Rico.Menu({ width: '22em' });
-  this.exportmenu = new Rico.Menu({ width: '24em' });
-  this.hideshowmenu = new Rico.Menu({ width: '22em' });
-  this.createDiv();
-  this.sortmenu.createDiv();
-  this.filtermenu.createDiv();
-  this.exportmenu.createDiv();
-  this.hideshowmenu.createDiv();
-},
-
-// Build context menu for grid
-buildGridMenu: function(r,c) {
-  this.clearMenu();
-  var livegrid=this.liveGrid;
-  var buffer=livegrid.buffer;
-  var totrows=buffer.totalRows;
-  var maxprint=livegrid.options.maxPrint;
-  var onBlankRow=(r >= totrows);
-  var column=livegrid.columns[c];
-  if (this.options.dataMenuHandler) {
-     var showDefaultMenu=this.options.dataMenuHandler(livegrid,r,c,onBlankRow);
-     if (!showDefaultMenu) return (this.itemCount > 0);
-  }
-
-  // menu items for sorting
-  if (column.sortable && totrows>0) {
-    this.sortmenu.clearMenu();
-    this.addSubMenuItem(Rico.getPhraseById("gridmenuSortBy",column.displayName), this.sortmenu, false);
-    this.sortmenu.addMenuItemId("gridmenuSortAsc", function() { column.sortAsc(); }, true);
-    this.sortmenu.addMenuItemId("gridmenuSortDesc", function() { column.sortDesc(); }, true);
-  }
-
-  // menu items for filtering
-  this.filtermenu.clearMenu();
-  if (column.canFilter() && !column.format.filterUI && (!onBlankRow || column.filterType == Rico.ColumnConst.USERFILTER)) {
-    this.addSubMenuItem(Rico.getPhraseById("gridmenuFilterBy",column.displayName), this.filtermenu, false);
-    column.userFilter=column.getValue(r);
-    if (column.filterType == Rico.ColumnConst.USERFILTER) {
-      this.filtermenu.addMenuItemId("gridmenuRemoveFilter", function() { column.setUnfiltered(false); }, true);
-      if (column.filterOp=='LIKE')
-        this.filtermenu.addMenuItemId("gridmenuChgKeyword", function() { livegrid.openKeyword(c); }, true);
-      if (column.filterOp=='NE' && !onBlankRow)
-        this.filtermenu.addMenuItemId("gridmenuExcludeAlso", function() { column.addFilterNE(); }, true);
-    } else if (!onBlankRow) {
-      this.filtermenu.addMenuItemId("gridmenuInclude", function() { column.setFilterEQ(); }, true);
-      this.filtermenu.addMenuItemId("gridmenuGreaterThan", function() { column.setFilterGE(); }, column.userFilter!='');
-      this.filtermenu.addMenuItemId("gridmenuLessThan", function() { column.setFilterLE(); }, column.userFilter!='');
-      if (column.isText)
-        this.filtermenu.addMenuItemId("gridmenuContains", function() { livegrid.openKeyword(c); }, true);
-      this.filtermenu.addMenuItemId("gridmenuExclude", function() { column.setFilterNE(); }, true);
-    }
-    if (livegrid.filterCount() > 0) {
-      this.filtermenu.addMenuItemId("gridmenuRefresh", function() { livegrid.filterHandler(); }, true);
-      this.filtermenu.addMenuItemId("gridmenuRemoveAll", function() { livegrid.clearFilters(); }, true);
-    }
-  } else if (livegrid.filterCount() > 0) {
-    this.addSubMenuItem(Rico.getPhraseById("gridmenuFilterBy",column.displayName), this.filtermenu, false);
-    this.filtermenu.addMenuItemId("gridmenuRemoveAll", function() { livegrid.clearFilters(); }, true);
-  }
-
-  // menu items for Print/Export
-  this.exportmenu.clearMenu();
-  if (buffer.printVisibleSQL && typeof(buffer.dataSource)=='string') {
-    // SQL buffer
-    this.addSubMenuItem(Rico.getPhraseById('gridmenuExport'),this.exportmenu,false);
-    this.exportmenu.addMenuItemId("gridmenuExportVis2Web", function() { buffer.printVisibleSQL('html'); });
-    this.exportmenu.addMenuItemId("gridmenuExportAll2Web", function() { buffer.printAllSQL('html'); }, buffer.totalRows <= maxprint);
-    this.exportmenu.addMenuBreak();
-    this.exportmenu.addMenuItemId("gridmenuExportVis2SS", function() { buffer.printVisibleSQL('xl'); });
-    this.exportmenu.addMenuItemId("gridmenuExportAll2SS", function() { buffer.printAllSQL('xl'); }, buffer.totalRows <= maxprint);
-  } else if (maxprint > 0 && totrows>0) {
-    // any other buffer
-    this.addSubMenuItem(Rico.getPhraseById('gridmenuExport'),this.exportmenu,false);
-    this.exportmenu.addMenuItemId("gridmenuExportVis2Web", function() { buffer.printVisible('plain'); });
-    this.exportmenu.addMenuItemId("gridmenuExportAll2Web", function() { buffer.printAll('plain'); }, buffer.totalRows <= maxprint);
-    if (Rico.isIE) {
-      this.exportmenu.addMenuBreak();
-      this.exportmenu.addMenuItemId("gridmenuExportVis2SS", function() { buffer.printVisible('owc'); });
-      this.exportmenu.addMenuItemId("gridmenuExportAll2SS", function() { buffer.printAll('owc'); }, buffer.totalRows <= maxprint);
-    }
-  }
-
-  // menu items for hide/unhide
-  var hiddenCols=livegrid.listInvisible();
-  for (var showableCnt=0,x=0; x<hiddenCols.length; x++) {
-    if (hiddenCols[x].canHideShow()) showableCnt++;
-  }
-  if (showableCnt > 0 || column.canHideShow()) {
-    this.hideshowmenu.clearMenu();
-    this.addSubMenuItem(Rico.getPhraseById('gridmenuHideShow'),this.hideshowmenu,false);
-    this.hideshowmenu.addMenuItemId('gridmenuChooseCols', function() { livegrid.chooseColumns(); },true,false);
-    var visibleCnt=livegrid.columns.length-hiddenCols.length;
-    var enabled=(visibleCnt>1 && column.visible && column.canHideShow());
-    this.hideshowmenu.addMenuItem(Rico.getPhraseById('gridmenuHide',column.displayName), function() { column.hideColumn(); }, enabled);
-    if (hiddenCols.length > 1)
-      this.hideshowmenu.addMenuItemId('gridmenuShowAll', function() { livegrid.showAll(); });
-  }
-  return true;
-}
-
-}
diff --git a/ricoClient/js/minsrc/ricoSearch.js b/ricoClient/js/minsrc/ricoSearch.js
deleted file mode 100644 (file)
index 0f9e440..0000000
+++ /dev/null
@@ -1,134 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-Rico.KeywordSearch = function(id,options) {
-  this.initialize(id,options);
-};
-
-Rico.KeywordSearch.prototype = {
-/**
- * @class Implements a pop-up keyword search control.
- * @extends Rico.Popup
- * @constructs
- * @param id unique identifier
- * @param options object may contain any of the following:<dl>
- *   <dt>showColorCode</dt><dd> show hex color code as user hovers over color grid? default=false</dd>
- *   <dt>cellsPerRow  </dt><dd> number of colors per row in the grid? default=18</dd>
- *   <dt>palette      </dt><dd> array of 6 digit hex values, default=216 "web safe" colors</dd>
- *</dl>
- */
-  initialize: function(id,options) {
-    this.id=id;
-    Rico.extend(this, new Rico.Window(Rico.getPhraseById("keywordTitle"),options));
-    Rico.addClass(this.content,'ricoKeywordSearch');
-    Rico.extend(this.options, {
-      listLength : 10,
-      maxSuggest : 20,
-      width: '12em'
-    });
-  },
-
-  atLoad : function() {
-    this.searchField=Rico.createFormField(this.contentDiv,'input','text',this.id+'_search');
-    this.searchField.style.display="block";
-    this.searchField.style.width=this.options.width;
-    Rico.eventBind(this.searchField,'keyup',Rico.eventHandle(this,'filterKeypress'),false);
-    this.selectList=Rico.createFormField(this.contentDiv,'select',null,this.id+'_list');
-    this.selectList.size=this.options.listLength;
-    this.selectList.style.display="block";
-    this.selectList.style.width=this.options.width;
-    Rico.eventBind(this.selectList,'change',Rico.eventHandle(this,'listClick'),false);
-    /**
-     * alias for closePopup
-     * @function
-     */
-    this.close=this.closePopup;
-    this.close();
-  },
-  
-  open: function(currentVal,column) {
-    this.column=column;
-    this.grid=this.column.liveGrid;
-    this.searchField.value='';
-    this.selectList.options.length=0;
-    this.openPopup();
-    this.searchField.focus();
-    this.selectValuesRequest('');
-  },
-
-  selectValuesRequest: function(filter) {
-    var colnum=this.column.index;
-    var options={};
-    Rico.extend(options, this.grid.buffer.ajaxOptions);
-    options.parameters = {id: this.grid.tableId, offset: '0', page_size: this.options.maxSuggest, edit: colnum};
-    options.parameters[this.grid.actionId]="query";
-    if (filter!='' && filter!='*') {
-      if (filter.indexOf('*')==-1) filter='*'+filter+'*';
-      options.parameters['f[1][op]']="LIKE";
-      options.parameters['f[1][len]']=1;
-      options.parameters['f[1][0]']=filter;
-    }
-    var self=this;
-    options.onComplete = function(request) { self.selectValuesUpdate(request); };
-    new Rico.ajaxRequest(this.grid.buffer.dataSource, options);
-  },
-
-  selectValuesUpdate: function(request) {
-    var response = request.responseXML.getElementsByTagName("ajax-response");
-    Rico.log("selectValuesUpdate: "+request.status);
-    if (response == null || response.length != 1) return;
-    response=response[0];
-    var error = response.getElementsByTagName('error');
-    if (error.length > 0) {
-      var errmsg=Rico.getContentAsString(error[0],this.grid.buffer.isEncoded);
-      Rico.log("Data provider returned an error:\n"+errmsg);
-      alert(Rico.getPhraseById("requestError",errmsg));
-      return null;
-    }
-    this.selectList.options.length=0;
-    response=response.getElementsByTagName('response')[0];
-    var rowsElement = response.getElementsByTagName('rows')[0];
-    var rows = this.grid.buffer.dom2jstable(rowsElement);
-    Rico.log("selectValuesUpdate: id="+this.selectList.id+' rows='+rows.length);
-    for (var i=0; i<rows.length; i++) {
-      if (rows[i].length>0) {
-        var c0=rows[i][0];
-        var c1=(rows[i].length>1) ? rows[i][1] : c0;
-        Rico.addSelectOption(this.selectList,c0,c1);
-      }
-    }
-  },
-
-  filterKeypress: function(e) {
-    var txtbox=Rico.eventElement(e);
-    if (typeof this.lastKeyFilter != 'string') this.lastKeyFilter='';
-    if (this.lastKeyFilter==txtbox.value) return;
-    var v=txtbox.value;
-    Rico.log("filterKeypress: "+this.index+' '+v);
-    this.lastKeyFilter=v;
-    this.selectValuesRequest(v);
-  },
-  
-  listClick: function(e) {
-    var elem=Rico.eventElement(e);
-    if (elem.tagName.toLowerCase() != 'select') return;
-    if (this.returnValue) {
-      var opt=elem.options[elem.selectedIndex];
-      this.returnValue(opt.value,opt.innerHTML);
-    }
-    this.close();
-  }
-
-};
diff --git a/ricoClient/js/minsrc/ricoSimpleGrid.js b/ricoClient/js/minsrc/ricoSimpleGrid.js
deleted file mode 100644 (file)
index 170c942..0000000
+++ /dev/null
@@ -1,401 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-if(typeof Rico=='undefined') throw("SimpleGrid requires the Rico JavaScript framework");
-
-Rico.SimpleGrid = function(tableId, options) {
-  this.initialize(tableId, options);
-}
-
-Rico.SimpleGrid.prototype = {
-/**
- * @class Create & manage an unbuffered grid.
- * Supports: frozen columns & headings, resizable columns.
- * @extends Rico.GridCommon
- * @constructs
- */
-  initialize: function( tableId, options ) {
-    Rico.extend(this, Rico.GridCommon);
-    this.baseInit();
-    Rico.setDebugArea(tableId+"_debugmsgs");    // if used, this should be a textarea
-    Rico.extend(this.options, options || {});
-    this.tableId = tableId;
-    Rico.log("SimpleGrid initialize start: "+tableId);\r
-    this.createDivs();
-    this.hdrTabs=new Array(2);
-    this.simpleGridInit();
-    Rico.log("SimpleGrid initialize end: "+tableId);\r
-  },
-
-  simpleGridInit: function() {
-    var i;
-    for (i=0; i<2; i++) {
-      Rico.log("simpleGridInit "+i);\r
-      this.tabs[i]=document.getElementById(this.tableId+'_tab'+i);
-      if (!this.tabs[i]) return;
-      this.hdrTabs[i]=document.getElementById(this.tableId+'_tab'+i+'h');
-      if (!this.hdrTabs[i]) return;
-      //if (i==0) this.tabs[i].style.position='absolute';
-      //if (i==0) this.tabs[i].style.left='0px';
-      //this.hdrTabs[i].style.position='absolute';
-      //this.hdrTabs[i].style.top='0px';
-      //this.hdrTabs[i].style.zIndex=1;
-      this.thead[i]=this.hdrTabs[i];
-      this.tbody[i]=this.tabs[i];
-      this.headerColCnt = this.getColumnInfo(this.hdrTabs[i].rows);
-      if (i==0) this.options.frozenColumns=this.headerColCnt;
-      if (Rico.theme.gridheader) Rico.addClass(this.thead[i],Rico.theme.gridheader);
-      if (Rico.theme.gridcontent) Rico.addClass(this.tbody[i],Rico.theme.gridcontent);
-    }
-    if (this.headerColCnt==0) {
-      alert('ERROR: no columns found in "'+this.tableId+'"');
-      return;
-    }
-    //this.hdrHt=Math.max(Rico.nan2zero(this.hdrTabs[0].offsetHeight),this.hdrTabs[1].offsetHeight);
-    //for (i=0; i<2; i++) {
-    //  if (i==0) this.tabs[i].style.top=this.hdrHt+'px';
-    //}
-    this.createColumnArray('SimpleGridColumn');
-    this.pageSize=this.columns[0].dataColDiv.childNodes.length;
-    this.sizeDivs();
-    if (typeof(this.options.FilterLocation)=='number')
-      this.createFilters(this.options.FilterLocation);
-    this.attachMenuEvents();
-    this.scrollEventFunc=Rico.eventHandle(this,'handleScroll');
-    this.pluginScroll();
-    if (this.options.windowResize)
-      Rico.eventBind(window,"resize", Rico.eventHandle(this,'sizeDivs'), false);
-  },
-
-  // return id string for a filter element
-  filterId: function(colnum) {
-    return 'RicoFilter_'+this.tableId+'_'+colnum;
-  },
-  
-  // create filter elements on heading row r
-  createFilters: function(r) {
-    if (r < 0) {
-      r=this.addHeadingRow();
-      this.sizeDivs();
-    }
-    for( var c=0; c < this.headerColCnt; c++ ) {
-      var col=this.columns[c];
-      var fmt=col.format;
-      if (typeof fmt.filterUI!='string') continue;
-      var cell=this.hdrCells[r][c].cell;
-      var field,name=this.filterId(c);\r
-      var divs=cell.getElementsByTagName('div');
-      switch (fmt.filterUI.charAt(0)) {
-        case 't':
-          field=Rico.createFormField(divs[1],'input','text',name,name);
-          var size=fmt.filterUI.match(/\d+/);
-          field.maxLength=fmt.Length || 50;\r
-          field.size=size ? parseInt(size,10) : 10;
-          Rico.eventBind(field,'keyup',Rico.eventHandle(col,'filterKeypress'),false);\r
-          break;\r
-        case 's':
-          field=Rico.createFormField(divs[1],'select',null,name);\r
-          Rico.addSelectOption(field,this.options.FilterAllToken,Rico.getPhraseById("filterAll"));
-          this.getFilterValues(col);
-          var keys=Rico.keys(col.filterHash);
-          keys.sort();
-          for (var i=0; i<keys.length; i++)
-            Rico.addSelectOption(field,keys[i],keys[i] || Rico.getPhraseById("filterBlank"));\r
-          Rico.eventBind(field,'change',Rico.eventHandle(col,'filterChange'),false);\r
-          break;\r
-      }
-    }
-    this.initFilterImage(r);
-  },
-  
-  getFilterValues: function(col) {
-    var h={};
-    var n=col.numRows();
-    for (var i=0; i<n; i++) {
-      var v=Rico.getInnerText(col.cell(i));
-      var hval=h[v];
-      if (hval)
-        hval.push(i);
-      else
-        h[v]=[i];
-    }
-    col.filterHash=h;\r
-  },
-  
-  // hide filtered rows
-  applyFilters: function() {
-    // rows to display will be the intersection of all filterRows arrays
-    var fcols=[];
-    for (var c=0; c<this.columns.length; c++) {
-      if (this.columns[c].filterRows)
-        fcols.push(this.columns[c].filterRows);
-    }
-    if (fcols.length==0) {
-      // no filters are set
-      this.showAllRows();
-      return;
-    }
-    for (var r=0; r<this.pageSize; r++) {
-      var showflag=true;
-      for (var j=0; j<fcols.length; j++) {
-        if (fcols[j].indexOf(r)==-1) {
-          showflag=false;
-          break;
-        }
-      }
-      if (showflag)
-        this.showRow(r);
-      else
-        this.hideRow(r);
-    }
-    this.sizeDivs();\r
-  },
-
-  handleScroll: function(e) {
-    var newTop=(-this.scrollDiv.scrollTop)+'px';
-    this.tabs[0].style.marginTop=newTop;
-    this.setHorizontalScroll();
-  },
-
-  /**
-   * Register a menu that will only be used in the scrolling part of the grid.
-   * If submenus are used, they must be registered after the main menu.
-   */
-  registerScrollMenu: function(menu) {
-    if (!this.menu) this.menu=menu;
-    menu.grid=this;
-    menu.showmenu=menu.showSimpleMenu;
-    menu.showSubMenu=menu.showSimpleSubMenu;
-    menu.createDiv(this.outerDiv);
-  },
-
-  handleMenuClick: function(e) {
-    if (!this.menu) return;
-    this.cancelMenu();
-    this.menuCell=Rico.getParentByTagName(Rico.eventElement(e),'div');
-    this.highlightEnabled=false;
-    if (this.hideScroll) this.scrollDiv.style.overflow="hidden";
-    if (this.menu.buildGridMenu) this.menu.buildGridMenu(this.menuCell);
-    this.menu.showmenu(e,this.closeMenu.bind(this));
-  },
-
-  closeMenu: function() {
-    if (this.hideScroll) this.scrollDiv.style.overflow="";
-    this.highlightEnabled=true;
-  },
-
-  sizeDivs: function() {
-    if (this.outerDiv.offsetParent.style.display=='none') return;
-    this.baseSizeDivs();
-    var maxHt=Math.max(this.options.maxHt || this.availHt(), 50);
-    var totHt=Math.min(this.hdrHt+this.dataHt, maxHt);
-    Rico.log('sizeDivs '+this.tableId+': hdrHt='+this.hdrHt+' dataHt='+this.dataHt);
-    this.dataHt=totHt-this.hdrHt;
-    if (this.scrWi>0) this.dataHt+=this.options.scrollBarWidth;
-    this.scrollDiv.style.height=this.dataHt+'px';
-    this.frozenTabs.style.height=this.scrollDiv.clientHeight+'px';
-    var divAdjust=2;
-    this.innerDiv.style.width=(this.scrWi-this.options.scrollBarWidth+divAdjust)+'px';
-    //this.innerDiv.style.height=(this.hdrHt+1)+'px';
-    totHt+=divAdjust;
-    this.resizeDiv.style.height=totHt+'px';
-    
-    //this.outerDiv.style.height=(totHt+this.options.scrollBarWidth)+'px';
-    this.handleScroll();
-  },
-  
-/**
- * Copies all rows to a new window as a simple html table.
- */
-  printVisible: function(exportType) {
-    this.showMsg(Rico.getPhraseById('exportInProgress'));
-    Rico.runLater(10,this,'_printVisible',exportType);  // allow message to paint
-  },
-
-  _printVisible: function(exportType) {
-    this.exportStart();
-    var exportStyles=this.getExportStyles(this.tbody[0]);
-    for(var r=0; r < this.pageSize; r++) {
-      if (this.columns[0].cell(r).style.display=='none') continue;
-      var exportText='';
-      for (var c=0; c<this.columns.length; c++) {
-        var col=this.columns[c];
-        if (col.visible) {
-          var v=col.getFormattedValue(r, !this.options.exportImgTags, !this.options.exportFormFields, 'NoExport');
-          if (col.format.exportPrefix) v=col.format.exportPrefix+v;
-          if (v=='') v='&nbsp;';
-          exportText+="<td style='"+this.exportStyle(col.cell(r),exportStyles)+"'>"+v+"</td>";
-        }
-      }
-      this.exportRows.push(exportText);
-    }
-    this.exportFinish(exportType);
-  },
-
-  /**
-   * Hide a row in the grid.
-   * sizeDivs() should be called after this function has completed.
-   */
-  hideRow: function(rownum) {
-    if (this.columns[0].cell(rownum).style.display=='none') return;
-    for (var i=0; i<this.columns.length; i++)
-      this.columns[i].cell(rownum).style.display='none';
-  },
-
-  /**
-   * Unhide a row in the grid.
-   * sizeDivs() should be called after this function has completed.
-   */
-  showRow: function(rownum) {
-    if (this.columns[0].cell(rownum).style.display=='') return;
-    for (var i=0; i<this.columns.length; i++)
-      this.columns[i].cell(rownum).style.display='';
-  },
-
-  /**
-   * Search for rows that contain SearchString in column ColIdx.
-   * If ShowMatch is false, then matching rows are hidden, if true then mismatching rows are hidden.
-   */
-  searchRows: function(ColIdx,SearchString,ShowMatch) {\r
-    if (!SearchString) return;\r
-    var re=new RegExp(SearchString);\r
-    var rowcnt=this.columns[ColIdx].numRows();\r
-    for(var r=0; r<rowcnt; r++) {\r
-      var txt=this.cell(r,ColIdx).innerHTML;\r
-      var matched=(txt.match(re) != null);\r
-      if (matched != ShowMatch) this.hideRow(r);\r
-    }
-    this.sizeDivs();
-    this.handleScroll();\r
-  },\r
-\r
-  /**
-   * Unhide all rows in the grid
-   */
-  showAllRows: function() {
-    for (var i=0; i<this.pageSize; i++)
-      this.showRow(i);
-    this.sizeDivs();\r
-  },
-  
-  openPopup: function(elem,popupobj) {
-    while (elem && !Rico.hasClass(elem,'ricoLG_cell'))
-      elem=elem.parentNode;
-    if (!elem) return false;
-    var td=Rico.getParentByTagName(elem,'td');
-  
-    var newLeft=Math.floor(td.offsetLeft-this.scrollDiv.scrollLeft+td.offsetWidth/2);
-    if (this.direction == 'rtl') {
-      if (newLeft > this.width) newLeft-=this.width;
-    } else {
-      if (newLeft+this.width+this.options.margin > this.scrollDiv.clientWidth) newLeft-=this.width;
-    }
-    popupobj.divPopup.style.visibility="hidden";
-    popupobj.divPopup.style.display="block";
-    var contentHt=popupobj.divPopup.offsetHeight;
-    var newTop=Math.floor(elem.offsetTop-this.scrollDiv.scrollTop+elem.offsetHeight/2);
-    if (newTop+contentHt+popupobj.options.margin > this.scrollDiv.clientHeight)
-      newTop=Math.max(newTop-contentHt,0);
-    popupobj.openPopup(this.frzWi+newLeft,this.hdrHt+newTop);
-    popupobj.divPopup.style.visibility ="visible";
-    return elem;
-  }
-
-}
-
-if (Rico.Menu) {
-Rico.extend(Rico.Menu.prototype, {
-
-showSimpleMenu: function(e,hideFunc) {
-  Rico.eventStop(e);
-  this.hideFunc=hideFunc;
-  if (this.div.childNodes.length==0) {
-    this.cancelmenu();
-    return false;
-  }
-  var elem=Rico.eventElement(e);
-  this.grid.openPopup(elem,this);
-  return elem;
-},
-
-showSimpleSubMenu: function(a,submenu) {
-  if (this.openSubMenu) this.hideSubMenu();
-  this.openSubMenu=submenu;
-  this.openMenuAnchor=a;
-  if (a.className=='ricoSubMenu') a.className='ricoSubMenuOpen';
-  var top=parseInt(this.div.style.top,10);
-  var left=parseInt(this.div.style.left,10);
-  submenu.openPopup(left+a.offsetWidth,top+a.offsetTop);
-  submenu.div.style.visibility ="visible";
-}
-
-});
-}
-
-
-Rico.SimpleGridColumn = function(grid,colIdx,hdrInfo,tabIdx) {
-  this.initialize(grid,colIdx,hdrInfo,tabIdx);
-}
-
-Rico.SimpleGridColumn.prototype = {
-/**
- * @class Implements a SimpleGrid column
- * @extends Rico.TableColumnBase
- * @constructs
- */
-initialize: function(grid,colIdx,hdrInfo,tabIdx) {
-  Rico.extend(this, new Rico.TableColumnBase());
-  this.baseInit(grid,colIdx,hdrInfo,tabIdx);
-},
-
-setUnfiltered: function() {
-  this.filterRows=null;
-},
-
-filterChange: function(e) {\r
-  var selbox=Rico.eventElement(e);
-  if (selbox.value==this.liveGrid.options.FilterAllToken)\r
-    this.setUnfiltered();\r
-  else
-    this.filterRows=this.filterHash[selbox.value];
-  this.liveGrid.applyFilters();
-},
-
-filterKeypress: function(e) {\r
-  var txtbox=Rico.eventElement(e);
-  if (typeof this.lastKeyFilter != 'string') this.lastKeyFilter='';\r
-  if (this.lastKeyFilter==txtbox.value) return;\r
-  var v=txtbox.value;\r
-  Rico.log("filterKeypress: "+this.index+' '+v);\r
-  this.lastKeyFilter=v;
-  if (v) {
-    v=v.replace('\\','\\\\');
-    v=v.replace('(','\\(').replace(')','\\)');
-    v=v.replace('.','\\.');
-    if (this.format.filterUI.indexOf('^') > 0) v='^'+v;
-    var re=new RegExp(v,'i');\r
-    this.filterRows=[];
-    var n=this.numRows();
-    for (var i=0; i<n; i++) {
-      var celltxt=Rico.getInnerText(this.cell(i));
-      if (celltxt.match(re)) this.filterRows.push(i);
-    }
-  } else {
-    this.setUnfiltered();\r
-  }
-  this.liveGrid.applyFilters();
-}
-
-}
diff --git a/ricoClient/js/minsrc/ricoTree.js b/ricoClient/js/minsrc/ricoTree.js
deleted file mode 100644 (file)
index 08fe1ca..0000000
+++ /dev/null
@@ -1,325 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-//  Rico Tree Control
-//  Requires prototype.js and ricoCommon.js
-//  Data for the tree can be obtained via AJAX requests
-//  Each record in the AJAX response should contain 5 or 6 cells:
-//   cells[0]=parent node id
-//   cells[1]=node id
-//   cells[2]=description
-//   cells[3]=L/zero (leaf), C/non-zero (container)
-//   cells[4]=0->not selectable, 1->selectable (use default action), otherwise the node is selectable and cells[4] contains the action
-//   cells[5]=leafIcon (optional)
-
-
-Rico.TreeControl = function(id,url,options) {
-  this.initialize(id,url,options);
-};
-
-Rico.TreeControl.prototype = {
-/**
- * @class Implements a pop-up tree control.
- * @extends Rico.Popup
- * @constructs
- * @param id unique identifier
- * @param url data source
- * @param options object may contain any of the following:<dl>
- *   <dt>nodeIdDisplay</dt><dd> first, last, tooltip, or none? default=none</dd>
- *   <dt>showCheckBox </dt><dd> show checkbox next to each item? default=false</dd>
- *   <dt>showFolders  </dt><dd> show folder icons? default=false</dd>
- *   <dt>showPlusMinus</dt><dd> show +/- icons to open/close branches? default=true</dd>
- *   <dt>showLines    </dt><dd> show vertical lines connecting each level? default=true</dd>
- *   <dt>defaultAction</dt><dd> Rico event handle to call when user clicks on an item, default is to call returnValue method</dd>
- *   <dt>height       </dt><dd> control height? default=300px</dd>
- *   <dt>width        </dt><dd> control width? default=300px</dd>
- *   <dt>leafIcon     </dt><dd> url to img, default=doc.gif ('none'=no leaf icon)</dd>
- *</dl>
- */
-  initialize: function(id,url,options) {
-    Rico.extend(this, new Rico.Popup());
-    Rico.extend(this.options, {
-      ignoreClicks:true,
-      nodeIdDisplay:'none',
-      showCheckBox: false,
-      showFolders: false,
-      showPlusMinus: true,
-      showLines: true,
-      defaultAction: Rico.eventHandle(this,'nodeClick'),
-      height: '300px',
-      width: '300px',
-      leafIcon: Rico.imgDir+'doc.gif'
-    });
-    Rico.extend(this.options, options || {});
-    this.id=id;
-    this.dataSource=url;
-    this.close=this.closePopup;
-    this.hoverSet = new Rico.HoverSet([]);
-  },
-
-  atLoad : function() {
-    var imgsrc = ["node.gif","nodelast.gif","folderopen.gif","folderclosed.gif"];
-    // preload images
-    for (var i=0;i<imgsrc.length;i++) {
-      var img=new Image();
-      img.src = Rico.imgDir+imgsrc[i];
-    }
-    this.treeDiv=document.createElement("div");
-    this.treeDiv.id=this.id;
-    this.treeDiv.className='ricoTree';
-    if (Rico.theme.treeContent) Rico.addClass(this.treeDiv,Rico.theme.treeContent);
-    this.treeDiv.style.height=this.options.height;
-    this.treeDiv.style.width=this.options.width;
-    this.createContainer();
-    this.content.className=Rico.theme.tree || 'ricoTreeContainer';
-    this.content.appendChild(this.treeDiv);
-    if (this.options.showCheckBox) {
-      this.buttonDiv=document.createElement("div");
-      this.buttonDiv.style.width=this.options.width;
-      this.buttonDiv.className='ricoTreeButtons';
-      if (Rico.getStyle(this.container,'position')=='absolute') {
-        var span=document.createElement("span");
-        span.innerHTML=RicoTranslate.getPhraseById('treeSave');
-        Rico.setStyle(span,{'float':'left',cursor:'pointer'});
-        this.buttonDiv.appendChild(span);
-        Rico.eventBind(span, 'click', Rico.eventHandle(this,'saveSelection'));
-      }
-      var span=document.createElement("span");
-      span.innerHTML=RicoTranslate.getPhraseById('treeClear');
-      Rico.setStyle(span,{'float':'right',cursor:'pointer'});
-      this.buttonDiv.appendChild(span);
-      this.content.appendChild(this.buttonDiv);
-      Rico.eventBind(span, 'click', Rico.eventHandle(this,'clrCheckBoxEvent'));
-    }
-    this.close();
-  },
-
-  setTreeDiv: function(divId) {
-    this.treeDiv = Rico.$(divId);
-    this.openPopup = function() {};
-  },
-
-  open: function() {
-    this.openPopup();
-    if (this.treeDiv.childNodes.length == 0 && this.dataSource) {
-      this.loadXMLDoc();
-    }
-  },
-
-  loadXMLDoc: function(branchPin) {
-    var parms = { id: this.id };
-    if (branchPin) parms.Parent=branchPin;
-    Rico.log('Tree loadXMLDoc: '+this.id);
-    var self=this;
-    new Rico.ajaxRequest(this.dataSource, {parameters:parms, method:'get', onComplete: function(request) { self.processResponse(request); } });
-  },
-
-  domID: function(nodeID,part) {
-    return 'RicoTree_'+part+'_'+this.id+'_'+nodeID;
-  },
-
-  processResponse: function(request) {
-    var response = request.responseXML.getElementsByTagName("ajax-response");
-    if (response == null || response.length != 1) return;
-    var rowsElement = response[0].getElementsByTagName('rows')[0];
-    var trs = rowsElement.getElementsByTagName("tr");
-    var rowdata=[];
-    for (var i=0; i < trs.length; i++) {
-      var cells = trs[i].getElementsByTagName("td");
-      if (cells.length < 5) continue;
-      var content=[];
-      content[5]=this.options.leafIcon;
-      for (var j=0; j<cells.length; j++) {
-        content[j]=Rico.getContentAsString(cells[j],true);
-      }
-      content[3] = content[3].match(/^0|L$/i) ? 0 : 1;
-      content[4] = parseInt(content[4]);
-      rowdata.push(content);
-    }
-    for (var i=0; i < rowdata.length; i++) {
-      var moreChildren=(i < rowdata.length-1) && (rowdata[i][0]==rowdata[i+1][0]);
-      this.addNode(rowdata[i][0],rowdata[i][1],rowdata[i][2],rowdata[i][3],rowdata[i][4],rowdata[i][5],!moreChildren);
-    }
-  },
-
-  DisplayImages: function(row,arNames) {
-    var i,img,td;
-    for(i=0;i<arNames.length;i++) {
-      img = document.createElement("img");
-      img.src=Rico.imgDir+arNames[i] + ".gif";
-      td=row.insertCell(-1);
-      td.appendChild(img);
-    }
-  },
-
-  addNode: function(parentId, nodeId, nodeDesc, isContainer, isSelectable, leafIcon, isLast) {
-    var parentNode=Rico.$(this.domID(parentId,'Parent'));
-    var parentChildren=Rico.$(this.domID(parentId,'Children'));
-    var level=parentNode ? parentNode.TreeLevel+1 : 0;
-    //alert("addNode at level " + level + " (" + nodeId + ")")
-    var tab = document.createElement("table");
-    var div = document.createElement("div");
-    div.id=this.domID(nodeId,'Children');
-    div.className='ricoTreeBranch';
-    div.style.display=parentNode ? 'none' : '';
-    tab.border=0;
-    tab.cellSpacing=0;
-    tab.cellPadding=0;
-    tab.id=this.domID(nodeId,'Parent');
-    tab.TreeLevel=level;
-    tab.TreeContainer=isContainer;
-    tab.TreeFetchedChildren=this.dataSource ? false : true;
-    var row=tab.insertRow(0);
-    var td=[];
-    for (var i=0; i<level-1; i++) {
-      td[i]=row.insertCell(-1);
-    }
-    if (level>1) {
-      var tdParent=parentNode.getElementsByTagName('td');
-      for (var i=0; i<level-2; i++) {
-        td[i].innerHTML=tdParent[i].innerHTML;
-      }
-      var img = document.createElement("img");
-      img.src=Rico.imgDir+(parentChildren.nextSibling && this.options.showLines ? "nodeline" : "nodeblank")+".gif";
-      td[level-2].appendChild(img);
-    }
-    if (level>0) {
-      var suffix=isLast && this.options.showLines ? 'last' : '';
-      var prefix=this.options.showLines ? 'node' : '';
-      if (this.options.showPlusMinus && isContainer) {
-        var img = document.createElement("img");
-        img.name=nodeId;
-        img.style.cursor='pointer';
-        Rico.eventBind(img, 'click', Rico.eventHandle(this,'clickBranch'));
-        img.src=Rico.imgDir+prefix+"p"+suffix+".gif";
-        row.insertCell(-1).appendChild(img);
-      } else if (this.options.showLines) {
-        var img = document.createElement("img");
-        img.src=Rico.imgDir+"node"+suffix+".gif";
-        row.insertCell(-1).appendChild(img);
-      }
-      if (this.options.showFolders && (isContainer || (leafIcon && leafIcon!='none'))) {
-        var img = document.createElement("img");
-        if (!isContainer) {
-          img.src=leafIcon;
-        } else {
-          img.name=nodeId;
-          img.style.cursor='pointer';
-          Rico.eventBind(img, 'click', Rico.eventHandle(this,'clickBranch'));
-          img.src=Rico.imgDir+"folderclosed.gif";
-        }
-        row.insertCell(-1).appendChild(img);
-      }
-    }
-    if (isSelectable && this.options.showCheckBox) {
-      var chkbx=document.createElement("input");
-      chkbx.type="checkbox";
-      chkbx.value=nodeId;
-      row.insertCell(-1).appendChild(chkbx);
-    }
-
-    if (isSelectable && !this.options.showCheckBox) {
-      var span=document.createElement('a');
-      if (typeof isSelectable=='string') {
-        span.href=isSelectable;
-      } else {
-        span.href='javascript:void(0)';
-        Rico.eventBind(span, 'click', this.options.defaultAction);
-      }
-      this.hoverSet.add(span);
-    } else {
-      var span=document.createElement('p');
-    }
-    span.id=this.domID(nodeId,'Desc');
-    span.className='ricoTreeLevel'+level;
-    switch (this.options.nodeIdDisplay) {
-      case 'last': nodeDesc+=' ('+nodeId+')'; break;
-      case 'first': nodeDesc=nodeId+' - '+nodeDesc; break;
-      case 'tooltip': span.title=nodeId; break;
-    }
-       span.appendChild(document.createTextNode(nodeDesc));
-    row.insertCell(-1).appendChild(span);
-
-    var parent=parentChildren || this.treeDiv;
-    parent.appendChild(tab);
-    parent.appendChild(div);
-  },
-
-  nodeClick: function(e) {
-    var node=Rico.eventElement(e);
-    if (this.returnValue) {
-      var t=this.domID('','Desc');
-      this.returnValue(node.id.substr(t.length),node.innerHTML);
-    }
-    this.close();
-  },
-
-  saveSelection: function(e) {
-    if (this.returnValue) {
-      this.returnValue(this.getCheckedItems());
-    }
-    this.close();
-  },
-
-  getCheckedItems: function() {
-    var inp=this.treeDiv.getElementsByTagName('input');
-    var vals=[];
-    for (var i=0; i<inp.length; i++) {
-      if (inp[i].type=='checkbox' && inp[i].checked) {
-        vals.push(inp[i].value);
-      }
-    }
-    return vals;
-  },
-
-  setCheckBoxes: function(val) {
-    var inp=this.treeDiv.getElementsByTagName('input');
-    for (var i=0; i<inp.length; i++) {
-      if (inp[i].type=='checkbox') {
-        inp[i].checked=val;
-      }
-    }
-  },
-
-  clrCheckBoxEvent: function(e) {
-    Rico.eventStop(e);
-    this.setCheckBoxes(false);
-  },
-
-  clickBranch: function(e) {
-    var node=Rico.eventElement(e);
-    var tab=Rico.getParentByTagName(node,'table');
-    if (!tab || !tab.TreeContainer) return;
-    var a=tab.id.split('_');
-    a[1]='Children';
-    var childDiv=Rico.$(a.join('_'));
-    Rico.toggle(childDiv);
-    if (node.tagName=='IMG') {
-      var v=Rico.visible(childDiv);
-      if (node.src.match(/node(p|m)(last)?\.gif$/)) {
-        node.src=node.src.replace(/nodep|nodem/,'node'+(v ? 'm' : 'p'));
-      } else if (node.src.match(/folder(open|closed)\.gif$/)) {
-        node.src=node.src.replace(/folder(open|closed)/,'folder'+(v ? 'open' : 'closed'));
-      } else if (node.src.match(/\b(m|p)\.gif$/)) {
-        node.src=node.src.replace(/(p|m)\.gif/,v ? 'm\.gif' : 'p\.gif');
-      }
-    }
-    if (!tab.TreeFetchedChildren) {
-      tab.TreeFetchedChildren=1;
-      this.loadXMLDoc(node.name);
-    }
-  }
-
-};
diff --git a/ricoClient/js/minsrc/ricoUI.js b/ricoClient/js/minsrc/ricoUI.js
deleted file mode 100644 (file)
index 8d49a26..0000000
+++ /dev/null
@@ -1,1379 +0,0 @@
-/*
- *  (c) 2005-2009 Richard Cowin (http://openrico.org)
- *  (c) 2005-2009 Matt Brown (http://dowdybrown.com)
- *
- *  Licensed under the Apache License, Version 2.0 (the "License"); you may not use this
- *  file except in compliance with the License. You may obtain a copy of the License at
- *
- *         http://www.apache.org/licenses/LICENSE-2.0
- *
- *  Unless required by applicable law or agreed to in writing, software distributed under the
- *  License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- *  either express or implied. See the License for the specific language governing permissions
- *  and limitations under the License.
- */
-
-Rico.applyShadow = function(elem,shadowFlag) {
-  if (typeof shadowFlag=='undefined') shadowFlag=true;
-  if (shadowFlag) Rico.addClass(elem,'ricoShadow');
-  return elem;
-};
-
-Rico.Popup = function(containerDiv,options) {
-  this.initialize(containerDiv,options);
-};
-
-Rico.Popup.prototype = {
-/**
- * @class Class to manage pop-up div windows.
- * @constructs
- * @param options object may contain any of the following:<dl>
- *   <dt>hideOnEscape</dt><dd> hide popup when escape key is pressed? default=true</dd>
- *   <dt>hideOnClick </dt><dd> hide popup when mouse button is clicked? default=true</dd>
- *   <dt>ignoreClicks</dt><dd> if true, mouse clicks within the popup are not allowed to bubble up to parent elements</dd>
- *   <dt>position    </dt><dd> defaults to absolute, use "auto" to auto-detect</dd>
- *   <dt>shadow      </dt><dd> display shadow with popup? default=true</dd>
- *   <dt>zIndex      </dt><dd> which layer? default=1</dd>
- *   <dt>canDrag     </dt><dd> boolean value (or function that returns a boolean) indicating if it is ok to drag/reposition popup, default=false</dd>
- *   <dt>onClose     </dt><dd> function to call when the popup is closed</dd>
- *</dl>
- * @param containerDiv if supplied, then setDiv() is called at the end of initialization
- */
-  initialize: function(containerDiv,options) {
-    this.options = {
-      hideOnEscape  : true,
-      hideOnClick   : true,
-      ignoreClicks  : false,
-      position      : 'absolute',
-      shadow        : true,
-      zIndex        : 2,
-      canDrag       : false,
-      dragElement   : false,
-      closeFunc     : false
-    };
-    if (containerDiv) this.setDiv(containerDiv,options);
-  },
-
-  createContainer: function(options) {
-    this.setDiv(document.createElement('div'), options);
-    if (options && options.parent) {
-      options.parent.appendChild(this.container);
-    } else {
-      document.getElementsByTagName("body")[0].appendChild(this.container);
-    }
-  },
-
-/**
- * Apply popup behavior to a div that already exists in the DOM
- * @param containerDiv div element (or element id) in the DOM. If null, then the div is created automatically.
- */
-  setDiv: function(containerDiv,options) {
-    Rico.extend(this.options, options || {});
-    this.container=Rico.$(containerDiv);
-    if (this.options.position == 'auto') {
-      this.position=Rico.getStyle(this.container,'position').toLowerCase();
-    } else {
-      this.position=this.container.style.position=this.options.position;
-    }
-    if (this.position != 'absolute') {
-      this.content=this.container;
-      return;
-    }
-    if (this.options.zIndex >= 0) this.container.style.zIndex=this.options.zIndex;
-    if (this.options.closeFunc) {
-      this.closeFunc=this.options.closeFunc;
-    } else {
-      var self=this;
-      this.closeFunc=function() { self.closePopup(); };
-    }
-    //this.container.style.overflow='hidden';
-    this.container.style.top='0px';
-    this.container.style.left='0px';
-    this.container.style.display='none';
-
-    this.content=document.createElement('div');
-    while (this.container.firstChild) {
-      this.content.appendChild(this.container.firstChild);
-    }
-    this.container.appendChild(this.content);
-    this.content.className='RicoPopupContent';
-    this.content.style.position='relative';
-
-    if (Rico.isIE && Rico.ieVersion < 7) {
-      // create iframe shim
-      this.ifr = document.createElement('iframe');
-      this.ifr.style.position="absolute";
-      this.ifr.style.top     = '0px';
-      this.ifr.style.left    = '0px';
-      this.ifr.style.width   = '2000px';
-      this.ifr.style.height  = '2000px';
-      this.ifr.style.zIndex  = -1;
-      this.ifr.frameBorder   = 0;
-      this.ifr.src="javascript:false;";
-      this.content.appendChild(this.ifr);
-    }
-    Rico.applyShadow(this.container,this.options.shadow);
-
-    if (this.options.hideOnClick)
-      Rico.eventBind(document,"click", Rico.eventHandle(this,'_docClick'));
-    if (this.options.hideOnEscape)
-      Rico.eventBind(document,"keyup", Rico.eventHandle(this,'_checkKey'));
-    this.dragEnabled=false;
-    this.mousedownHandler = Rico.eventHandle(this,'_startDrag');
-    this.dragHandler = Rico.eventHandle(this,'_drag');
-    this.dropHandler = Rico.eventHandle(this,'_endDrag');
-    if (this.options.canDrag) this.enableDragging();
-    if (this.options.ignoreClicks || this.options.canDrag) this.ignoreClicks();
-  },
-
-  clearContent: function() {
-    this.content.innerHTML="";
-  },
-
-  setContent: function(content) {
-    this.content.innerHTML=content;
-  },
-
-  enableDragging: function() {
-    if (!this.dragEnabled && this.options.dragElement) {
-      Rico.eventBind(this.options.dragElement, "mousedown", this.mousedownHandler);
-      this.dragEnabled=true;
-    }
-    return this.dragEnabled;
-  },
-
-  disableDragging: function() {
-    if (!this.dragEnabled) return;
-    Rico.eventUnbind(this.options.dragElement, "mousedown", this.mousedownHandler);
-    this.dragEnabled=false;
-  },
-
-  setZ: function(zIndex) {
-    this.container.style.zIndex=zIndex;
-  },
-
-/** @private */
-  ignoreClicks: function() {
-    Rico.eventBind(this.container,"click", Rico.eventHandle(this,'_ignoreClick'));
-  },
-
-  _ignoreClick: function(e) {
-    if (e.stopPropagation)
-      e.stopPropagation();
-    else
-      e.cancelBubble = true;
-    return true;
-  },
-
-  // event handler to process keyup events (hide menu on escape key)
-  _checkKey: function(e) {
-    if (Rico.eventKey(e)!=27 || !this.visible()) return true;
-    //alert('closing popup: '+this.container.className);
-    Rico.eventStop(e);
-    this.closeFunc();
-    return false;
-  },
-
-  _docClick: function(e) {
-    this.closeFunc();
-    return true;
-  },
-
-/**
- * Move popup to specified position
- */
-  move: function(left,top) {
-    if (typeof left=='number') this.container.style.left=left+'px';
-    if (typeof top=='number') this.container.style.top=top+'px';
-  },
-
-  _startDrag : function(event){
-    var elem=Rico.eventElement(event);
-    this.container.style.cursor='move';
-    this.lastMouse = Rico.eventClient(event);
-    Rico.eventBind(document, "mousemove", this.dragHandler);
-    Rico.eventBind(document, "mouseup", this.dropHandler);
-    Rico.eventStop(event);
-  },
-
-  _drag : function(event){
-    var newMouse = Rico.eventClient(event);
-    var newLeft = parseInt(this.container.style.left,10) + newMouse.x - this.lastMouse.x;
-    var newTop = parseInt(this.container.style.top,10) + newMouse.y - this.lastMouse.y;
-    this.move(newLeft, newTop);
-    this.lastMouse = newMouse;
-    Rico.eventStop(event);
-  },
-
-  _endDrag : function(){
-    this.container.style.cursor='';
-    Rico.eventUnbind(document, "mousemove", this.dragHandler);
-    Rico.eventUnbind(document, "mouseup", this.dropHandler);
-  },
-
-/**
- * Display popup at specified position
- */
-  openPopup: function(left,top) {
-    this.container.style.display=this.position=='absolute' ? "block" : Rico.isIE && Rico.ieVersion<8 ? "inline" : "inline-block";
-    if (typeof left=='number') this.container.style.left=left+'px';
-    if (typeof top=='number') this.container.style.top=top+'px';
-    if (this.container.id) Rico.log('openPopup '+this.container.id+' at '+left+','+top);
-  },
-
-  centerPopup: function() {
-    this.openPopup();
-    var msgWidth=this.container.offsetWidth;
-    var msgHeight=this.container.offsetHeight;
-    var divwi=this.container.parentNode.offsetWidth;
-    var divht=this.container.parentNode.offsetHeight;
-    this.move(parseInt(Math.max((divwi-msgWidth)/2,0),10), parseInt(Math.max((divht-msgHeight)/2,0),10));
-  },
-
-  visible: function() {
-    return Rico.visible(this.container);
-  },
-
-/**
- * Hide popup
- */
-  closePopup: function() {
-    if (!Rico.visible(this.container)) return;
-    if (this.container.id) Rico.log('closePopup '+this.container.id);
-    if (this.dragEnabled) this._endDrag();
-    this.container.style.display="none";
-    if (this.options.onClose) this.options.onClose();
-  }
-
-};
-
-Rico.closeButton = function(handle) {
-  var a = document.createElement('a');
-  a.className='RicoCloseAnchor';
-  if (Rico.theme.closeAnchor) Rico.addClass(a,Rico.theme.closeAnchor);
-  var span = a.appendChild(document.createElement('span'));
-  span.title=Rico.getPhraseById('close');
-  new Rico.HoverSet([a]);
-  Rico.addClass(span,Rico.theme.close || 'RicoClose');
-  Rico.eventBind(a,"click", handle);
-  return a;
-};
-
-Rico.floatButton = function(buttonName, handle, title) {
-  var a=document.createElement("a");
-  a.className='RicoButtonAnchor'
-  Rico.addClass(a,Rico.theme.buttonAnchor || 'RicoButtonAnchorNative');
-  var span=a.appendChild(document.createElement("span"));
-  if (title) span.title=title;
-  span.className=Rico.theme[buttonName.toLowerCase()] || 'Rico'+buttonName;
-  Rico.eventBind(a,"click", handle, false);
-  new Rico.HoverSet([a]);
-  return a
-}
-
-Rico.clearButton = function(handle) {
-  var span=document.createElement("span");
-  span.title=Rico.getPhraseById('clear');
-  span.className='ricoClear';
-  Rico.addClass(span, Rico.theme.clear || 'ricoClearNative');
-  span.style.display='inline-block';
-  span.style.cursor='pointer';
-  Rico.eventBind(span,"click", handle);
-  return span;
-}
-
-Rico.Window = function(title, options, contentParam) {
-  this.initialize(title, options, contentParam);
-};
-
-Rico.Window.prototype = {
-
-/**
- * Create popup div with a title bar.
- */
-  initialize: function(title, options, contentParam) {
-    options=options || {overflow:'auto'};
-    Rico.extend(this, new Rico.Popup());
-
-    this.titleDiv = document.createElement('div');
-    this.options.canDrag=true;
-    this.options.dragElement=this.titleDiv;
-    this.options.hideOnClick=false;
-    this.createContainer(options);
-    this.content.appendChild(this.titleDiv);
-    contentParam=Rico.$(contentParam);
-    this.contentDiv=contentParam || document.createElement('div');
-    this.content.appendChild(this.contentDiv);
-
-    // create title area
-    this.titleDiv.className='ricoTitle';
-    if (Rico.theme.dialogTitle) Rico.addClass(this.titleDiv,Rico.theme.dialogTitle);
-    this.titleDiv.style.position='relative';
-    this.titleContent = document.createElement('span');
-    this.titleContent.className='ricoTitleSpan';
-    this.titleDiv.appendChild(this.titleContent);
-    this.titleDiv.appendChild(Rico.closeButton(Rico.eventHandle(this,'closePopup')));
-    if (!title && contentParam) {
-      title=contentParam.title;
-      contentParam.title='';
-    }
-    this.setTitle(title || '&nbsp;');
-
-    // create content area
-    this.contentDiv.className='ricoContent';
-    if (Rico.theme.dialogContent) Rico.addClass(this.contentDiv,Rico.theme.dialogContent);
-    this.contentDiv.style.position='relative';
-    if (options.height) this.contentDiv.style.height=options.height;
-    if (options.width) this.contentDiv.style.width=options.width;
-    if (options.overflow) this.contentDiv.style.overflow=options.overflow;
-    Rico.addClass(this.content,'ricoWindow');
-    if (Rico.theme.dialog) Rico.addClass(this.content,Rico.theme.dialog);
-    if (Rico.isIE) {
-      // fix float'ed content in IE
-      this.titleDiv.style.zoom=1;
-      this.contentDiv.style.zoom=1;
-    }
-
-    this.content=this.contentDiv;
-  },
-
-  setTitle: function(title) {
-    this.titleContent.innerHTML=title;
-  }
-
-}
-
-
-Rico.Menu = function(options) {
-  this.initialize(options);
-}
-
-Rico.Menu.prototype = {
-/**
- * @class Implements popup menus and submenus
- * @extends Rico.Popup
- * @constructs
- */
-  initialize: function(options) {
-    Rico.extend(this, new Rico.Popup());
-    Rico.extend(this.options, {
-      width        : "15em",
-      showDisabled : false
-    });
-    if (typeof options=='string')
-      this.options.width=options;
-    else
-      Rico.extend(this.options, options || {});
-    this.hideFunc=null;
-    this.highlightElem=null;
-    new Image().src = Rico.imgDir+'left_b.gif';
-    new Image().src = Rico.imgDir+'right_b.gif';
-  },
-
-  createDiv: function(parentNode) {
-    if (this.container) return;
-    var self=this;
-    var options={ closeFunc: function() { self.cancelmenu(); } };
-    if (parentNode) options.parent=parentNode;
-    this.createContainer(options);
-    this.content.className = Rico.isWebKit ? 'ricoMenuSafari' : 'ricoMenu';
-    this.content.style.width=this.options.width;
-    this.direction=Rico.getStyle(this.container,'direction') || 'ltr';
-    this.direction=this.direction.toLowerCase();  // ltr or rtl
-    this.hidemenu();
-    this.itemCount=0;
-  },
-
-  showmenu: function(e,hideFunc){
-    Rico.eventStop(e);
-    this.hideFunc=hideFunc;
-    if (this.content.childNodes.length==0) {
-      this.cancelmenu();
-      return false;
-    }
-    var mousePos = Rico.eventClient(e);
-    this.openmenu(mousePos.x,mousePos.y,0,0);
-  },
-
-  openmenu: function(x,y,clickItemWi,clickItemHt,noOffset) {
-    var newLeft=x + (noOffset ? 0 : Rico.docScrollLeft());
-    this.container.style.visibility="hidden";
-    this.container.style.display="block";
-    var w=this.container.offsetWidth;
-    var cw=this.content.offsetWidth;
-    //window.status='openmenu: newLeft='+newLeft+' width='+w+' clickItemWi='+clickItemWi+' windowWi='+Rico.windowWidth();
-    if (this.direction == 'rtl') {
-      if (newLeft > w+clickItemWi) newLeft-=cw+clickItemWi;
-    } else {
-      if (x+w > Rico.windowWidth()) newLeft-=cw+clickItemWi-2;
-    }
-    var scrTop=Rico.docScrollTop();
-    var newTop=y + (noOffset ? 0 : scrTop);
-    if (y+this.container.offsetHeight-scrTop > Rico.windowHeight())
-      newTop=Math.max(newTop-this.content.offsetHeight+clickItemHt,0);
-    this.openPopup(newLeft,newTop);
-    this.container.style.visibility ="visible";
-    return false;
-  },
-
-  clearMenu: function() {
-    this.clearContent();
-    this.defaultAction=null;
-    this.itemCount=0;
-  },
-
-  addMenuHeading: function(hdg) {
-    var el=document.createElement('div');
-    el.innerHTML=hdg;
-    el.className='ricoMenuHeading';
-    this.content.appendChild(el);
-  },
-
-  addMenuBreak: function() {
-    var brk=document.createElement('div');
-    brk.className="ricoMenuBreak";
-    this.content.appendChild(brk);
-  },
-
-  addSubMenuItem: function(menutext, submenu, translate) {
-    var dir=this.direction=='rtl' ? 'left' : 'right';
-    var a=this.addMenuItem(menutext,null,true,null,translate);
-    a.className='ricoSubMenu';
-    a.style.backgroundImage='url('+Rico.imgDir+dir+'_b.gif)';
-    a.style.backgroundRepeat='no-repeat';
-    a.style.backgroundPosition=dir;
-    a.RicoSubmenu=submenu;
-    Rico.eventBind(a,"mouseover", Rico.eventHandle(this,'showSubMenu'));
-    Rico.eventBind(a,"mouseout", Rico.eventHandle(this,'subMenuOut'));
-  },
-
-  showSubMenu: function(e) {
-    if (this.openSubMenu) this.hideSubMenu();
-    var a=Rico.eventElement(e);
-    this.openSubMenu=a.RicoSubmenu;
-    this.openMenuAnchor=a;
-    if (a.className=='ricoSubMenu') a.className='ricoSubMenuOpen';
-    a.RicoSubmenu.openmenu(parseInt(this.container.style.left)+a.offsetWidth, parseInt(this.container.style.top)+a.offsetTop, a.offsetWidth-2, a.offsetHeight+2,true);
-  },
-
-  subMenuOut: function(e) {
-    if (!this.openSubMenu) return;
-    Rico.eventStop(e);
-    var elem=Rico.eventElement(e);
-    var reltg = Rico.eventRelatedTarget(e) || e.toElement;
-    try {
-      while (reltg != null && reltg != this.openSubMenu.div)
-        reltg=reltg.parentNode;
-    } catch(err) {}
-    if (reltg == this.openSubMenu.div) return;
-    this.hideSubMenu();
-  },
-
-  hideSubMenu: function() {
-    if (this.openMenuAnchor) {
-      this.openMenuAnchor.className='ricoSubMenu';
-      this.openMenuAnchor=null;
-    }
-    if (this.openSubMenu) {
-      this.openSubMenu.hidemenu();
-      this.openSubMenu=null;
-    }
-  },
-
-  addMenuItemId: function(phraseId,action,enabled,title,target) {
-    if ( arguments.length < 3 ) enabled=true;
-    this.addMenuItem(Rico.getPhraseById(phraseId),action,enabled,title,target);
-  },
-
-// if action is a string, then it is assumed to be a URL and the target parm can be used indicate which window gets the content
-// action can also be a function
-// action can also be a Rico.eventHandle, but set target='event' in this case
-  addMenuItem: function(menutext,action,enabled,title,target) {
-    if (arguments.length >= 3 && !enabled && !this.options.showDisabled) return null;
-    this.itemCount++;
-    var a = document.createElement(typeof action=='string' ? 'a' : 'div');
-    if ( arguments.length < 3 || enabled ) {
-      if (typeof action=='string') {
-        a.href = action;
-        if (target) a.target = target;
-      } else if (target=='event') {
-        Rico.eventBind(a,"click", action);
-      } else {
-        a.onclick=action;
-      }
-      a.className = 'enabled';
-      if (this.defaultAction==null) this.defaultAction=action;
-    } else {
-      a.disabled = true;
-      a.className = 'disabled';
-    }
-    a.innerHTML = menutext;
-    if (typeof title=='string')
-      a.title = title;
-    a=this.content.appendChild(a);
-    Rico.eventBind(a,"mouseover", Rico.eventHandle(this,'mouseOver'));
-    Rico.eventBind(a,"mouseout", Rico.eventHandle(this,'mouseOut'));
-    return a;
-  },
-
-  mouseOver: function(e) {
-    if (this.highlightElem && this.highlightElem.className=='enabled-hover') {
-      // required for Safari
-      this.highlightElem.className='enabled';
-      this.highlightElem=null;
-    }
-    var elem=Rico.eventElement(e);
-    if (this.openMenuAnchor && this.openMenuAnchor!=elem)
-      this.hideSubMenu();
-    if (elem.className=='enabled') {
-      elem.className='enabled-hover';
-      this.highlightElem=elem;
-    }
-  },
-
-  mouseOut: function(e) {
-    var elem=Rico.eventElement(e);
-    if (elem.className=='enabled-hover') elem.className='enabled';
-    if (this.highlightElem==elem) this.highlightElem=null;
-  },
-
-  cancelmenu: function() {
-    if (!this.visible()) return;
-    if (this.hideFunc) this.hideFunc();
-    this.hideFunc=null;
-    this.hidemenu();
-  },
-
-  hidemenu: function() {
-    if (this.openSubMenu) this.openSubMenu.hidemenu();
-    this.closePopup();
-  }
-
-}
-
-
-Rico.SelectionSet = function(selectionSet, options) {
-  this.initialize(selectionSet, options);
-}
-
-Rico.SelectionSet.prototype = {
-/**
- * @class
- * @constructs
- * @param selectionSet collection of DOM elements (or a CSS selection string)
- * @param options object may contain any of the following:<dl>
- *   <dt>selectedClass</dt><dd>class name to add when element is selected, default is "selected"</dd>
- *   <dt>selectNode   </dt><dd>optional function that returns the element to be selected</dd>
- *   <dt>onSelect     </dt><dd>optional function that gets called when element is selected</dd>
- *   <dt>onFirstSelect</dt><dd>optional function that gets called the first time element is selected</dd>
- *   <dt>noDefault    </dt><dd>when true, no element in the set is initially selected, default is false</dd>
- *   <dt>selectedIndex</dt><dd>index of the element that should be initially selected, default is 0</dd>
- *   <dt>cookieName   </dt><dd>optional name of cookie to use to remember selected element. If specified, and the cookie exists, then the cookie value overrides selectedIndex.</dd>
- *   <dt>cookieDays   </dt><dd>specifies how long cookie should persist (in days). If unspecified, then the cookie persists for the current session.</dd>
- *   <dt>cookiePath   </dt><dd>optional cookie path</dd>
- *   <dt>cookieDomain </dt><dd>optional cookie domain</dd>
- *</dl>
- */
-  initialize: function(selectionSet, options){
-    Rico.log('SelectionSet#initialize');
-    this.options = options || {};
-    if (typeof selectionSet == 'string')
-      selectionSet = Rico.select(selectionSet);
-    this.previouslySelected = [];
-    this.selectionSet = [];
-    this.selectedClassName = this.options.selectedClass || Rico.theme.selected || "selected";
-    this.selectNode = this.options.selectNode || function(e){return e;};
-    this.onSelect = this.options.onSelect;
-    this.onFirstSelect = this.options.onFirstSelect;
-    var self=this;
-    this.clickHandler = function(idx) { self.selectIndex(idx); };
-    this.selectedIndex=-1;
-    for (var i=0; i<selectionSet.length; i++)
-      this.add(selectionSet[i]);
-    if (!this.options.noDefault) {
-      var cookieIndex=this.options.cookieName ? this.getCookie() : 0;
-      this.selectIndex(cookieIndex || this.options.selectedIndex || 0);
-    }
-  },
-  getCookie: function() {
-    var cookie = Rico.getCookie(this.options.cookieName);
-    if (!cookie) return 0;
-    var index = parseInt(cookie);
-    return index < this.selectionSet.length ? index : 0;
-  },
-  reset: function(){
-    this.previouslySelected = [];
-    this._notifySelected(this.selectedIndex);
-  },
-  clearSelected: function() {
-    if (this.selected)
-      Rico.removeClass(this.selectNode(this.selected), this.selectedClassName);
-  },
-  getIndex: function(element) {
-    for (var i=0; i<this.selectionSet.length; i++) {
-      if (element == this.selectionSet[i]) return i;
-    }
-    return -1;
-  },
-  select: function(element){
-    if (this.selected == element) return;
-    var i=this.getIndex(element);
-    if (i >= 0) this.selectIndex(i);
-  },
-  _notifySelected: function(index){
-    if (index < 0) return;
-    var element = this.selectionSet[index];
-    if (this.options.cookieName)
-      Rico.setCookie(this.options.cookieName, index, this.options.cookieDays, this.options.cookiePath, this.options.cookieDomain);
-    if (this.onFirstSelect && !this.previouslySelected[index]){
-      this.onFirstSelect(element, index);
-      this.previouslySelected[index] = true;
-    }
-    if (this.onSelect)
-      try{
-        this.onSelect(index);
-      } catch (e) {};
-  },
-  selectIndex: function(index){
-    if (this.selectedIndex == index || index >= this.selectionSet.length) return;
-    this.clearSelected();
-    this._notifySelected(index);
-    this.selectedIndex = index;
-    this.selected=this.selectionSet[index].element;
-    Rico.addClass(this.selectNode(this.selected), this.selectedClassName);
-  },
-  nextSelectIndex: function(){
-    return (this.selectedIndex + 1) % this.selectionSet.length;
-  },
-  nextSelectItem: function(){
-    return this.selectionSet[this.nextSelectIndex()];
-  },
-  selectNext: function(){
-    this.selectIndex(this.nextSelectIndex());
-  },
-  add: function(item){
-    var index=this.selectionSet.length;
-    this.selectionSet[index] = new Rico._SelectionItem(item,index,this.clickHandler);
-  },
-  remove: function(item){
-    if (item==this.selected) this.clearSelected();
-    var i=this.getIndex(item);
-    if (i < 0) return;
-    this.selectionSet[i].remove();
-    this.selectionSet.splice(i,1);
-  },
-  removeAll: function(){
-    this.clearSelected();
-    while (this.selectionSet.length > 0) {
-      this.selectionSet.pop().remove();
-    }
-  }
-};
-
-
-Rico._SelectionItem=function(element,index,callback) {
-  this.add(element,index,callback);
-};
-
-Rico._SelectionItem.prototype = {
-  add: function(element,index,callback) {
-    this.element=element;
-    this.index=index;
-    this.callback=callback;
-    this.handle=Rico.eventHandle(this,'click');
-    Rico.eventBind(element, "click", this.handle);
-  },
-
-  click: function(ev) {
-    this.callback(this.index);
-  },
-
-  remove: function() {
-    Rico.eventUnbind(this.element, "click", this.handle);
-  }
-};
-
-
-Rico.HoverSet = function(hoverSet, options) {
-  this.initialize(hoverSet, options);
-};
-
-Rico.HoverSet.prototype = {
-/**
- * @class
- * @constructs
- * @param hoverSet collection of DOM elements
- * @param options object may contain any of the following:<dl>
- *   <dt>hoverClass</dt><dd> class name to add when mouse is over element, default is "hover"</dd>
- *   <dt>hoverNodes</dt><dd> optional function to select/filter which nodes are in the set</dd>
- *</dl>
- */
-  initialize: function(hoverSet, options){
-    Rico.log('HoverSet#initialize');
-    options = options || {};
-    this.hoverClass = options.hoverClass || Rico.theme.hover || "hover";
-    this.hoverFunc = options.hoverNodes || function(e){return [e];};
-    this.hoverSet=[];
-    if (!hoverSet) return;
-    for (var i=0; i<hoverSet.length; i++)
-      this.add(hoverSet[i]);
-  },
-  add: function(item) {
-    this.hoverSet.push(new Rico._HoverItem(item,this.hoverFunc,this.hoverClass));
-  },
-  removeAll: function(){
-    while (this.hoverSet.length > 0) {
-      this.hoverSet.pop().remove();
-    }
-  }
-};
-
-
-Rico._HoverItem=function(element,selectFunc,hoverClass) {
-  this.add(element,selectFunc,hoverClass);
-};
-
-Rico._HoverItem.prototype = {
-  add: function(element,selectFunc,hoverClass) {
-    this.element=element;
-    this.selectFunc=selectFunc;
-    this.hoverClass=hoverClass;
-    this.movehandle=Rico.eventHandle(this,'move');
-    this.outhandle=Rico.eventHandle(this,'mouseout');
-    Rico.eventBind(element, "mousemove", this.movehandle);
-    Rico.eventBind(element, "mouseout", this.outhandle);
-  },
-
-  move: function(ev) {
-    var elems=this.selectFunc(this.element);
-    for (var i=0; i<elems.length; i++)
-      Rico.addClass(elems[i],this.hoverClass);
-  },
-
-  mouseout: function(ev) {
-    var elems=this.selectFunc(this.element);
-    for (var i=0; i<elems.length; i++)
-      Rico.removeClass(elems[i],this.hoverClass);
-  },
-
-  remove: function() {
-    Rico.eventUnbind(element, "mousemove", this.movehandle);
-    Rico.eventUnbind(element, "mouseout", this.outhandle);
-  }
-};
-
-
-/** @namespace */
-Rico.Effect = {};
-Rico.Effect.easeIn = function(step){
-  return Math.sqrt(step);
-};
-Rico.Effect.easeOut = function(step){
-  return step*step;
-};
-
-
-/** @class core methods for transition effects */
-Rico.ContentTransitionBase = function() {};
-Rico.ContentTransitionBase.prototype = {
-  initBase: function(titles, contents, options) {
-    Rico.log('ContentTransitionBase#initBase');
-    if (typeof titles == 'string')
-      titles = Rico.select(titles);
-    if (typeof contents == 'string')
-      contents = Rico.select(contents);
-
-    this.options = options || {};
-    this.titles = titles;
-    this.contents = contents;
-    this.hoverSet = new Rico.HoverSet(titles, options);
-    for (var i=0; i<contents.length; i++) {
-      if (contents[i]) Rico.hide(contents[i]);
-    }
-    var self=this;
-    this.selectionSet = new Rico.SelectionSet(titles, Rico.extend(options, { onSelect: function(idx) { self._finishSelect(idx); } }));
-  },
-  reset: function(){
-    this.selectionSet.reset();
-  },
-  select: function(index) {
-    this.selectionSet.selectIndex(index);
-  },
-  _finishSelect: function(index) {
-    Rico.log('ContentTransitionBase#_finishSelect');
-    var panel = this.contents[index];
-    if (!panel) {
-      alert('Internal error: no panel @index='+index);
-      return;
-    }
-    if ( this.selected == panel) return;
-    if (this.transition){
-      if (this.selected){
-        this.transition(panel);
-      } else {
-        panel.style.display='block';
-      }
-    } else {
-      if (this.selected) Rico.hide(this.selected);
-      panel.style.display='block';
-    }
-    this.selected = panel;
-  },
-  addBase: function(title, content){
-    this.titles.push(title);
-    this.contents.push(content);
-    this.hoverSet.add(title);
-    this.selectionSet.add(title);
-    Rico.hide(content);
-    //this.selectionSet.select(title);
-  },
-  removeBase: function(title){},
-  removeAll: function(){
-    this.hoverSet.removeAll();
-    this.selectionSet.removeAll();
-  }
-};
-
-
-/**
- * @class Implements accordion effect
- * @see Rico.ContentTransitionBase#initialize for construction parameters
- * @extends Rico.ContentTransitionBase
- */
-Rico.Accordion = function(element, options) {
-  this.initialize(element, options);
-};
-
-Rico.Accordion.prototype = Rico.extend(new Rico.ContentTransitionBase(),
-/** @lends Rico.Accordion# */
-{
-  initialize: function(element, options) {
-    element=Rico.$(element);
-    element.style.overflow='hidden';
-    element.className=options.accClass || Rico.theme.accordion || "Rico_accordion";
-    if (typeof options.panelWidth=='number') options.panelWidth+="px";
-    if (options.panelWidth) element.style.width = options.panelWidth;
-    var panels=Rico.getDirectChildrenByTag(element,'div');
-    var items,titles=[], contents=[];
-    for (var i=0; i<panels.length; i++) {
-      items=Rico.getDirectChildrenByTag(panels[i],'div');
-      if (items.length>=2) {
-        items[0].className=options.titleClass || Rico.theme.accTitle || "Rico_accTitle";
-        items[1].className=options.contentClass || Rico.theme.accContent || "Rico_accContent";
-        titles.push(items[0]);
-        contents.push(items[1]);
-        var a=Rico.wrapChildren(items[0],'','','a');
-        a.href="javascript:void(0)";
-      }
-    }
-    Rico.log('creating Rico.Accordion for '+element.id+' with '+titles.length+' panels');
-    this.initBase(titles, contents, options);
-    this.selected.style.height = this.options.panelHeight + "px";
-    this.totSteps=(typeof options.duration =='number' ? options.duration : 200)/25;
-  },
-  transition: function(p){
-    if (!this.options.noAnimate) {
-      this.closing=this.selected;
-      this.opening=p;
-      this.curStep=0;
-      var self=this;
-      this.timer=setInterval(function() { self.step(); },25);
-    } else {
-      p.style.height = this.options.panelHeight + "px";
-      if (this.selected) Rico.hide(this.selected);
-      p.style.display='block';
-    }
-  },
-  step: function() {
-    this.curStep++;
-    var oheight=Math.round(this.curStep/this.totSteps*this.options.panelHeight);
-    this.opening.style.height=oheight+'px';
-    this.closing.style.height=(this.options.panelHeight - oheight)+'px';
-    if (this.curStep==1) {
-      this.opening.style.paddingTop=this.opening.style.paddingBottom='0px';
-      this.opening.style.display='block';
-    }
-    if (this.curStep==this.totSteps) {
-      clearInterval(this.timer);
-      this.opening.style.paddingTop=this.opening.style.paddingBottom='';
-      Rico.hide(this.closing);
-    }
-  },
-  setPanelHeight: function(h) {
-    this.options.panelHeight = h;
-    this.selected.style.height = this.options.panelHeight + "px";
-  }
-});
-
-
-/**
- * @class Implements tabbed panel effect
- * @see Rico.ContentTransitionBase#initialize for construction parameters
- * @extends Rico.ContentTransitionBase
- */
-Rico.TabbedPanel = function(element, options) {
-  this.initialize(element, options);
-};
-
-Rico.TabbedPanel.prototype = Rico.extend(new Rico.ContentTransitionBase(),
-{
-  initialize: function(element, options) {
-    element=Rico.$(element);
-    options=options || {};
-    if (typeof options.panelWidth=='number') options.panelWidth+="px";
-    if (typeof options.panelHeight=='number') options.panelHeight+="px";
-    element.className=options.tabClass || Rico.theme.tabPanel || "Rico_tabPanel";
-    if (options.panelWidth) element.style.width = options.panelWidth;
-    var items = [];
-    var allKids = element.childNodes;
-    for( var i = 0 ; i < allKids.length ; i++ ) {
-      if (allKids[i] && allKids[i].tagName && allKids[i].tagName.match(/^div|ul$/i))
-        items.push(allKids[i]);
-    }
-    if (items.length < 2) return;
-    var childTag=items[0].tagName.toLowerCase()=='ul' ? 'li' : 'div';
-    items[0].className=options.navContainerClass || Rico.theme.tabNavContainer || "Rico_tabNavContainer";
-    items[0].style.listStyle='none';
-    items[1].className=options.contentContainerClass || Rico.theme.tabContentContainer || "Rico_tabContentContainer";
-    var titles=Rico.getDirectChildrenByTag(items[0], childTag);
-    var contents=Rico.getDirectChildrenByTag(items[1],'div');
-    if (!options.corners) options.corners='top';
-    for (var i=0; i<titles.length; i++) {
-      titles[i].className=options.titleClass || Rico.theme.tabTitle || "Rico_tabTitle";
-      var a=Rico.wrapChildren(titles[i],'','','a');
-      a.href="javascript:void(0)";
-      contents[i].className=options.contentClass || Rico.theme.tabContent || "Rico_tabContent";
-      if (options.panelHeight) contents[i].style.overflow='auto';
-      if (options.corners!='none') {
-        if (options.panelHdrWidth) titles[i].style.width=options.panelHdrWidth;
-        Rico.Corner.round(titles[i], Rico.theme.tabCornerOptions || options);
-      }
-    }
-    options.selectedClass=Rico.theme.tabSelected || 'selected';
-    this.initBase(titles, contents, options);
-    if (this.selected) this.transition(this.selected);
-  },
-  transition: function(p){
-    Rico.log('TabbedPanel#transition '+typeof(p));
-    if (this.selected) Rico.hide(this.selected);
-    Rico.show(p);
-    if (this.options.panelHeight) p.style.height = this.options.panelHeight;
-  }
-});
-
-
-/**
- * @namespace
- */
-Rico.Corner = {
-
-   round: function(e, options) {
-      e = Rico.$(e);
-      this.options = {
-         corners : "all",
-         bgColor : "fromParent",
-         compact : false,
-         nativeCorners: false  // only use native corners?
-      };
-      Rico.extend(this.options, options || {});
-      if (typeof(Rico.getStyle(e,'border-radius'))=='string')
-        this._roundCornersStdCss(e);
-      else if (typeof(Rico.getStyle(e,'-webkit-border-radius'))=='string')
-        this._roundCornersWebKit(e);
-      else if (typeof(Rico.getStyle(e,'-moz-border-radius'))=='string')
-        this._roundCornersMoz(e);
-      else if (!this.options.nativeCorners)
-        this._roundCornersImpl(e);
-   },
-
-    _roundCornersStdCss: function(e) {
-      var radius=this.options.compact ? '4px' : '8px';
-      if (this._hasString(this.options.corners, "all"))
-        Rico.setStyle(e, {borderRadius:radius});
-      else {
-        if (this._hasString(this.options.corners, "top", "tl")) Rico.setStyle(e, {borderTopLeftRadius:radius});
-        if (this._hasString(this.options.corners, "top", "tr")) Rico.setStyle(e, {borderTopRightRadius:radius});
-        if (this._hasString(this.options.corners, "bottom", "bl")) Rico.setStyle(e, {borderBottomLeftRadius:radius});
-        if (this._hasString(this.options.corners, "bottom", "br")) Rico.setStyle(e, {borderBottomRightRadius:radius});
-      }
-   },
-
-   _roundCornersWebKit: function(e) {
-      var radius=this.options.compact ? '4px' : '8px';
-      if (this._hasString(this.options.corners, "all"))
-        Rico.setStyle(e, {WebkitBorderRadius:radius});
-      else {
-        if (this._hasString(this.options.corners, "top", "tl")) Rico.setStyle(e, {WebkitBorderTopLeftRadius:radius});
-        if (this._hasString(this.options.corners, "top", "tr")) Rico.setStyle(e, {WebkitBorderTopRightRadius:radius});
-        if (this._hasString(this.options.corners, "bottom", "bl")) Rico.setStyle(e, {WebkitBorderBottomLeftRadius:radius});
-        if (this._hasString(this.options.corners, "bottom", "br")) Rico.setStyle(e, {WebkitBorderBottomRightRadius:radius});
-      }
-   },
-
-   _roundCornersMoz: function(e) {
-      var radius=this.options.compact ? '4px' : '8px';
-      if (this._hasString(this.options.corners, "all"))
-        Rico.setStyle(e, {MozBorderRadius:radius});
-      else {
-        if (this._hasString(this.options.corners, "top", "tl")) Rico.setStyle(e, {MozBorderRadiusTopleft:radius});
-        if (this._hasString(this.options.corners, "top", "tr")) Rico.setStyle(e, {MozBorderRadiusTopright:radius});
-        if (this._hasString(this.options.corners, "bottom", "bl")) Rico.setStyle(e, {MozBorderRadiusBottomleft:radius});
-        if (this._hasString(this.options.corners, "bottom", "br")) Rico.setStyle(e, {MozBorderRadiusBottomright:radius});
-      }
-   },
-
-  _roundCornersImpl: function(e) {
-      var bgColor = this.options.bgColor == "fromParent" ? this._background(e.parentNode) : this.options.bgColor;
-      e.style.position='relative';
-      //this.options.numSlices = this.options.compact ? 2 : 4;
-      if (this._hasString(this.options.corners, "all", "top", "tl")) this._createCorner(e,'top','left',bgColor);
-      if (this._hasString(this.options.corners, "all", "top", "tr")) this._createCorner(e,'top','right',bgColor);
-      if (this._hasString(this.options.corners, "all", "bottom", "bl")) this._createCorner(e,'bottom','left',bgColor);
-      if (this._hasString(this.options.corners, "all", "bottom", "br")) this._createCorner(e,'bottom','right',bgColor);
-   },
-
-  _createCorner: function(elem,tb,lr,bgColor) {
-    //alert('Corner: '+tb+' '+lr+' bgColor='+typeof(bgColor));
-    var corner = document.createElement("div");
-    corner.className='ricoCorner';
-    Rico.setStyle(corner,{width:'6px', height:'5px'});
-    var borderStyle = Rico.getStyle(elem,'border-'+tb+'-style');
-    var borderColor = borderStyle=='none' ? bgColor : Rico.getStyle(elem,'border-'+tb+'-color');
-    //alert('Corner: '+tb+' '+borderStyle+borderColor+' '+);
-    var pos=borderStyle=='none' ? '0px' : '-1px';
-    corner.style[tb]=pos;
-    corner.style[lr]=Rico.isIE && Rico.ieVersion<7 && lr=='right' && borderStyle!='none' ? '-2px' : '-1px';
-    //corner.style[lr]='-1px';
-    elem.appendChild(corner);
-    var marginSizes = [ 0, 2, 3, 4, 4 ];
-    if (tb=='bottom') marginSizes.reverse();
-    var borderVal= borderStyle=='none' ? '0px none' : '1px solid '+borderColor;
-    var d= lr=='left' ? 'Right' : 'Left';
-    for (var i=0; i<marginSizes.length; i++) {
-      var slice = document.createElement("div");
-      Rico.setStyle(slice,{backgroundColor:bgColor,height:'1px'});
-      slice.style['margin'+d]=marginSizes[i]+'px';
-      slice.style['border'+d]=borderVal;
-      corner.appendChild(slice);
-    }
-  },
-
-  _background: function(elem) {
-     try {
-       var actualColor = Rico.getStyle(elem, "backgroundColor");
-
-       // if color is tranparent, check parent
-       // Safari returns "rgba(0, 0, 0, 0)", which means transparent
-       if ( actualColor.match(/^(transparent|rgba\(0,\s*0,\s*0,\s*0\))$/i) && elem.parentNode )
-          return this._background(elem.parentNode);
-
-       return actualColor == null ? "#ffffff" : actualColor;
-     } catch(err) {
-       return "#ffffff";
-     }
-   },
-
-   _hasString: function(str) {
-     for(var i=1 ; i<arguments.length ; i++) {
-       if (str.indexOf(arguments[i]) >= 0) return true;
-     }
-     return false;
-   }
-
-};
-
-Rico.toColorPart = function(c) {
-  return Rico.zFill(c, 2, 16);
-};
-
-
-Rico.Color = function(red, green, blue) {
-  this.initialize(red, green, blue);
-};
-
-Rico.Color.prototype = {
-/**
- * @class Methods to manipulate color values.
- * @constructs
- * @param red integer (0-255)
- * @param green integer (0-255)
- * @param blue integer (0-255)
- */
-   initialize: function(red, green, blue) {
-      this.rgb = { r: red, g : green, b : blue };
-   },
-
-   setRed: function(r) {
-      this.rgb.r = r;
-   },
-
-   setGreen: function(g) {
-      this.rgb.g = g;
-   },
-
-   setBlue: function(b) {
-      this.rgb.b = b;
-   },
-
-   setHue: function(h) {
-
-      // get an HSB model, and set the new hue...
-      var hsb = this.asHSB();
-      hsb.h = h;
-
-      // convert back to RGB...
-      this.rgb = Rico.Color.HSBtoRGB(hsb.h, hsb.s, hsb.b);
-   },
-
-   setSaturation: function(s) {
-      // get an HSB model, and set the new hue...
-      var hsb = this.asHSB();
-      hsb.s = s;
-
-      // convert back to RGB and set values...
-      this.rgb = Rico.Color.HSBtoRGB(hsb.h, hsb.s, hsb.b);
-   },
-
-   setBrightness: function(b) {
-      // get an HSB model, and set the new hue...
-      var hsb = this.asHSB();
-      hsb.b = b;
-
-      // convert back to RGB and set values...
-      this.rgb = Rico.Color.HSBtoRGB( hsb.h, hsb.s, hsb.b );
-   },
-
-   darken: function(percent) {
-      var hsb  = this.asHSB();
-      this.rgb = Rico.Color.HSBtoRGB(hsb.h, hsb.s, Math.max(hsb.b - percent,0));
-   },
-
-   brighten: function(percent) {
-      var hsb  = this.asHSB();
-      this.rgb = Rico.Color.HSBtoRGB(hsb.h, hsb.s, Math.min(hsb.b + percent,1));
-   },
-
-   blend: function(other) {
-      this.rgb.r = Math.floor((this.rgb.r + other.rgb.r)/2);
-      this.rgb.g = Math.floor((this.rgb.g + other.rgb.g)/2);
-      this.rgb.b = Math.floor((this.rgb.b + other.rgb.b)/2);
-   },
-
-   isBright: function() {
-      var hsb = this.asHSB();
-      return this.asHSB().b > 0.5;
-   },
-
-   isDark: function() {
-      return ! this.isBright();
-   },
-
-   asRGB: function() {
-      return "rgb(" + this.rgb.r + "," + this.rgb.g + "," + this.rgb.b + ")";
-   },
-
-   asHex: function() {
-      return "#" + Rico.toColorPart(this.rgb.r) + Rico.toColorPart(this.rgb.g) + Rico.toColorPart(this.rgb.b);
-   },
-
-   asHSB: function() {
-      return Rico.Color.RGBtoHSB(this.rgb.r, this.rgb.g, this.rgb.b);
-   },
-
-   toString: function() {
-      return this.asHex();
-   }
-
-};
-
-/**
- * Factory method for creating a color from an RGB string
- * @param hexCode a 3 or 6 digit hex string, optionally preceded by a # symbol
- * @returns a Rico.Color object
- */
-Rico.Color.createFromHex = function(hexCode) {
-  if(hexCode.length==4) {
-    var shortHexCode = hexCode;
-    hexCode = '#';
-    for(var i=1;i<4;i++)
-      hexCode += (shortHexCode.charAt(i) + shortHexCode.charAt(i));
-  }
-  if ( hexCode.indexOf('#') == 0 )
-    hexCode = hexCode.substring(1);
-  if (!hexCode.match(/^[0-9A-Fa-f]{6}$/)) return null;
-  var red   = hexCode.substring(0,2);
-  var green = hexCode.substring(2,4);
-  var blue  = hexCode.substring(4,6);
-  return new Rico.Color( parseInt(red,16), parseInt(green,16), parseInt(blue,16) );
-};
-
-/**
- * Retrieves the background color of an HTML element
- * @param elem the DOM element whose background color should be retreived
- * @returns a Rico.Color object
- */
-Rico.Color.createColorFromBackground = function(elem) {
-
-   if (!elem.style) return new Rico.Color(255,255,255);
-   var actualColor = Rico.getStyle(elem, "background-color");
-
-   // if color is tranparent, check parent
-   // Safari returns "rgba(0, 0, 0, 0)", which means transparent
-   if ( actualColor.match(/^(transparent|rgba\(0,\s*0,\s*0,\s*0\))$/i) && elem.parentNode )
-      return Rico.Color.createColorFromBackground(elem.parentNode);
-
-   if (actualColor == null) return new Rico.Color(255,255,255);
-
-   if ( actualColor.indexOf("rgb(") == 0 ) {
-      var colors = actualColor.substring(4, actualColor.length - 1 );
-      var colorArray = colors.split(",");
-      return new Rico.Color( parseInt( colorArray[0],10 ),
-                             parseInt( colorArray[1],10 ),
-                             parseInt( colorArray[2],10 )  );
-
-   }
-   else if ( actualColor.indexOf("#") == 0 ) {
-      return Rico.Color.createFromHex(actualColor);
-   }
-   else
-      return new Rico.Color(255,255,255);
-};
-
-/**
- * Converts hue/saturation/brightness to RGB
- * @returns a 3-element object: r=red, g=green, b=blue.
- */
-Rico.Color.HSBtoRGB = function(hue, saturation, brightness) {
-
-  var red   = 0;
-  var green = 0;
-  var blue  = 0;
-
-  if (saturation == 0) {
-     red = parseInt(brightness * 255.0 + 0.5,10);
-     green = red;
-     blue = red;
-  }
-  else {
-      var h = (hue - Math.floor(hue)) * 6.0;
-      var f = h - Math.floor(h);
-      var p = brightness * (1.0 - saturation);
-      var q = brightness * (1.0 - saturation * f);
-      var t = brightness * (1.0 - (saturation * (1.0 - f)));
-
-      switch (parseInt(h,10)) {
-         case 0:
-            red   = (brightness * 255.0 + 0.5);
-            green = (t * 255.0 + 0.5);
-            blue  = (p * 255.0 + 0.5);
-            break;
-         case 1:
-            red   = (q * 255.0 + 0.5);
-            green = (brightness * 255.0 + 0.5);
-            blue  = (p * 255.0 + 0.5);
-            break;
-         case 2:
-            red   = (p * 255.0 + 0.5);
-            green = (brightness * 255.0 + 0.5);
-            blue  = (t * 255.0 + 0.5);
-            break;
-         case 3:
-            red   = (p * 255.0 + 0.5);
-            green = (q * 255.0 + 0.5);
-            blue  = (brightness * 255.0 + 0.5);
-            break;
-         case 4:
-            red   = (t * 255.0 + 0.5);
-            green = (p * 255.0 + 0.5);
-            blue  = (brightness * 255.0 + 0.5);
-            break;
-          case 5:
-            red   = (brightness * 255.0 + 0.5);
-            green = (p * 255.0 + 0.5);
-            blue  = (q * 255.0 + 0.5);
-            break;
-      }
-  }
-
-   return { r : parseInt(red,10), g : parseInt(green,10) , b : parseInt(blue,10) };
-};
-
-/**
- * Converts RGB value to hue/saturation/brightness
- * @param r integer (0-255)
- * @param g integer (0-255)
- * @param b integer (0-255)
- * @returns a 3-element object: h=hue, s=saturation, b=brightness.
- * (unlike some HSB documentation which states hue should be a value 0-360, this routine returns hue values from 0 to 1.0)
- */
-Rico.Color.RGBtoHSB = function(r, g, b) {
-
-   var hue;
-   var saturation;
-   var brightness;
-
-   var cmax = (r > g) ? r : g;
-   if (b > cmax)
-      cmax = b;
-
-   var cmin = (r < g) ? r : g;
-   if (b < cmin)
-      cmin = b;
-
-   brightness = cmax / 255.0;
-   if (cmax != 0)
-      saturation = (cmax - cmin)/cmax;
-   else
-      saturation = 0;
-
-   if (saturation == 0)
-      hue = 0;
-   else {
-      var redc   = (cmax - r)/(cmax - cmin);
-      var greenc = (cmax - g)/(cmax - cmin);
-      var bluec  = (cmax - b)/(cmax - cmin);
-
-      if (r == cmax)
-         hue = bluec - greenc;
-      else if (g == cmax)
-         hue = 2.0 + redc - bluec;
-      else
-         hue = 4.0 + greenc - redc;
-
-      hue = hue / 6.0;
-      if (hue < 0)
-         hue = hue + 1.0;
-   }
-
-   return { h : hue, s : saturation, b : brightness };
-};
-
-/**
- * Returns a new XML document object
- */
-Rico.createXmlDocument = function() {
-  if (document.implementation && document.implementation.createDocument) {
-    var doc = document.implementation.createDocument("", "", null);
-    // some older versions of Moz did not support the readyState property
-    // and the onreadystate event so we patch it!
-    if (doc.readyState == null) {
-      doc.readyState = 1;
-      doc.addEventListener("load", function () {
-        doc.readyState = 4;
-        if (typeof doc.onreadystatechange == "function") {
-          doc.onreadystatechange();
-        }
-      }, false);
-    }
-    return doc;
-  }
-
-  if (window.ActiveXObject)
-      return Rico.tryFunctions(
-        function() { return new ActiveXObject('MSXML2.DomDocument');   },
-        function() { return new ActiveXObject('Microsoft.DomDocument');},
-        function() { return new ActiveXObject('MSXML.DomDocument');    },
-        function() { return new ActiveXObject('MSXML3.DomDocument');   }
-      ) || false;
-  return null;
-}
index 239684f..9118ebc 100644 (file)
@@ -18,8 +18,6 @@
  */
 var Rico = {
   Version: '3.0b2',
-  loadRequested: 1,
-  loadComplete: 2,
   theme: {},
 
   init : function() {
@@ -27,105 +25,66 @@ var Rico = {
       document.execCommand("BackgroundImageCache", false, true);
     } catch(err) {}
     if (typeof Rico_CONFIG == 'object') {
-      this.setPaths(Rico_CONFIG.jsDir,Rico_CONFIG.cssDir,Rico_CONFIG.imgDir);
+      this.setPaths(Rico_CONFIG.jsDir);
+      this.setBackgroundStyles();
       if (Rico_CONFIG.enableLogging) this.enableLogging();
     }
     this.preloadMsgs='';
     this.baseHref= location.protocol + "//" + location.host;
-    this.loadedFiles={};
     this.windowIsLoaded=false;
     this.onLoadCallbacks=[];
     this.onLoad(function() { Rico.log('Pre-load messages:\n'+Rico.preloadMsgs); });
   },
-  
+
   _bindLoadEvent : function() {
     Rico.eventBind(window,"load", Rico.eventHandle(Rico,'windowLoaded'));
   },
 
-  setPaths : function(jsDir,cssDir,imgDir,htmDir) {
+  setPaths : function(jsDir,htmDir) {
     this.jsDir = jsDir;
-    this.cssDir = cssDir;
-    this.imgDir = imgDir;
     this.htmDir = htmDir || jsDir;
   },
 
-  languages : {
-    de: "translations/ricoLocale_de.js",
-    en: "translations/ricoLocale_en.js",
-    es: "translations/ricoLocale_es.js",
-    fr: "translations/ricoLocale_fr.js",
-    it: "translations/ricoLocale_it.js",
-    ja: "translations/ricoLocale_ja.js",
-    ko: "translations/ricoLocale_ko.js",
-    pt: "translations/ricoLocale_pt.js",
-    zh: "translations/ricoLocale_zh.js"
-  },
-
-  languageInclude : function(lang2) {
-    var filename=this.languages[lang2];
-    if (filename) this.include(filename);
-    return !!filename;\r
-  },
-
-  acceptLanguage : function(acceptLang) {
-    var arLang=acceptLang.toLowerCase().split(',');\r
-    for (var i=0; i<arLang.length; i++) {\r
-      var lang2=arLang[i].match(/\w\w/);
-      if (!lang2) continue;\r
-      if (this.languageInclude(lang2)) return true;\r
+  setBackgroundStyles: function() {
+    var el = document.createElement('style');
+    document.getElementsByTagName('head')[0].appendChild(el);
+    if (!window.createPopup) { /* For Safari */
+      el.appendChild(document.createTextNode(''));
+    }
+    var s = document.styleSheets[document.styleSheets.length - 1];
+    this.addCssRule(s,'.rico-icon',Rico_CONFIG.imgIcons,'no-repeat');
+    this.addCssRule(s,'.ricoLG_Resize',Rico_CONFIG.imgResize,'repeat');
+    if (Rico_CONFIG.imgHeading) {
+      this.addCssRule(s,'tr.ricoLG_hdg th, tr.ricoLG_hdg td, table.ricoLiveGrid thead td, table.ricoLiveGrid thead th, .ricoTitle, .Rico_accTitle',Rico_CONFIG.imgHeading,'repeat-x scroll center left');
     }
-    return false;\r
   },
 
-  include : function(filename) {
-    if (this.loadedFiles[filename]) return;
-    this.addPreloadMsg('include: '+filename);
-    var ext = filename.substr(filename.lastIndexOf('.')+1);
-    switch (ext.toLowerCase()) {
-      case 'js':
-        this.loadedFiles[filename]=filename.substring(0,4)=='rico' ? this.loadRequested : this.loadComplete;
-        var el = document.createElement('script');
-        el.type = 'text/javascript';
-        el.src = this.jsDir+filename;
-        document.getElementsByTagName('head')[0].appendChild(el);
-        break;
-      case 'css':
-        var el = document.createElement('link');
-        el.type = 'text/css';
-        el.rel = 'stylesheet';
-        el.href = this.cssDir+filename;
-        this.loadedFiles[filename]=this.loadComplete;
-        document.getElementsByTagName('head')[0].appendChild(el);
-        break;
+  addCssRule: function(sheet,selector,imageUrl,repeat) {
+    if (!imageUrl) return;
+    var rule="background:url('"+imageUrl+"') "+repeat;
+    if (sheet.addRule) {
+      sheet.addRule(selector, rule);
+    } else if (sheet.insertRule) {
+      sheet.insertRule (selector+" { "+rule+" }", 0);
+    } else {
+      alert('unable to add rule: '+rule);
     }
   },
 
-  // called after a script file has finished loading
-  includeLoaded: function(filename) {
-    this.loadedFiles[filename]=this.loadComplete;
-    this.checkIfComplete();
+  languageInclude : function(lang2) {
+    var el = document.createElement('script');
+    el.type = 'text/javascript';
+    el.src = this.jsDir+"translations/ricoLocale_"+lang2+".js";
+    document.getElementsByTagName('head')[0].appendChild(el);
   },
 
   // called by the document onload event
   windowLoaded: function() {
     this.windowIsLoaded=true;
-    this.checkIfComplete();
-  },
-
-  checkIfComplete: function() {
-    var waitingFor=this.windowIsLoaded ? '' : 'window';
-    for(var filename in  this.loadedFiles) {
-      if (this.loadedFiles[filename]==this.loadRequested)
-        waitingFor+=' '+filename;
-    }
-    //window.status='waitingFor: '+waitingFor;
-    this.addPreloadMsg('waitingFor: '+waitingFor);
-    if (waitingFor.length==0) {
-      this.addPreloadMsg('Processing callbacks');
-      while (this.onLoadCallbacks.length > 0) {
-        var callback=this.onLoadCallbacks.shift();
-        if (callback) callback();
-      }
+    this.addPreloadMsg('Processing callbacks');
+    while (this.onLoadCallbacks.length > 0) {
+      var callback=this.onLoadCallbacks.shift();
+      if (callback) callback();
     }
   },
 
@@ -134,7 +93,7 @@ var Rico = {
       this.onLoadCallbacks.unshift(callback);
     else
       this.onLoadCallbacks.push(callback);
-    this.checkIfComplete();
+    if (this.windowIsLoaded) this.windowLoaded();
   },
 
   isKonqueror : navigator.userAgent.toLowerCase().indexOf("konqueror") > -1,
@@ -795,7 +754,7 @@ createXmlDocument : function() {
   if (document.implementation && document.implementation.createDocument) {
     var doc = document.implementation.createDocument("", "", null);
     // some older versions of Moz did not support the readyState property
-    // and the onreadystate event so we patch it! 
+    // and the onreadystate event so we patch it!
     if (doc.readyState == null) {
       doc.readyState = 1;
       doc.addEventListener("load", function () {
index 830ce76..4fd19f6 100644 (file)
@@ -2,9 +2,25 @@ Rico.applyShadow=function(b,a){if(typeof a=="undefined"){a=true
 }if(a){Rico.addClass(b,"ricoShadow")
 }return b
 };
+Rico._OpenPopupList=[];
+Rico._RemoveOpenPopup=function(a){if(a.openIndex>=0&&a.openIndex<Rico._OpenPopupList.length){Rico._OpenPopupList.splice(a.openIndex,1)
+}a.openIndex=-1
+};
+Rico._AddOpenPopup=function(a){a.openIndex=Rico._OpenPopupList.push(a)-1
+};
+Rico._checkEscKey=function(b){if(Rico.eventKey(b)!=27){return true
+}while(Rico._OpenPopupList.length>0){var a=Rico._OpenPopupList.pop();
+if(a&&a.visible()){a.openIndex=-1;
+Rico.eventStop(b);
+a.closeFunc();
+return false
+}}return true
+};
+Rico.eventBind(document,"keyup",Rico.eventHandle(Rico,"_checkEscKey"));
 Rico.Popup=function(b,a){this.initialize(b,a)
 };
-Rico.Popup.prototype={initialize:function(b,a){this.options={hideOnEscape:true,hideOnClick:true,ignoreClicks:false,position:"absolute",shadow:true,zIndex:2,canDrag:false,dragElement:false,closeFunc:false};
+Rico.Popup.prototype={initialize:function(b,a){this.options={hideOnClick:false,ignoreClicks:false,position:"absolute",shadow:true,zIndex:2,canDrag:false,dragElement:false,closeFunc:false};
+this.openIndex=-1;
 if(b){this.setDiv(b,a)
 }},createContainer:function(a){this.setDiv(document.createElement("div"),a);
 if(a&&a.parent){a.parent.appendChild(this.container)
@@ -40,7 +56,6 @@ this.ifr.src="javascript:false;";
 this.content.appendChild(this.ifr)
 }Rico.applyShadow(this.container,this.options.shadow);
 if(this.options.hideOnClick){Rico.eventBind(document,"click",Rico.eventHandle(this,"_docClick"))
-}if(this.options.hideOnEscape){Rico.eventBind(document,"keyup",Rico.eventHandle(this,"_checkKey"))
 }this.dragEnabled=false;
 this.mousedownHandler=Rico.eventHandle(this,"_startDrag");
 this.dragHandler=Rico.eventHandle(this,"_drag");
@@ -60,10 +75,6 @@ this.dragEnabled=false
 },_ignoreClick:function(a){if(a.stopPropagation){a.stopPropagation()
 }else{a.cancelBubble=true
 }return true
-},_checkKey:function(a){if(Rico.eventKey(a)!=27||!this.visible()){return true
-}Rico.eventStop(a);
-this.closeFunc();
-return false
 },_docClick:function(a){this.closeFunc();
 return true
 },move:function(b,a){if(typeof b=="number"){this.container.style.left=b+"px"
@@ -87,14 +98,16 @@ Rico.eventUnbind(document,"mouseup",this.dropHandler)
 if(typeof b=="number"){this.container.style.left=b+"px"
 }if(typeof a=="number"){this.container.style.top=a+"px"
 }if(this.container.id){Rico.log("openPopup "+this.container.id+" at "+b+","+a)
-}},centerPopup:function(){this.openPopup();
+}Rico._AddOpenPopup(this)
+},centerPopup:function(){this.openPopup();
 var d=this.container.offsetWidth;
 var c=this.container.offsetHeight;
 var a=this.container.parentNode.offsetWidth;
 var b=this.container.parentNode.offsetHeight;
 this.move(parseInt(Math.max((a-d)/2,0),10),parseInt(Math.max((b-c)/2,0),10))
 },visible:function(){return Rico.visible(this.container)
-},closePopup:function(){if(!Rico.visible(this.container)){return
+},closePopup:function(){Rico._RemoveOpenPopup(this);
+if(!Rico.visible(this.container)){return
 }if(this.container.id){Rico.log("closePopup "+this.container.id)
 }if(this.dragEnabled){this._endDrag()
 }this.container.style.display="none";
@@ -106,7 +119,7 @@ if(Rico.theme.closeAnchor){Rico.addClass(b,Rico.theme.closeAnchor)
 }var c=b.appendChild(document.createElement("span"));
 c.title=Rico.getPhraseById("close");
 new Rico.HoverSet([b]);
-Rico.addClass(c,Rico.theme.close||"RicoClose");
+Rico.addClass(c,Rico.theme.close||"rico-icon RicoClose");
 Rico.eventBind(b,"click",d);
 return b
 };
@@ -115,7 +128,7 @@ c.className="RicoButtonAnchor";
 Rico.addClass(c,Rico.theme.buttonAnchor||"RicoButtonAnchorNative");
 var d=c.appendChild(document.createElement("span"));
 if(f){d.title=f
-}d.className=Rico.theme[b.toLowerCase()]||"Rico"+b;
+}d.className=Rico.theme[b.toLowerCase()]||"rico-icon Rico"+b;
 Rico.eventBind(c,"click",e,false);
 new Rico.HoverSet([c]);
 return c
@@ -123,9 +136,7 @@ return c
 Rico.clearButton=function(b){var a=document.createElement("span");
 a.title=Rico.getPhraseById("clear");
 a.className="ricoClear";
-Rico.addClass(a,Rico.theme.clear||"ricoClearNative");
-a.style.display="inline-block";
-a.style.cursor="pointer";
+Rico.addClass(a,Rico.theme.clear||"rico-icon ricoClearNative");
 Rico.eventBind(a,"click",b);
 return a
 };
@@ -136,7 +147,6 @@ Rico.extend(this,new Rico.Popup());
 this.titleDiv=document.createElement("div");
 this.options.canDrag=true;
 this.options.dragElement=this.titleDiv;
-this.options.hideOnClick=false;
 this.createContainer(a);
 this.content.appendChild(this.titleDiv);
 b=Rico.$(b);
@@ -168,13 +178,11 @@ this.contentDiv.style.zoom=1
 Rico.Menu=function(a){this.initialize(a)
 };
 Rico.Menu.prototype={initialize:function(a){Rico.extend(this,new Rico.Popup());
-Rico.extend(this.options,{width:"15em",showDisabled:false});
+Rico.extend(this.options,{width:"15em",arrowColor:"b",showDisabled:false,hideOnClick:true});
 if(typeof a=="string"){this.options.width=a
 }else{Rico.extend(this.options,a||{})
 }this.hideFunc=null;
-this.highlightElem=null;
-new Image().src=Rico.imgDir+"left_b.gif";
-new Image().src=Rico.imgDir+"right_b.gif"
+this.highlightElem=null
 },createDiv:function(a){if(this.container){return
 }var b=this;
 var c={closeFunc:function(){b.cancelmenu()
@@ -216,20 +224,24 @@ this.content.appendChild(a)
 },addMenuBreak:function(){var a=document.createElement("div");
 a.className="ricoMenuBreak";
 this.content.appendChild(a)
-},addSubMenuItem:function(b,e,f){var d=this.direction=="rtl"?"left":"right";
-var c=this.addMenuItem(b,null,true,null,f);
+},addSubMenuItem:function(b,e,g){var d=this.direction=="rtl"?"left":"right";
+var c=this.addMenuItem(b,null,true,null,g);
 c.className="ricoSubMenu";
-c.style.backgroundImage="url("+Rico.imgDir+d+"_b.gif)";
-c.style.backgroundRepeat="no-repeat";
-c.style.backgroundPosition=d;
+var f=c.appendChild(document.createElement("div"));
+f.className="rico-icon rico-"+d+"-"+this.options.arrowColor;
+Rico.setStyle(f,{position:"absolute",top:"2px"});
+f.style[d]="0px";
 c.RicoSubmenu=e;
 Rico.eventBind(c,"mouseover",Rico.eventHandle(this,"showSubMenu"));
 Rico.eventBind(c,"mouseout",Rico.eventHandle(this,"subMenuOut"))
 },showSubMenu:function(c){if(this.openSubMenu){this.hideSubMenu()
 }var b=Rico.eventElement(c);
-this.openSubMenu=b.RicoSubmenu;
+if(!b.RicoSubmenu){b=b.parentNode
+}if(!b.RicoSubmenu){return
+}this.openSubMenu=b.RicoSubmenu;
 this.openMenuAnchor=b;
-if(b.className=="ricoSubMenu"){b.className="ricoSubMenuOpen"
+if(Rico.hasClass(b,"ricoSubMenu")){Rico.removeClass(b,"ricoSubMenu");
+Rico.addClass(b,"ricoSubMenuOpen")
 }b.RicoSubmenu.openmenu(parseInt(this.container.style.left)+b.offsetWidth,parseInt(this.container.style.top)+b.offsetTop,b.offsetWidth-2,b.offsetHeight+2,true)
 },subMenuOut:function(d){if(!this.openSubMenu){return
 }Rico.eventStop(d);
@@ -238,7 +250,8 @@ var a=Rico.eventRelatedTarget(d)||d.toElement;
 try{while(a!=null&&a!=this.openSubMenu.div){a=a.parentNode
 }}catch(b){}if(a==this.openSubMenu.div){return
 }this.hideSubMenu()
-},hideSubMenu:function(){if(this.openMenuAnchor){this.openMenuAnchor.className="ricoSubMenu";
+},hideSubMenu:function(){if(this.openMenuAnchor){Rico.removeClass(this.openMenuAnchor,"ricoSubMenuOpen");
+Rico.addClass(this.openMenuAnchor,"ricoSubMenu");
 this.openMenuAnchor=null
 }if(this.openSubMenu){this.openSubMenu.hidemenu();
 this.openSubMenu=null
@@ -846,11 +859,10 @@ this.yearNow=a.getFullYear();
 this.monthSelected=this.monthNow;
 this.yearSelected=this.yearNow;
 this.constructCalendar()
-},createNavArrow:function(a,b){var c;
-c=document.createElement("span");
-c.className=Rico.theme[b+"Arrow"]||"Rico_"+b+"Arrow";
-Rico.eventBind(c,"click",Rico.eventHandle(this,a),false);
-return c
+},createNavArrow:function(a,c){var b=document.createElement("span");
+b.className=Rico.theme[c+"Arrow"]||"rico-icon Rico_"+c+"Arrow";
+Rico.eventBind(b,"click",Rico.eventHandle(this,a),false);
+return b
 },isValidMonth:function(a,b){if(a<this.options.minDate.getFullYear()){return false
 }if(a==this.options.minDate.getFullYear()&&b<this.options.minDate.getMonth()){return false
 }if(a>this.options.maxDate.getFullYear()){return false
@@ -1379,18 +1391,13 @@ this.returnValue(a.value,a.innerHTML)
 Rico.TreeControl=function(c,b,a){this.initialize(c,b,a)
 };
 Rico.TreeControl.prototype={initialize:function(c,b,a){Rico.extend(this,new Rico.Popup());
-Rico.extend(this.options,{ignoreClicks:true,nodeIdDisplay:"none",showCheckBox:false,showFolders:false,showPlusMinus:true,showLines:true,defaultAction:Rico.eventHandle(this,"nodeClick"),height:"300px",width:"300px",leafIcon:Rico.imgDir+"doc.gif"});
+Rico.extend(this.options,{ignoreClicks:true,nodeIdDisplay:"none",showCheckBox:false,showFolders:false,showPlusMinus:true,showLines:true,defaultAction:Rico.eventHandle(this,"nodeClick"),height:"300px",width:"300px",leafIcon:"rico-icon rico-doc"});
 Rico.extend(this.options,a||{});
 this.id=c;
 this.dataSource=b;
 this.close=this.closePopup;
 this.hoverSet=new Rico.HoverSet([])
-},atLoad:function(){var d=["node.gif","nodelast.gif","folderopen.gif","folderclosed.gif"];
-for(var b=0;
-b<d.length;
-b++){var a=new Image();
-a.src=Rico.imgDir+d[b]
-}this.treeDiv=document.createElement("div");
+},atLoad:function(){this.treeDiv=document.createElement("div");
 this.treeDiv.id=this.id;
 this.treeDiv.className="ricoTree";
 if(Rico.theme.treeContent){Rico.addClass(this.treeDiv,Rico.theme.treeContent)
@@ -1402,17 +1409,17 @@ this.content.appendChild(this.treeDiv);
 if(this.options.showCheckBox){this.buttonDiv=document.createElement("div");
 this.buttonDiv.style.width=this.options.width;
 this.buttonDiv.className="ricoTreeButtons";
-if(Rico.getStyle(this.container,"position")=="absolute"){var c=document.createElement("span");
-c.innerHTML=RicoTranslate.getPhraseById("treeSave");
-Rico.setStyle(c,{"float":"left",cursor:"pointer"});
-this.buttonDiv.appendChild(c);
-Rico.eventBind(c,"click",Rico.eventHandle(this,"saveSelection"))
-}var c=document.createElement("span");
-c.innerHTML=RicoTranslate.getPhraseById("treeClear");
-Rico.setStyle(c,{"float":"right",cursor:"pointer"});
-this.buttonDiv.appendChild(c);
+if(Rico.getStyle(this.container,"position")=="absolute"){var a=document.createElement("span");
+a.innerHTML=RicoTranslate.getPhraseById("treeSave");
+Rico.setStyle(a,{"float":"left",cursor:"pointer"});
+this.buttonDiv.appendChild(a);
+Rico.eventBind(a,"click",Rico.eventHandle(this,"saveSelection"))
+}var a=document.createElement("span");
+a.innerHTML=RicoTranslate.getPhraseById("treeClear");
+Rico.setStyle(a,{"float":"right",cursor:"pointer"});
+this.buttonDiv.appendChild(a);
 this.content.appendChild(this.buttonDiv);
-Rico.eventBind(c,"click",Rico.eventHandle(this,"clrCheckBoxEvent"))
+Rico.eventBind(a,"click",Rico.eventHandle(this,"clrCheckBoxEvent"))
 }this.close()
 },setTreeDiv:function(a){this.treeDiv=Rico.$(a);
 this.openPopup=function(){}
@@ -1446,13 +1453,6 @@ k.push(h)
 e<k.length;
 e++){var a=(e<k.length-1)&&(k[e][0]==k[e+1][0]);
 this.addNode(k[e][0],k[e][1],k[e][2],k[e][3],k[e][4],k[e][5],!a)
-}},DisplayImages:function(d,a){var c,b,e;
-for(c=0;
-c<a.length;
-c++){b=document.createElement("img");
-b.src=Rico.imgDir+a[c]+".gif";
-e=d.insertCell(-1);
-e.appendChild(b)
 }},addNode:function(q,l,t,f,u,s,k){var h=Rico.$(this.domID(q,"Parent"));
 var v=Rico.$(this.domID(q,"Children"));
 var a=h?h.TreeLevel+1:0;
@@ -1477,26 +1477,26 @@ r++){g[r]=d.insertCell(-1)
 for(var r=0;
 r<a-2;
 r++){g[r].innerHTML=e[r].innerHTML
-}var w=document.createElement("img");
-w.src=Rico.imgDir+(v.nextSibling&&this.options.showLines?"nodeline":"nodeblank")+".gif";
+}var w=document.createElement("div");
+w.className="rico-icon rico-tree-"+(v.nextSibling&&this.options.showLines?"nodeline":"nodeblank");
 g[a-2].appendChild(w)
 }if(a>0){var c=k&&this.options.showLines?"last":"";
 var o=this.options.showLines?"node":"";
-if(this.options.showPlusMinus&&f){var w=document.createElement("img");
+if(this.options.showPlusMinus&&f){var w=document.createElement("div");
 w.name=l;
 w.style.cursor="pointer";
 Rico.eventBind(w,"click",Rico.eventHandle(this,"clickBranch"));
-w.src=Rico.imgDir+o+"p"+c+".gif";
+w.className="rico-icon rico-tree-"+o+"p"+c;
 d.insertCell(-1).appendChild(w)
-}else{if(this.options.showLines){var w=document.createElement("img");
-w.src=Rico.imgDir+"node"+c+".gif";
+}else{if(this.options.showLines){var w=document.createElement("div");
+w.className="rico-icon rico-tree-node"+c;
 d.insertCell(-1).appendChild(w)
-}}if(this.options.showFolders&&(f||(s&&s!="none"))){var w=document.createElement("img");
-if(!f){w.src=s
+}}if(this.options.showFolders&&(f||(s&&s!="none"))){var w=document.createElement("div");
+if(!f){w.className=s
 }else{w.name=l;
 w.style.cursor="pointer";
 Rico.eventBind(w,"click",Rico.eventHandle(this,"clickBranch"));
-w.src=Rico.imgDir+"folderclosed.gif"
+w.className="rico-icon rico-folderclosed"
 }d.insertCell(-1).appendChild(w)
 }}if(u&&this.options.showCheckBox){var n=document.createElement("input");
 n.type="checkbox";
@@ -1546,15 +1546,15 @@ if(!f||!f.TreeContainer){return
 c[1]="Children";
 var b=Rico.$(c.join("_"));
 Rico.toggle(b);
-if(g.tagName=="IMG"){var d=Rico.visible(b);
-if(g.src.match(/node(p|m)(last)?\.gif$/)){g.src=g.src.replace(/nodep|nodem/,"node"+(d?"m":"p"))
-}else{if(g.src.match(/folder(open|closed)\.gif$/)){g.src=g.src.replace(/folder(open|closed)/,"folder"+(d?"open":"closed"))
-}else{if(g.src.match(/\b(m|p)\.gif$/)){g.src=g.src.replace(/(p|m)\.gif/,d?"m.gif":"p.gif")
+if(g.tagName=="DIV"){var d=Rico.visible(b);
+if(g.className.match(/node(p|m)(last)?$/)){g.className=g.className.replace(/nodep|nodem/,"node"+(d?"m":"p"))
+}else{if(g.className.match(/folder(open|closed)$/)){g.className=g.className.replace(/folder(open|closed)/,"folder"+(d?"open":"closed"))
+}else{if(g.className.match(/\b(m|p)$/)){g.className=g.className.replace(/(p|m)$/,d?"m":"p")
 }}}}if(!f.TreeFetchedChildren){f.TreeFetchedChildren=1;
 this.loadXMLDoc(g.name)
 }}};
 if(typeof Rico=="undefined"){throw ("GridCommon requires the Rico JavaScript framework")
-}Rico.GridCommon={baseInit:function(){this.options={resizeBackground:"resize.gif",saveColumnInfo:{width:true,filter:false,sort:false},cookiePrefix:"RicoGrid.",allowColResize:true,windowResize:true,click:null,dblclick:null,contextmenu:null,menuEvent:"dblclick",defaultWidth:-1,scrollBarWidth:19,minScrollWidth:100,frozenColumns:0,exportWindow:"height=400,width=500,scrollbars=1,menubar=1,resizable=1",exportStyleList:["background-color","color","text-align","font-weight","font-size","font-family"],exportImgTags:false,exportFormFields:true,FilterLocation:null,FilterAllToken:"___ALL___",columnSpecs:[]};
+}Rico.GridCommon={baseInit:function(){this.options={saveColumnInfo:{width:true,filter:false,sort:false},cookiePrefix:"RicoGrid.",allowColResize:true,windowResize:true,click:null,dblclick:null,contextmenu:null,menuEvent:"dblclick",defaultWidth:-1,scrollBarWidth:19,minScrollWidth:100,frozenColumns:0,exportWindow:"height=400,width=500,scrollbars=1,menubar=1,resizable=1",exportStyleList:["background-color","color","text-align","font-weight","font-size","font-family"],exportImgTags:false,exportFormFields:true,FilterLocation:null,FilterAllToken:"___ALL___",columnSpecs:[]};
 this.colWidths=[];
 this.hdrCells=[];
 this.headerColCnt=0;
@@ -1984,14 +1984,11 @@ this.setColWidth(a);
 if(!this.visible){this.setDisplay("none")
 }},colClassName:function(){return this.format.ClassName?this.format.ClassName:this.liveGrid.tableId+"_col"+this.index
 },insertResizer:function(){if(this.format.noResize){return
-}var b=document.createElement("div");
-b.className="ricoLG_Resize";
-b.style[this.liveGrid.align[1]]="0px";
-if(this.options.resizeBackground){var a=Rico.imgDir+this.liveGrid.options.resizeBackground;
-if(Rico.isIE&&Rico.ieVersion<8){a=location.protocol+a
-}b.style.backgroundImage="url("+a+")"
-}this.hdrCellDiv.appendChild(b);
-Rico.eventBind(b,"mousedown",this.mouseDownHandler,false)
+}var a=document.createElement("div");
+a.className="ricoLG_Resize";
+a.style[this.liveGrid.align[1]]="0px";
+this.hdrCellDiv.appendChild(a);
+Rico.eventBind(a,"mousedown",this.mouseDownHandler,false)
 },getDisplayName:function(a){var b=a.getElementsByTagName("A");
 if(b.length>0){return b[0].innerHTML
 }else{return Rico.stripTags(a.innerHTML)
@@ -2088,11 +2085,11 @@ this.hideshow(true,false)
 },chooseColumn:function(b){var a=Rico.eventElement(b);
 this.hideshow(a.checked,false)
 },setImage:function(){if(this.currentSort==Rico.ColumnConst.SORT_ASC){this.imgSort.style.display="inline-block";
-this.imgSort.className=Rico.theme.sortAsc||"ricoLG_sortAsc"
+this.imgSort.className=Rico.theme.sortAsc||"rico-icon ricoLG_sortAsc"
 }else{if(this.currentSort==Rico.ColumnConst.SORT_DESC){this.imgSort.style.display="inline-block";
-this.imgSort.className=Rico.theme.sortDesc||"ricoLG_sortDesc"
+this.imgSort.className=Rico.theme.sortDesc||"rico-icon ricoLG_sortDesc"
 }else{this.imgSort.style.display="none"
-}}if(this.filterType==Rico.ColumnConst.USERFILTER){this.imgFilter.style.display="";
+}}if(this.filterType==Rico.ColumnConst.USERFILTER){this.imgFilter.style.display="inline-block";
 this.imgFilter.title=this.getFilterText()
 }else{this.imgFilter.style.display="none"
 }},canHideShow:function(){return this.hideable
@@ -2579,7 +2576,7 @@ this.tableId=e;
 this.buffer=b;
 this.actionId="_action_"+e;
 Rico.setDebugArea(e+"_debugmsgs");
-Rico.extend(this.options,{visibleRows:-3,frozenColumns:0,offset:0,prefetchBuffer:true,minPageRows:2,maxPageRows:50,canSortDefault:true,canFilterDefault:b.options.canFilter,canHideDefault:true,highlightElem:"none",highlightSection:3,highlightMethod:"class",highlightClass:Rico.theme.gridHighlightClass||"ricoLG_selection",maxPrint:5000,headingSort:"link",hdrIconsFirst:true,filterImg:"filtercol.gif"});
+Rico.extend(this.options,{visibleRows:-3,frozenColumns:0,offset:0,prefetchBuffer:true,minPageRows:2,maxPageRows:50,canSortDefault:true,canFilterDefault:b.options.canFilter,canHideDefault:true,highlightElem:"none",highlightSection:3,highlightMethod:"class",highlightClass:Rico.theme.gridHighlightClass||"ricoLG_selection",maxPrint:5000,headingSort:"link",hdrIconsFirst:true});
 var c=this;
 this.options.sortHandler=function(){c.sortHandler()
 };
@@ -2638,8 +2635,6 @@ if(this.buffer.setBufferSize){this.buffer.setBufferSize(this.pageSize)
 }this.scrollTimeout=null;
 this.lastScrollPos=0;
 this.attachMenuEvents();
-new Image().src=Rico.imgDir+this.options.filterImg;
-Rico.log("images preloaded");
 this.setSortUI(this.options.sortCol,this.options.sortDir);
 this.setImages();
 if(this.listInvisible().length==this.columns.length){this.columns[0].showColumn()
@@ -3638,12 +3633,12 @@ Rico.eventBind(b,"click",c);
 break;
 case"hover":Rico.eventBind(this.hdrCellDiv,"click",c);
 break
-}}this.imgFilter=document.createElement("img");
+}}this.imgFilter=document.createElement("span");
 this.imgFilter.style.display="none";
-this.imgFilter.src=Rico.imgDir+this.options.filterImg;
-this.imgFilter.className="ricoLG_HdrIcon";
+this.imgFilter.className="rico-icon ricoLG_filterCol";
 this.imgSort=document.createElement("span");
 this.imgSort.style.display="none";
+this.imgSort.style.verticalAlign="top";
 if(e){this.hdrCellDiv.insertBefore(this.imgSort,this.hdrCellDiv.firstChild);
 this.hdrCellDiv.insertBefore(this.imgFilter,this.hdrCellDiv.firstChild)
 }else{this.hdrCellDiv.appendChild(this.imgFilter);
@@ -4511,8 +4506,7 @@ break
 }}var g="";
 h.className="ricoEditLabel";
 if(b.Help){h.title=b.Help;
-g="&nbsp;<img src='"+Rico.imgDir+"info_icon.gif'>";
-Rico.addClass(h,"ricoHelp")
+g="&nbsp;<span class='rico-icon rico-info'></span>"
 }var d=b.EntryType.length>1&&b.EntryType.charAt(1)=="L"?c.next.displayName:c.displayName;
 h.innerHTML=d+g
 },addSelectNone:function(a){this.addSelectOption(a,this.options.TableSelectNone,Rico.getPhraseById("selectNone"))
@@ -4907,12 +4901,14 @@ a.atLoadComplete=true
 }}},applyTo:function(e,f){var c=this.widgetList[e.format.SelectCtl];
 if(!c){return
 }Rico.log("Rico.EditControls.applyTo: "+e.displayName+" : "+e.format.SelectCtl);
-var d=document.createElement("span");
-var a=document.createElement("img");
-a.style.paddingLeft="4px";
+var a,d=document.createElement("span");
+if(c.imgsrc.indexOf(".")==-1&&c.imgsrc.indexOf("/")==-1){a=document.createElement("span");
+a.className=c.imgsrc
+}else{a=document.createElement("img");
+a.src=c.imgsrc
+}a.style.verticalAlign="top";
+a.style.marginLeft="4px";
 a.style.cursor="pointer";
-a.align="top";
-a.src=c.imgsrc;
 a.id=this.imgId(e.format.FieldName);
 Rico.eventBind(a,"click",Rico.eventHandle(this,"processClick"));
 f.parentNode.appendChild(d);