]> source.dussan.org Git - sonarqube.git/commitdiff
SONAR-2704 Upgrade to Prototype 1.7/Scriptaculous 1.9
authorsimonbrandhof <simon.brandhof@gmail.com>
Mon, 15 Aug 2011 03:11:21 +0000 (05:11 +0200)
committersimonbrandhof <simon.brandhof@gmail.com>
Mon, 15 Aug 2011 03:11:21 +0000 (05:11 +0200)
Yahoo JS libs used in timemachine have also been removed from default HTML header. They are
available only in the timemachine page.

Protovis and its patch for IE have been added.

sonar-server/pom.xml
sonar-server/src/main/webapp/WEB-INF/app/views/layouts/_head.html.erb
sonar-server/src/main/webapp/WEB-INF/app/views/timemachine/index.html.erb
sonar-server/src/main/webapp/javascripts/calendar/calendar-min.js [new file with mode: 0644]
sonar-server/src/main/webapp/javascripts/calendar/calendar.js [deleted file]
sonar-server/src/main/webapp/javascripts/calendar/yahoo-dom-event-min.js [new file with mode: 0644]
sonar-server/src/main/webapp/javascripts/calendar/yahoo-dom-event.js [deleted file]
sonar-server/src/main/webapp/javascripts/prototype.js
sonar-server/src/main/webapp/javascripts/protovis-msie-shim.js [new file with mode: 0644]
sonar-server/src/main/webapp/javascripts/protovis.js [new file with mode: 0755]
sonar-server/src/main/webapp/javascripts/scriptaculous.js

index 9e3db32e600ec6e6aa8095b4b5aab5ffcb861678..db14a8a969dc9dd39156991c8c9de3a9889257ee 100644 (file)
           <aggregations>
             <aggregation>
               <includes>
-                <include>**/yahoo-dom-event-min.js</include>
-                <include>**/calendar-min.js</include>
                 <include>**/application-min.js</include>
                 <include>**/prototype-min.js</include>
                 <include>**/scriptaculous-min.js</include>
                 <include>**/tablekit-min.js</include>
                 <include>**/prototip-min.js</include>
-                <include>**/tooltip-min.js</include>
                 <include>**/dashboard-min.js</include>
-
+                <include>**/protovis-min.js</include>
               </includes>
               <output>${project.build.directory}/${project.build.finalName}/javascripts/sonar.js</output>
             </aggregation>
         <artifactId>maven-war-plugin</artifactId>
         <configuration>
           <packagingExcludes>
-            **/*.log,*.iml,WEB-INF/script/,WEB-INF/test/
+            **/*.log,*.iml,WEB-INF/script/,WEB-INF/test/,javascripts/application*.js,javascripts/prototype*.js,javascripts/scriptaculous*.js,javascripts/tablekit*.js,javascripts/tablekit*.js,javascripts/prototip*.js,javascripts/dashboard*.js,javascripts/protovis*.js
           </packagingExcludes>
           <warSourceExcludes>
             **/*
index 36d9d613da0031ea0335940ab5a23e5b9a3881b5..03ec71338e988666a4d9b20793d48de27d2d3a5d 100644 (file)
 %>
 <title><%= title %></title>
 <% if ENV['RAILS_ENV'] == 'production'
-%><%= stylesheet_link_tag 'sonar', :media => 'all' %><%= javascript_include_tag 'sonar'
-%><% else %>
+%><%= stylesheet_link_tag 'sonar', :media => 'all' -%><%= javascript_include_tag 'sonar'
+-%><% else %>
 <%= stylesheet_link_tag 'yui-2.6.0.css', :media => 'all' %>
 <%= stylesheet_link_tag 'calendar', :media => 'all' %>
 <%= stylesheet_link_tag 'style', :media => 'all' %>
 <%= stylesheet_link_tag 'sonar-colorizer', :media => 'all' %>
 <%= stylesheet_link_tag 'dashboard', :media => 'all' %>
-<%= javascript_include_tag 'calendar/yahoo-dom-event.js' %>
-<%= javascript_include_tag 'calendar/calendar.js' %>
 <%= javascript_include_tag 'application' %>
 <%= javascript_include_tag 'prototype' %>
 <%= javascript_include_tag 'scriptaculous' %>
 <%= javascript_include_tag 'tablekit' %>
 <%= javascript_include_tag 'prototip' %>
 <%= javascript_include_tag 'dashboard' %>
+<%= javascript_include_tag 'protovis' %>
 <% end %>
 <!--[if lte IE 6]>
 <link href="<%= ApplicationController.root_context -%>/ie6/index" media="all" rel="stylesheet" type="text/css" />
 <![endif]-->
+<!--[if lt IE 9]>
+<%= javascript_include_tag 'protovis-msie-shim' -%>
+<![endif]-->
 <link rel="shortcut icon" type="image/x-icon" href="<%= image_path('favicon.ico') -%>" />
 <%
 if @project %>
index a641423752cdde18ff91987b28be0944e61db3bc..5aada6364a2e3fab014dc00f084d5048bdbd32a4 100644 (file)
@@ -1,3 +1,6 @@
+<%= javascript_include_tag 'calendar/yahoo-dom-event-min.js' -%>
+<%= javascript_include_tag 'calendar/calendar-min.js' -%>
+
 <div id="timemachine">
 <style type="text/css">
     #calContainer { display:none; position:absolute; }
diff --git a/sonar-server/src/main/webapp/javascripts/calendar/calendar-min.js b/sonar-server/src/main/webapp/javascripts/calendar/calendar-min.js
new file mode 100644 (file)
index 0000000..0d8406a
--- /dev/null
@@ -0,0 +1,1514 @@
+(function(){YAHOO.util.Config=function(D){if(D){this.init(D)
+}};
+var B=YAHOO.lang,C=YAHOO.util.CustomEvent,A=YAHOO.util.Config;
+A.CONFIG_CHANGED_EVENT="configChanged";
+A.BOOLEAN_TYPE="boolean";
+A.prototype={owner:null,queueInProgress:false,config:null,initialConfig:null,eventQueue:null,configChangedEvent:null,init:function(D){this.owner=D;
+this.configChangedEvent=this.createEvent(A.CONFIG_CHANGED_EVENT);
+this.configChangedEvent.signature=C.LIST;
+this.queueInProgress=false;
+this.config={};
+this.initialConfig={};
+this.eventQueue=[]
+},checkBoolean:function(D){return(typeof D==A.BOOLEAN_TYPE)
+},checkNumber:function(D){return(!isNaN(D))
+},fireEvent:function(D,F){var E=this.config[D];
+if(E&&E.event){E.event.fire(F)
+}},addProperty:function(E,D){E=E.toLowerCase();
+this.config[E]=D;
+D.event=this.createEvent(E,{scope:this.owner});
+D.event.signature=C.LIST;
+D.key=E;
+if(D.handler){D.event.subscribe(D.handler,this.owner)
+}this.setProperty(E,D.value,true);
+if(!D.suppressEvent){this.queueProperty(E,D.value)
+}},getConfig:function(){var D={},F,E;
+for(F in this.config){E=this.config[F];
+if(E&&E.event){D[F]=E.value
+}}return D
+},getProperty:function(D){var E=this.config[D.toLowerCase()];
+if(E&&E.event){return E.value
+}else{return undefined
+}},resetProperty:function(D){D=D.toLowerCase();
+var E=this.config[D];
+if(E&&E.event){if(this.initialConfig[D]&&!B.isUndefined(this.initialConfig[D])){this.setProperty(D,this.initialConfig[D]);
+return true
+}}else{return false
+}},setProperty:function(E,G,D){var F;
+E=E.toLowerCase();
+if(this.queueInProgress&&!D){this.queueProperty(E,G);
+return true
+}else{F=this.config[E];
+if(F&&F.event){if(F.validator&&!F.validator(G)){return false
+}else{F.value=G;
+if(!D){this.fireEvent(E,G);
+this.configChangedEvent.fire([E,G])
+}return true
+}}else{return false
+}}},queueProperty:function(S,P){S=S.toLowerCase();
+var R=this.config[S],K=false,J,G,H,I,O,Q,F,M,N,D,L,T,E;
+if(R&&R.event){if(!B.isUndefined(P)&&R.validator&&!R.validator(P)){return false
+}else{if(!B.isUndefined(P)){R.value=P
+}else{P=R.value
+}K=false;
+J=this.eventQueue.length;
+for(L=0;
+L<J;
+L++){G=this.eventQueue[L];
+if(G){H=G[0];
+I=G[1];
+if(H==S){this.eventQueue[L]=null;
+this.eventQueue.push([S,(!B.isUndefined(P)?P:I)]);
+K=true;
+break
+}}}if(!K&&!B.isUndefined(P)){this.eventQueue.push([S,P])
+}}if(R.supercedes){O=R.supercedes.length;
+for(T=0;
+T<O;
+T++){Q=R.supercedes[T];
+F=this.eventQueue.length;
+for(E=0;
+E<F;
+E++){M=this.eventQueue[E];
+if(M){N=M[0];
+D=M[1];
+if(N==Q.toLowerCase()){this.eventQueue.push([N,D]);
+this.eventQueue[E]=null;
+break
+}}}}}return true
+}else{return false
+}},refireEvent:function(D){D=D.toLowerCase();
+var E=this.config[D];
+if(E&&E.event&&!B.isUndefined(E.value)){if(this.queueInProgress){this.queueProperty(D)
+}else{this.fireEvent(D,E.value)
+}}},applyConfig:function(D,G){var F,E;
+if(G){E={};
+for(F in D){if(B.hasOwnProperty(D,F)){E[F.toLowerCase()]=D[F]
+}}this.initialConfig=E
+}for(F in D){if(B.hasOwnProperty(D,F)){this.queueProperty(F,D[F])
+}}},refresh:function(){var D;
+for(D in this.config){this.refireEvent(D)
+}},fireQueue:function(){var E,H,D,G,F;
+this.queueInProgress=true;
+for(E=0;
+E<this.eventQueue.length;
+E++){H=this.eventQueue[E];
+if(H){D=H[0];
+G=H[1];
+F=this.config[D];
+F.value=G;
+this.fireEvent(D,G)
+}}this.queueInProgress=false;
+this.eventQueue=[]
+},subscribeToConfigEvent:function(E,F,H,D){var G=this.config[E.toLowerCase()];
+if(G&&G.event){if(!A.alreadySubscribed(G.event,F,H)){G.event.subscribe(F,H,D)
+}return true
+}else{return false
+}},unsubscribeFromConfigEvent:function(D,E,G){var F=this.config[D.toLowerCase()];
+if(F&&F.event){return F.event.unsubscribe(E,G)
+}else{return false
+}},toString:function(){var D="Config";
+if(this.owner){D+=" ["+this.owner.toString()+"]"
+}return D
+},outputEventQueue:function(){var D="",G,E,F=this.eventQueue.length;
+for(E=0;
+E<F;
+E++){G=this.eventQueue[E];
+if(G){D+=G[0]+"="+G[1]+", "
+}}return D
+},destroy:function(){var E=this.config,D,F;
+for(D in E){if(B.hasOwnProperty(E,D)){F=E[D];
+F.event.unsubscribeAll();
+F.event=null
+}}this.configChangedEvent.unsubscribeAll();
+this.configChangedEvent=null;
+this.owner=null;
+this.config=null;
+this.initialConfig=null;
+this.eventQueue=null
+}};
+A.alreadySubscribed=function(E,H,I){var F=E.subscribers.length,D,G;
+if(F>0){G=F-1;
+do{D=E.subscribers[G];
+if(D&&D.obj==I&&D.fn==H){return true
+}}while(G--)
+}return false
+};
+YAHOO.lang.augmentProto(A,YAHOO.util.EventProvider)
+}());
+YAHOO.widget.DateMath={DAY:"D",WEEK:"W",YEAR:"Y",MONTH:"M",ONE_DAY_MS:1000*60*60*24,add:function(A,D,C){var F=new Date(A.getTime());
+switch(D){case this.MONTH:var E=A.getMonth()+C;
+var B=0;
+if(E<0){while(E<0){E+=12;
+B-=1
+}}else{if(E>11){while(E>11){E-=12;
+B+=1
+}}}F.setMonth(E);
+F.setFullYear(A.getFullYear()+B);
+break;
+case this.DAY:F.setDate(A.getDate()+C);
+break;
+case this.YEAR:F.setFullYear(A.getFullYear()+C);
+break;
+case this.WEEK:F.setDate(A.getDate()+(C*7));
+break
+}return F
+},subtract:function(A,C,B){return this.add(A,C,(B*-1))
+},before:function(C,B){var A=B.getTime();
+if(C.getTime()<A){return true
+}else{return false
+}},after:function(C,B){var A=B.getTime();
+if(C.getTime()>A){return true
+}else{return false
+}},between:function(B,A,C){if(this.after(B,A)&&this.before(B,C)){return true
+}else{return false
+}},getJan1:function(A){return this.getDate(A,0,1)
+},getDayOffset:function(B,D){var C=this.getJan1(D);
+var A=Math.ceil((B.getTime()-C.getTime())/this.ONE_DAY_MS);
+return A
+},getWeekNumber:function(C,F){C=this.clearTime(C);
+var E=new Date(C.getTime()+(4*this.ONE_DAY_MS)-((C.getDay())*this.ONE_DAY_MS));
+var B=this.getDate(E.getFullYear(),0,1);
+var A=((E.getTime()-B.getTime())/this.ONE_DAY_MS)-1;
+var D=Math.ceil((A)/7);
+return D
+},isYearOverlapWeek:function(A){var C=false;
+var B=this.add(A,this.DAY,6);
+if(B.getFullYear()!=A.getFullYear()){C=true
+}return C
+},isMonthOverlapWeek:function(A){var C=false;
+var B=this.add(A,this.DAY,6);
+if(B.getMonth()!=A.getMonth()){C=true
+}return C
+},findMonthStart:function(A){var B=this.getDate(A.getFullYear(),A.getMonth(),1);
+return B
+},findMonthEnd:function(B){var D=this.findMonthStart(B);
+var C=this.add(D,this.MONTH,1);
+var A=this.subtract(C,this.DAY,1);
+return A
+},clearTime:function(A){A.setHours(12,0,0,0);
+return A
+},getDate:function(D,A,C){var B=null;
+if(YAHOO.lang.isUndefined(C)){C=1
+}if(D>=100){B=new Date(D,A,C)
+}else{B=new Date();
+B.setFullYear(D);
+B.setMonth(A);
+B.setDate(C);
+B.setHours(0,0,0,0)
+}return B
+}};
+YAHOO.widget.Calendar=function(C,A,B){this.init.apply(this,arguments)
+};
+YAHOO.widget.Calendar.IMG_ROOT=null;
+YAHOO.widget.Calendar.DATE="D";
+YAHOO.widget.Calendar.MONTH_DAY="MD";
+YAHOO.widget.Calendar.WEEKDAY="WD";
+YAHOO.widget.Calendar.RANGE="R";
+YAHOO.widget.Calendar.MONTH="M";
+YAHOO.widget.Calendar.DISPLAY_DAYS=42;
+YAHOO.widget.Calendar.STOP_RENDER="S";
+YAHOO.widget.Calendar.SHORT="short";
+YAHOO.widget.Calendar.LONG="long";
+YAHOO.widget.Calendar.MEDIUM="medium";
+YAHOO.widget.Calendar.ONE_CHAR="1char";
+YAHOO.widget.Calendar._DEFAULT_CONFIG={PAGEDATE:{key:"pagedate",value:null},SELECTED:{key:"selected",value:null},TITLE:{key:"title",value:""},CLOSE:{key:"close",value:false},IFRAME:{key:"iframe",value:(YAHOO.env.ua.ie&&YAHOO.env.ua.ie<=6)?true:false},MINDATE:{key:"mindate",value:null},MAXDATE:{key:"maxdate",value:null},MULTI_SELECT:{key:"multi_select",value:false},START_WEEKDAY:{key:"start_weekday",value:0},SHOW_WEEKDAYS:{key:"show_weekdays",value:true},SHOW_WEEK_HEADER:{key:"show_week_header",value:false},SHOW_WEEK_FOOTER:{key:"show_week_footer",value:false},HIDE_BLANK_WEEKS:{key:"hide_blank_weeks",value:false},NAV_ARROW_LEFT:{key:"nav_arrow_left",value:null},NAV_ARROW_RIGHT:{key:"nav_arrow_right",value:null},MONTHS_SHORT:{key:"months_short",value:["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"]},MONTHS_LONG:{key:"months_long",value:["January","February","March","April","May","June","July","August","September","October","November","December"]},WEEKDAYS_1CHAR:{key:"weekdays_1char",value:["S","M","T","W","T","F","S"]},WEEKDAYS_SHORT:{key:"weekdays_short",value:["Su","Mo","Tu","We","Th","Fr","Sa"]},WEEKDAYS_MEDIUM:{key:"weekdays_medium",value:["Sun","Mon","Tue","Wed","Thu","Fri","Sat"]},WEEKDAYS_LONG:{key:"weekdays_long",value:["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"]},LOCALE_MONTHS:{key:"locale_months",value:"long"},LOCALE_WEEKDAYS:{key:"locale_weekdays",value:"short"},DATE_DELIMITER:{key:"date_delimiter",value:","},DATE_FIELD_DELIMITER:{key:"date_field_delimiter",value:"/"},DATE_RANGE_DELIMITER:{key:"date_range_delimiter",value:"-"},MY_MONTH_POSITION:{key:"my_month_position",value:1},MY_YEAR_POSITION:{key:"my_year_position",value:2},MD_MONTH_POSITION:{key:"md_month_position",value:1},MD_DAY_POSITION:{key:"md_day_position",value:2},MDY_MONTH_POSITION:{key:"mdy_month_position",value:1},MDY_DAY_POSITION:{key:"mdy_day_position",value:2},MDY_YEAR_POSITION:{key:"mdy_year_position",value:3},MY_LABEL_MONTH_POSITION:{key:"my_label_month_position",value:1},MY_LABEL_YEAR_POSITION:{key:"my_label_year_position",value:2},MY_LABEL_MONTH_SUFFIX:{key:"my_label_month_suffix",value:" "},MY_LABEL_YEAR_SUFFIX:{key:"my_label_year_suffix",value:""},NAV:{key:"navigator",value:null}};
+YAHOO.widget.Calendar._EVENT_TYPES={BEFORE_SELECT:"beforeSelect",SELECT:"select",BEFORE_DESELECT:"beforeDeselect",DESELECT:"deselect",CHANGE_PAGE:"changePage",BEFORE_RENDER:"beforeRender",RENDER:"render",RESET:"reset",CLEAR:"clear",BEFORE_HIDE:"beforeHide",HIDE:"hide",BEFORE_SHOW:"beforeShow",SHOW:"show",BEFORE_HIDE_NAV:"beforeHideNav",HIDE_NAV:"hideNav",BEFORE_SHOW_NAV:"beforeShowNav",SHOW_NAV:"showNav",BEFORE_RENDER_NAV:"beforeRenderNav",RENDER_NAV:"renderNav"};
+YAHOO.widget.Calendar._STYLES={CSS_ROW_HEADER:"calrowhead",CSS_ROW_FOOTER:"calrowfoot",CSS_CELL:"calcell",CSS_CELL_SELECTOR:"selector",CSS_CELL_SELECTED:"selected",CSS_CELL_SELECTABLE:"selectable",CSS_CELL_RESTRICTED:"restricted",CSS_CELL_TODAY:"today",CSS_CELL_OOM:"oom",CSS_CELL_OOB:"previous",CSS_HEADER:"calheader",CSS_HEADER_TEXT:"calhead",CSS_BODY:"calbody",CSS_WEEKDAY_CELL:"calweekdaycell",CSS_WEEKDAY_ROW:"calweekdayrow",CSS_FOOTER:"calfoot",CSS_CALENDAR:"yui-calendar",CSS_SINGLE:"single",CSS_CONTAINER:"yui-calcontainer",CSS_NAV_LEFT:"calnavleft",CSS_NAV_RIGHT:"calnavright",CSS_NAV:"calnav",CSS_CLOSE:"calclose",CSS_CELL_TOP:"calcelltop",CSS_CELL_LEFT:"calcellleft",CSS_CELL_RIGHT:"calcellright",CSS_CELL_BOTTOM:"calcellbottom",CSS_CELL_HOVER:"calcellhover",CSS_CELL_HIGHLIGHT1:"highlight1",CSS_CELL_HIGHLIGHT2:"highlight2",CSS_CELL_HIGHLIGHT3:"highlight3",CSS_CELL_HIGHLIGHT4:"highlight4"};
+YAHOO.widget.Calendar.prototype={Config:null,parent:null,index:-1,cells:null,cellDates:null,id:null,containerId:null,oDomContainer:null,today:null,renderStack:null,_renderStack:null,oNavigator:null,_selectedDates:null,domEventMap:null,_parseArgs:function(B){var A={id:null,container:null,config:null};
+if(B&&B.length&&B.length>0){switch(B.length){case 1:A.id=null;
+A.container=B[0];
+A.config=null;
+break;
+case 2:if(YAHOO.lang.isObject(B[1])&&!B[1].tagName&&!(B[1] instanceof String)){A.id=null;
+A.container=B[0];
+A.config=B[1]
+}else{A.id=B[0];
+A.container=B[1];
+A.config=null
+}break;
+default:A.id=B[0];
+A.container=B[1];
+A.config=B[2];
+break
+}}else{}return A
+},init:function(D,B,C){var A=this._parseArgs(arguments);
+D=A.id;
+B=A.container;
+C=A.config;
+this.oDomContainer=YAHOO.util.Dom.get(B);
+if(!this.oDomContainer.id){this.oDomContainer.id=YAHOO.util.Dom.generateId()
+}if(!D){D=this.oDomContainer.id+"_t"
+}this.id=D;
+this.containerId=this.oDomContainer.id;
+this.initEvents();
+this.today=new Date();
+YAHOO.widget.DateMath.clearTime(this.today);
+this.cfg=new YAHOO.util.Config(this);
+this.Options={};
+this.Locale={};
+this.initStyles();
+YAHOO.util.Dom.addClass(this.oDomContainer,this.Style.CSS_CONTAINER);
+YAHOO.util.Dom.addClass(this.oDomContainer,this.Style.CSS_SINGLE);
+this.cellDates=[];
+this.cells=[];
+this.renderStack=[];
+this._renderStack=[];
+this.setupConfig();
+if(C){this.cfg.applyConfig(C,true)
+}this.cfg.fireQueue()
+},configIframe:function(C,B,D){var A=B[0];
+if(!this.parent){if(YAHOO.util.Dom.inDocument(this.oDomContainer)){if(A){var E=YAHOO.util.Dom.getStyle(this.oDomContainer,"position");
+if(E=="absolute"||E=="relative"){if(!YAHOO.util.Dom.inDocument(this.iframe)){this.iframe=document.createElement("iframe");
+this.iframe.src="javascript:false;";
+YAHOO.util.Dom.setStyle(this.iframe,"opacity","0");
+if(YAHOO.env.ua.ie&&YAHOO.env.ua.ie<=6){YAHOO.util.Dom.addClass(this.iframe,"fixedsize")
+}this.oDomContainer.insertBefore(this.iframe,this.oDomContainer.firstChild)
+}}}else{if(this.iframe){if(this.iframe.parentNode){this.iframe.parentNode.removeChild(this.iframe)
+}this.iframe=null
+}}}}},configTitle:function(B,A,C){var E=A[0];
+if(E){this.createTitleBar(E)
+}else{var D=this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.CLOSE.key);
+if(!D){this.removeTitleBar()
+}else{this.createTitleBar("&#160;")
+}}},configClose:function(B,A,C){var E=A[0],D=this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.TITLE.key);
+if(E){if(!D){this.createTitleBar("&#160;")
+}this.createCloseButton()
+}else{this.removeCloseButton();
+if(!D){this.removeTitleBar()
+}}},initEvents:function(){var A=YAHOO.widget.Calendar._EVENT_TYPES;
+this.beforeSelectEvent=new YAHOO.util.CustomEvent(A.BEFORE_SELECT);
+this.selectEvent=new YAHOO.util.CustomEvent(A.SELECT);
+this.beforeDeselectEvent=new YAHOO.util.CustomEvent(A.BEFORE_DESELECT);
+this.deselectEvent=new YAHOO.util.CustomEvent(A.DESELECT);
+this.changePageEvent=new YAHOO.util.CustomEvent(A.CHANGE_PAGE);
+this.beforeRenderEvent=new YAHOO.util.CustomEvent(A.BEFORE_RENDER);
+this.renderEvent=new YAHOO.util.CustomEvent(A.RENDER);
+this.resetEvent=new YAHOO.util.CustomEvent(A.RESET);
+this.clearEvent=new YAHOO.util.CustomEvent(A.CLEAR);
+this.beforeShowEvent=new YAHOO.util.CustomEvent(A.BEFORE_SHOW);
+this.showEvent=new YAHOO.util.CustomEvent(A.SHOW);
+this.beforeHideEvent=new YAHOO.util.CustomEvent(A.BEFORE_HIDE);
+this.hideEvent=new YAHOO.util.CustomEvent(A.HIDE);
+this.beforeShowNavEvent=new YAHOO.util.CustomEvent(A.BEFORE_SHOW_NAV);
+this.showNavEvent=new YAHOO.util.CustomEvent(A.SHOW_NAV);
+this.beforeHideNavEvent=new YAHOO.util.CustomEvent(A.BEFORE_HIDE_NAV);
+this.hideNavEvent=new YAHOO.util.CustomEvent(A.HIDE_NAV);
+this.beforeRenderNavEvent=new YAHOO.util.CustomEvent(A.BEFORE_RENDER_NAV);
+this.renderNavEvent=new YAHOO.util.CustomEvent(A.RENDER_NAV);
+this.beforeSelectEvent.subscribe(this.onBeforeSelect,this,true);
+this.selectEvent.subscribe(this.onSelect,this,true);
+this.beforeDeselectEvent.subscribe(this.onBeforeDeselect,this,true);
+this.deselectEvent.subscribe(this.onDeselect,this,true);
+this.changePageEvent.subscribe(this.onChangePage,this,true);
+this.renderEvent.subscribe(this.onRender,this,true);
+this.resetEvent.subscribe(this.onReset,this,true);
+this.clearEvent.subscribe(this.onClear,this,true)
+},doSelectCell:function(G,A){var L,F,I,C;
+var H=YAHOO.util.Event.getTarget(G);
+var B=H.tagName.toLowerCase();
+var E=false;
+while(B!="td"&&!YAHOO.util.Dom.hasClass(H,A.Style.CSS_CELL_SELECTABLE)){if(!E&&B=="a"&&YAHOO.util.Dom.hasClass(H,A.Style.CSS_CELL_SELECTOR)){E=true
+}H=H.parentNode;
+B=H.tagName.toLowerCase();
+if(B=="html"){return 
+}}if(E){YAHOO.util.Event.preventDefault(G)
+}L=H;
+if(YAHOO.util.Dom.hasClass(L,A.Style.CSS_CELL_SELECTABLE)){F=L.id.split("cell")[1];
+I=A.cellDates[F];
+C=YAHOO.widget.DateMath.getDate(I[0],I[1]-1,I[2]);
+var K;
+if(A.Options.MULTI_SELECT){K=L.getElementsByTagName("a")[0];
+if(K){K.blur()
+}var D=A.cellDates[F];
+var J=A._indexOfSelectedFieldArray(D);
+if(J>-1){A.deselectCell(F)
+}else{A.selectCell(F)
+}}else{K=L.getElementsByTagName("a")[0];
+if(K){K.blur()
+}A.selectCell(F)
+}}},doCellMouseOver:function(C,B){var A;
+if(C){A=YAHOO.util.Event.getTarget(C)
+}else{A=this
+}while(A.tagName&&A.tagName.toLowerCase()!="td"){A=A.parentNode;
+if(!A.tagName||A.tagName.toLowerCase()=="html"){return 
+}}if(YAHOO.util.Dom.hasClass(A,B.Style.CSS_CELL_SELECTABLE)){YAHOO.util.Dom.addClass(A,B.Style.CSS_CELL_HOVER)
+}},doCellMouseOut:function(C,B){var A;
+if(C){A=YAHOO.util.Event.getTarget(C)
+}else{A=this
+}while(A.tagName&&A.tagName.toLowerCase()!="td"){A=A.parentNode;
+if(!A.tagName||A.tagName.toLowerCase()=="html"){return 
+}}if(YAHOO.util.Dom.hasClass(A,B.Style.CSS_CELL_SELECTABLE)){YAHOO.util.Dom.removeClass(A,B.Style.CSS_CELL_HOVER)
+}},setupConfig:function(){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+this.cfg.addProperty(A.PAGEDATE.key,{value:new Date(),handler:this.configPageDate});
+this.cfg.addProperty(A.SELECTED.key,{value:[],handler:this.configSelected});
+this.cfg.addProperty(A.TITLE.key,{value:A.TITLE.value,handler:this.configTitle});
+this.cfg.addProperty(A.CLOSE.key,{value:A.CLOSE.value,handler:this.configClose});
+this.cfg.addProperty(A.IFRAME.key,{value:A.IFRAME.value,handler:this.configIframe,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.MINDATE.key,{value:A.MINDATE.value,handler:this.configMinDate});
+this.cfg.addProperty(A.MAXDATE.key,{value:A.MAXDATE.value,handler:this.configMaxDate});
+this.cfg.addProperty(A.MULTI_SELECT.key,{value:A.MULTI_SELECT.value,handler:this.configOptions,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.START_WEEKDAY.key,{value:A.START_WEEKDAY.value,handler:this.configOptions,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.SHOW_WEEKDAYS.key,{value:A.SHOW_WEEKDAYS.value,handler:this.configOptions,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.SHOW_WEEK_HEADER.key,{value:A.SHOW_WEEK_HEADER.value,handler:this.configOptions,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.SHOW_WEEK_FOOTER.key,{value:A.SHOW_WEEK_FOOTER.value,handler:this.configOptions,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.HIDE_BLANK_WEEKS.key,{value:A.HIDE_BLANK_WEEKS.value,handler:this.configOptions,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.NAV_ARROW_LEFT.key,{value:A.NAV_ARROW_LEFT.value,handler:this.configOptions});
+this.cfg.addProperty(A.NAV_ARROW_RIGHT.key,{value:A.NAV_ARROW_RIGHT.value,handler:this.configOptions});
+this.cfg.addProperty(A.MONTHS_SHORT.key,{value:A.MONTHS_SHORT.value,handler:this.configLocale});
+this.cfg.addProperty(A.MONTHS_LONG.key,{value:A.MONTHS_LONG.value,handler:this.configLocale});
+this.cfg.addProperty(A.WEEKDAYS_1CHAR.key,{value:A.WEEKDAYS_1CHAR.value,handler:this.configLocale});
+this.cfg.addProperty(A.WEEKDAYS_SHORT.key,{value:A.WEEKDAYS_SHORT.value,handler:this.configLocale});
+this.cfg.addProperty(A.WEEKDAYS_MEDIUM.key,{value:A.WEEKDAYS_MEDIUM.value,handler:this.configLocale});
+this.cfg.addProperty(A.WEEKDAYS_LONG.key,{value:A.WEEKDAYS_LONG.value,handler:this.configLocale});
+var B=function(){this.cfg.refireEvent(A.LOCALE_MONTHS.key);
+this.cfg.refireEvent(A.LOCALE_WEEKDAYS.key)
+};
+this.cfg.subscribeToConfigEvent(A.START_WEEKDAY.key,B,this,true);
+this.cfg.subscribeToConfigEvent(A.MONTHS_SHORT.key,B,this,true);
+this.cfg.subscribeToConfigEvent(A.MONTHS_LONG.key,B,this,true);
+this.cfg.subscribeToConfigEvent(A.WEEKDAYS_1CHAR.key,B,this,true);
+this.cfg.subscribeToConfigEvent(A.WEEKDAYS_SHORT.key,B,this,true);
+this.cfg.subscribeToConfigEvent(A.WEEKDAYS_MEDIUM.key,B,this,true);
+this.cfg.subscribeToConfigEvent(A.WEEKDAYS_LONG.key,B,this,true);
+this.cfg.addProperty(A.LOCALE_MONTHS.key,{value:A.LOCALE_MONTHS.value,handler:this.configLocaleValues});
+this.cfg.addProperty(A.LOCALE_WEEKDAYS.key,{value:A.LOCALE_WEEKDAYS.value,handler:this.configLocaleValues});
+this.cfg.addProperty(A.DATE_DELIMITER.key,{value:A.DATE_DELIMITER.value,handler:this.configLocale});
+this.cfg.addProperty(A.DATE_FIELD_DELIMITER.key,{value:A.DATE_FIELD_DELIMITER.value,handler:this.configLocale});
+this.cfg.addProperty(A.DATE_RANGE_DELIMITER.key,{value:A.DATE_RANGE_DELIMITER.value,handler:this.configLocale});
+this.cfg.addProperty(A.MY_MONTH_POSITION.key,{value:A.MY_MONTH_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MY_YEAR_POSITION.key,{value:A.MY_YEAR_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MD_MONTH_POSITION.key,{value:A.MD_MONTH_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MD_DAY_POSITION.key,{value:A.MD_DAY_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MDY_MONTH_POSITION.key,{value:A.MDY_MONTH_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MDY_DAY_POSITION.key,{value:A.MDY_DAY_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MDY_YEAR_POSITION.key,{value:A.MDY_YEAR_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MY_LABEL_MONTH_POSITION.key,{value:A.MY_LABEL_MONTH_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MY_LABEL_YEAR_POSITION.key,{value:A.MY_LABEL_YEAR_POSITION.value,handler:this.configLocale,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MY_LABEL_MONTH_SUFFIX.key,{value:A.MY_LABEL_MONTH_SUFFIX.value,handler:this.configLocale});
+this.cfg.addProperty(A.MY_LABEL_YEAR_SUFFIX.key,{value:A.MY_LABEL_YEAR_SUFFIX.value,handler:this.configLocale});
+this.cfg.addProperty(A.NAV.key,{value:A.NAV.value,handler:this.configNavigator})
+},configPageDate:function(B,A,C){this.cfg.setProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key,this._parsePageDate(A[0]),true)
+},configMinDate:function(B,A,C){var D=A[0];
+if(YAHOO.lang.isString(D)){D=this._parseDate(D);
+this.cfg.setProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.MINDATE.key,YAHOO.widget.DateMath.getDate(D[0],(D[1]-1),D[2]))
+}},configMaxDate:function(B,A,C){var D=A[0];
+if(YAHOO.lang.isString(D)){D=this._parseDate(D);
+this.cfg.setProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.MAXDATE.key,YAHOO.widget.DateMath.getDate(D[0],(D[1]-1),D[2]))
+}},configSelected:function(C,A,E){var B=A[0];
+var D=YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
+if(B){if(YAHOO.lang.isString(B)){this.cfg.setProperty(D,this._parseDates(B),true)
+}}if(!this._selectedDates){this._selectedDates=this.cfg.getProperty(D)
+}},configOptions:function(B,A,C){this.Options[B.toUpperCase()]=A[0]
+},configLocale:function(C,B,D){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+this.Locale[C.toUpperCase()]=B[0];
+this.cfg.refireEvent(A.LOCALE_MONTHS.key);
+this.cfg.refireEvent(A.LOCALE_WEEKDAYS.key)
+},configLocaleValues:function(D,C,E){var B=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+D=D.toLowerCase();
+var G=C[0];
+switch(D){case B.LOCALE_MONTHS.key:switch(G){case YAHOO.widget.Calendar.SHORT:this.Locale.LOCALE_MONTHS=this.cfg.getProperty(B.MONTHS_SHORT.key).concat();
+break;
+case YAHOO.widget.Calendar.LONG:this.Locale.LOCALE_MONTHS=this.cfg.getProperty(B.MONTHS_LONG.key).concat();
+break
+}break;
+case B.LOCALE_WEEKDAYS.key:switch(G){case YAHOO.widget.Calendar.ONE_CHAR:this.Locale.LOCALE_WEEKDAYS=this.cfg.getProperty(B.WEEKDAYS_1CHAR.key).concat();
+break;
+case YAHOO.widget.Calendar.SHORT:this.Locale.LOCALE_WEEKDAYS=this.cfg.getProperty(B.WEEKDAYS_SHORT.key).concat();
+break;
+case YAHOO.widget.Calendar.MEDIUM:this.Locale.LOCALE_WEEKDAYS=this.cfg.getProperty(B.WEEKDAYS_MEDIUM.key).concat();
+break;
+case YAHOO.widget.Calendar.LONG:this.Locale.LOCALE_WEEKDAYS=this.cfg.getProperty(B.WEEKDAYS_LONG.key).concat();
+break
+}var F=this.cfg.getProperty(B.START_WEEKDAY.key);
+if(F>0){for(var A=0;
+A<F;
+++A){this.Locale.LOCALE_WEEKDAYS.push(this.Locale.LOCALE_WEEKDAYS.shift())
+}}break
+}},configNavigator:function(C,A,D){var E=A[0];
+if(YAHOO.widget.CalendarNavigator&&(E===true||YAHOO.lang.isObject(E))){if(!this.oNavigator){this.oNavigator=new YAHOO.widget.CalendarNavigator(this);
+function B(){if(!this.pages){this.oNavigator.erase()
+}}this.beforeRenderEvent.subscribe(B,this,true)
+}}else{if(this.oNavigator){this.oNavigator.destroy();
+this.oNavigator=null
+}}},initStyles:function(){var A=YAHOO.widget.Calendar._STYLES;
+this.Style={CSS_ROW_HEADER:A.CSS_ROW_HEADER,CSS_ROW_FOOTER:A.CSS_ROW_FOOTER,CSS_CELL:A.CSS_CELL,CSS_CELL_SELECTOR:A.CSS_CELL_SELECTOR,CSS_CELL_SELECTED:A.CSS_CELL_SELECTED,CSS_CELL_SELECTABLE:A.CSS_CELL_SELECTABLE,CSS_CELL_RESTRICTED:A.CSS_CELL_RESTRICTED,CSS_CELL_TODAY:A.CSS_CELL_TODAY,CSS_CELL_OOM:A.CSS_CELL_OOM,CSS_CELL_OOB:A.CSS_CELL_OOB,CSS_HEADER:A.CSS_HEADER,CSS_HEADER_TEXT:A.CSS_HEADER_TEXT,CSS_BODY:A.CSS_BODY,CSS_WEEKDAY_CELL:A.CSS_WEEKDAY_CELL,CSS_WEEKDAY_ROW:A.CSS_WEEKDAY_ROW,CSS_FOOTER:A.CSS_FOOTER,CSS_CALENDAR:A.CSS_CALENDAR,CSS_SINGLE:A.CSS_SINGLE,CSS_CONTAINER:A.CSS_CONTAINER,CSS_NAV_LEFT:A.CSS_NAV_LEFT,CSS_NAV_RIGHT:A.CSS_NAV_RIGHT,CSS_NAV:A.CSS_NAV,CSS_CLOSE:A.CSS_CLOSE,CSS_CELL_TOP:A.CSS_CELL_TOP,CSS_CELL_LEFT:A.CSS_CELL_LEFT,CSS_CELL_RIGHT:A.CSS_CELL_RIGHT,CSS_CELL_BOTTOM:A.CSS_CELL_BOTTOM,CSS_CELL_HOVER:A.CSS_CELL_HOVER,CSS_CELL_HIGHLIGHT1:A.CSS_CELL_HIGHLIGHT1,CSS_CELL_HIGHLIGHT2:A.CSS_CELL_HIGHLIGHT2,CSS_CELL_HIGHLIGHT3:A.CSS_CELL_HIGHLIGHT3,CSS_CELL_HIGHLIGHT4:A.CSS_CELL_HIGHLIGHT4}
+},buildMonthLabel:function(){var A=this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key);
+var C=this.Locale.LOCALE_MONTHS[A.getMonth()]+this.Locale.MY_LABEL_MONTH_SUFFIX;
+var B=A.getFullYear()+this.Locale.MY_LABEL_YEAR_SUFFIX;
+if(this.Locale.MY_LABEL_MONTH_POSITION==2||this.Locale.MY_LABEL_YEAR_POSITION==1){return B+C
+}else{return C+B
+}},buildDayLabel:function(A){return A.getDate()
+},createTitleBar:function(A){var B=YAHOO.util.Dom.getElementsByClassName(YAHOO.widget.CalendarGroup.CSS_2UPTITLE,"div",this.oDomContainer)[0]||document.createElement("div");
+B.className=YAHOO.widget.CalendarGroup.CSS_2UPTITLE;
+B.innerHTML=A;
+this.oDomContainer.insertBefore(B,this.oDomContainer.firstChild);
+YAHOO.util.Dom.addClass(this.oDomContainer,"withtitle");
+return B
+},removeTitleBar:function(){var A=YAHOO.util.Dom.getElementsByClassName(YAHOO.widget.CalendarGroup.CSS_2UPTITLE,"div",this.oDomContainer)[0]||null;
+if(A){YAHOO.util.Event.purgeElement(A);
+this.oDomContainer.removeChild(A)
+}YAHOO.util.Dom.removeClass(this.oDomContainer,"withtitle")
+},createCloseButton:function(){var D=YAHOO.util.Dom,A=YAHOO.util.Event,C=YAHOO.widget.CalendarGroup.CSS_2UPCLOSE,F="us/my/bn/x_d.gif";
+var E=D.getElementsByClassName("link-close","a",this.oDomContainer)[0];
+if(!E){E=document.createElement("a");
+A.addListener(E,"click",function(H,G){G.hide();
+A.preventDefault(H)
+},this)
+}E.href="#";
+E.className="link-close";
+if(YAHOO.widget.Calendar.IMG_ROOT!==null){var B=D.getElementsByClassName(C,"img",E)[0]||document.createElement("img");
+B.src=YAHOO.widget.Calendar.IMG_ROOT+F;
+B.className=C;
+E.appendChild(B)
+}else{E.innerHTML='<span class="'+C+" "+this.Style.CSS_CLOSE+'"></span>'
+}this.oDomContainer.appendChild(E);
+return E
+},removeCloseButton:function(){var A=YAHOO.util.Dom.getElementsByClassName("link-close","a",this.oDomContainer)[0]||null;
+if(A){YAHOO.util.Event.purgeElement(A);
+this.oDomContainer.removeChild(A)
+}},renderHeader:function(E){var H=7;
+var F="us/tr/callt.gif";
+var G="us/tr/calrt.gif";
+var M=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+if(this.cfg.getProperty(M.SHOW_WEEK_HEADER.key)){H+=1
+}if(this.cfg.getProperty(M.SHOW_WEEK_FOOTER.key)){H+=1
+}E[E.length]="<thead>";
+E[E.length]="<tr>";
+E[E.length]='<th colspan="'+H+'" class="'+this.Style.CSS_HEADER_TEXT+'">';
+E[E.length]='<div class="'+this.Style.CSS_HEADER+'">';
+var K,L=false;
+if(this.parent){if(this.index===0){K=true
+}if(this.index==(this.parent.cfg.getProperty("pages")-1)){L=true
+}}else{K=true;
+L=true
+}if(K){var A=this.cfg.getProperty(M.NAV_ARROW_LEFT.key);
+if(A===null&&YAHOO.widget.Calendar.IMG_ROOT!==null){A=YAHOO.widget.Calendar.IMG_ROOT+F
+}var C=(A===null)?"":' style="background-image:url('+A+')"';
+E[E.length]='<a class="'+this.Style.CSS_NAV_LEFT+'"'+C+" >&#160;</a>"
+}var J=this.buildMonthLabel();
+var B=this.parent||this;
+if(B.cfg.getProperty("navigator")){J='<a class="'+this.Style.CSS_NAV+'" href="#">'+J+"</a>"
+}E[E.length]=J;
+if(L){var D=this.cfg.getProperty(M.NAV_ARROW_RIGHT.key);
+if(D===null&&YAHOO.widget.Calendar.IMG_ROOT!==null){D=YAHOO.widget.Calendar.IMG_ROOT+G
+}var I=(D===null)?"":' style="background-image:url('+D+')"';
+E[E.length]='<a class="'+this.Style.CSS_NAV_RIGHT+'"'+I+" >&#160;</a>"
+}E[E.length]="</div>\n</th>\n</tr>";
+if(this.cfg.getProperty(M.SHOW_WEEKDAYS.key)){E=this.buildWeekdays(E)
+}E[E.length]="</thead>";
+return E
+},buildWeekdays:function(C){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+C[C.length]='<tr class="'+this.Style.CSS_WEEKDAY_ROW+'">';
+if(this.cfg.getProperty(A.SHOW_WEEK_HEADER.key)){C[C.length]="<th>&#160;</th>"
+}for(var B=0;
+B<this.Locale.LOCALE_WEEKDAYS.length;
+++B){C[C.length]='<th class="calweekdaycell">'+this.Locale.LOCALE_WEEKDAYS[B]+"</th>"
+}if(this.cfg.getProperty(A.SHOW_WEEK_FOOTER.key)){C[C.length]="<th>&#160;</th>"
+}C[C.length]="</tr>";
+return C
+},renderBody:function(c,a){var m=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+var AB=this.cfg.getProperty(m.START_WEEKDAY.key);
+this.preMonthDays=c.getDay();
+if(AB>0){this.preMonthDays-=AB
+}if(this.preMonthDays<0){this.preMonthDays+=7
+}this.monthDays=YAHOO.widget.DateMath.findMonthEnd(c).getDate();
+this.postMonthDays=YAHOO.widget.Calendar.DISPLAY_DAYS-this.preMonthDays-this.monthDays;
+c=YAHOO.widget.DateMath.subtract(c,YAHOO.widget.DateMath.DAY,this.preMonthDays);
+var Q,H;
+var G="w";
+var W="_cell";
+var U="wd";
+var k="d";
+var I;
+var h;
+var O=this.today.getFullYear();
+var j=this.today.getMonth();
+var D=this.today.getDate();
+var q=this.cfg.getProperty(m.PAGEDATE.key);
+var C=this.cfg.getProperty(m.HIDE_BLANK_WEEKS.key);
+var Z=this.cfg.getProperty(m.SHOW_WEEK_FOOTER.key);
+var T=this.cfg.getProperty(m.SHOW_WEEK_HEADER.key);
+var M=this.cfg.getProperty(m.MINDATE.key);
+var S=this.cfg.getProperty(m.MAXDATE.key);
+if(M){M=YAHOO.widget.DateMath.clearTime(M)
+}if(S){S=YAHOO.widget.DateMath.clearTime(S)
+}a[a.length]='<tbody class="m'+(q.getMonth()+1)+" "+this.Style.CSS_BODY+'">';
+var z=0;
+var J=document.createElement("div");
+var b=document.createElement("td");
+J.appendChild(b);
+var o=this.parent||this;
+for(var u=0;
+u<6;
+u++){Q=YAHOO.widget.DateMath.getWeekNumber(c,q.getFullYear(),AB);
+H=G+Q;
+if(u!==0&&C===true&&c.getMonth()!=q.getMonth()){break
+}else{a[a.length]='<tr class="'+H+'">';
+if(T){a=this.renderRowHeader(Q,a)
+}for(var AA=0;
+AA<7;
+AA++){I=[];
+this.clearElement(b);
+b.className=this.Style.CSS_CELL;
+b.id=this.id+W+z;
+if(c.getDate()==D&&c.getMonth()==j&&c.getFullYear()==O){I[I.length]=o.renderCellStyleToday
+}var R=[c.getFullYear(),c.getMonth()+1,c.getDate()];
+this.cellDates[this.cellDates.length]=R;
+if(c.getMonth()!=q.getMonth()){I[I.length]=o.renderCellNotThisMonth
+}else{YAHOO.util.Dom.addClass(b,U+c.getDay());
+YAHOO.util.Dom.addClass(b,k+c.getDate());
+for(var t=0;
+t<this.renderStack.length;
+++t){h=null;
+var l=this.renderStack[t];
+var AC=l[0];
+var B;
+var V;
+var F;
+switch(AC){case YAHOO.widget.Calendar.DATE:B=l[1][1];
+V=l[1][2];
+F=l[1][0];
+if(c.getMonth()+1==B&&c.getDate()==V&&c.getFullYear()==F){h=l[2];
+this.renderStack.splice(t,1)
+}break;
+case YAHOO.widget.Calendar.MONTH_DAY:B=l[1][0];
+V=l[1][1];
+if(c.getMonth()+1==B&&c.getDate()==V){h=l[2];
+this.renderStack.splice(t,1)
+}break;
+case YAHOO.widget.Calendar.RANGE:var Y=l[1][0];
+var X=l[1][1];
+var e=Y[1];
+var L=Y[2];
+var P=Y[0];
+var y=YAHOO.widget.DateMath.getDate(P,e-1,L);
+var E=X[1];
+var g=X[2];
+var A=X[0];
+var w=YAHOO.widget.DateMath.getDate(A,E-1,g);
+if(c.getTime()>=y.getTime()&&c.getTime()<=w.getTime()){h=l[2];
+if(c.getTime()==w.getTime()){this.renderStack.splice(t,1)
+}}break;
+case YAHOO.widget.Calendar.WEEKDAY:var K=l[1][0];
+if(c.getDay()+1==K){h=l[2]
+}break;
+case YAHOO.widget.Calendar.MONTH:B=l[1][0];
+if(c.getMonth()+1==B){h=l[2]
+}break
+}if(h){I[I.length]=h
+}}}if(this._indexOfSelectedFieldArray(R)>-1){I[I.length]=o.renderCellStyleSelected
+}if((M&&(c.getTime()<M.getTime()))||(S&&(c.getTime()>S.getTime()))){I[I.length]=o.renderOutOfBoundsDate
+}else{I[I.length]=o.styleCellDefault;
+I[I.length]=o.renderCellDefault
+}for(var n=0;
+n<I.length;
+++n){if(I[n].call(o,c,b)==YAHOO.widget.Calendar.STOP_RENDER){break
+}}c.setTime(c.getTime()+YAHOO.widget.DateMath.ONE_DAY_MS);
+if(z>=0&&z<=6){YAHOO.util.Dom.addClass(b,this.Style.CSS_CELL_TOP)
+}if((z%7)===0){YAHOO.util.Dom.addClass(b,this.Style.CSS_CELL_LEFT)
+}if(((z+1)%7)===0){YAHOO.util.Dom.addClass(b,this.Style.CSS_CELL_RIGHT)
+}var f=this.postMonthDays;
+if(C&&f>=7){var N=Math.floor(f/7);
+for(var v=0;
+v<N;
+++v){f-=7
+}}if(z>=((this.preMonthDays+f+this.monthDays)-7)){YAHOO.util.Dom.addClass(b,this.Style.CSS_CELL_BOTTOM)
+}a[a.length]=J.innerHTML;
+z++
+}if(Z){a=this.renderRowFooter(Q,a)
+}a[a.length]="</tr>"
+}}a[a.length]="</tbody>";
+return a
+},renderFooter:function(A){return A
+},render:function(){this.beforeRenderEvent.fire();
+var A=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+var C=YAHOO.widget.DateMath.findMonthStart(this.cfg.getProperty(A.PAGEDATE.key));
+this.resetRenderers();
+this.cellDates.length=0;
+YAHOO.util.Event.purgeElement(this.oDomContainer,true);
+var B=[];
+B[B.length]='<table cellSpacing="0" class="'+this.Style.CSS_CALENDAR+" y"+C.getFullYear()+'" id="'+this.id+'">';
+B=this.renderHeader(B);
+B=this.renderBody(C,B);
+B=this.renderFooter(B);
+B[B.length]="</table>";
+this.oDomContainer.innerHTML=B.join("\n");
+this.applyListeners();
+this.cells=this.oDomContainer.getElementsByTagName("td");
+this.cfg.refireEvent(A.TITLE.key);
+this.cfg.refireEvent(A.CLOSE.key);
+this.cfg.refireEvent(A.IFRAME.key);
+this.renderEvent.fire()
+},applyListeners:function(){var K=this.oDomContainer;
+var B=this.parent||this;
+var G="a";
+var D="mousedown";
+var H=YAHOO.util.Dom.getElementsByClassName(this.Style.CSS_NAV_LEFT,G,K);
+var C=YAHOO.util.Dom.getElementsByClassName(this.Style.CSS_NAV_RIGHT,G,K);
+if(H&&H.length>0){this.linkLeft=H[0];
+YAHOO.util.Event.addListener(this.linkLeft,D,B.previousMonth,B,true)
+}if(C&&C.length>0){this.linkRight=C[0];
+YAHOO.util.Event.addListener(this.linkRight,D,B.nextMonth,B,true)
+}if(B.cfg.getProperty("navigator")!==null){this.applyNavListeners()
+}if(this.domEventMap){var E,A;
+for(var M in this.domEventMap){if(YAHOO.lang.hasOwnProperty(this.domEventMap,M)){var I=this.domEventMap[M];
+if(!(I instanceof Array)){I=[I]
+}for(var F=0;
+F<I.length;
+F++){var L=I[F];
+A=YAHOO.util.Dom.getElementsByClassName(M,L.tag,this.oDomContainer);
+for(var J=0;
+J<A.length;
+J++){E=A[J];
+YAHOO.util.Event.addListener(E,L.event,L.handler,L.scope,L.correct)
+}}}}}YAHOO.util.Event.addListener(this.oDomContainer,"click",this.doSelectCell,this);
+YAHOO.util.Event.addListener(this.oDomContainer,"mouseover",this.doCellMouseOver,this);
+YAHOO.util.Event.addListener(this.oDomContainer,"mouseout",this.doCellMouseOut,this)
+},applyNavListeners:function(){var D=YAHOO.util.Event;
+var C=this.parent||this;
+var F=this;
+var B=YAHOO.util.Dom.getElementsByClassName(this.Style.CSS_NAV,"a",this.oDomContainer);
+if(B.length>0){function A(J,I){var H=D.getTarget(J);
+if(this===H||YAHOO.util.Dom.isAncestor(this,H)){D.preventDefault(J)
+}var E=C.oNavigator;
+if(E){var G=F.cfg.getProperty("pagedate");
+E.setYear(G.getFullYear());
+E.setMonth(G.getMonth());
+E.show()
+}}D.addListener(B,"click",A)
+}},getDateByCellId:function(B){var A=this.getDateFieldsByCellId(B);
+return YAHOO.widget.DateMath.getDate(A[0],A[1]-1,A[2])
+},getDateFieldsByCellId:function(A){A=A.toLowerCase().split("_cell")[1];
+A=parseInt(A,10);
+return this.cellDates[A]
+},getCellIndex:function(C){var B=-1;
+if(C){var A=C.getMonth(),H=C.getFullYear(),G=C.getDate(),E=this.cellDates;
+for(var D=0;
+D<E.length;
+++D){var F=E[D];
+if(F[0]===H&&F[1]===A+1&&F[2]===G){B=D;
+break
+}}}return B
+},renderOutOfBoundsDate:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_OOB);
+A.innerHTML=B.getDate();
+return YAHOO.widget.Calendar.STOP_RENDER
+},renderRowHeader:function(B,A){A[A.length]='<th class="calrowhead">'+B+"</th>";
+return A
+},renderRowFooter:function(B,A){A[A.length]='<th class="calrowfoot">'+B+"</th>";
+return A
+},renderCellDefault:function(B,A){A.innerHTML='<a href="#" class="'+this.Style.CSS_CELL_SELECTOR+'">'+this.buildDayLabel(B)+"</a>"
+},styleCellDefault:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_SELECTABLE)
+},renderCellStyleHighlight1:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_HIGHLIGHT1)
+},renderCellStyleHighlight2:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_HIGHLIGHT2)
+},renderCellStyleHighlight3:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_HIGHLIGHT3)
+},renderCellStyleHighlight4:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_HIGHLIGHT4)
+},renderCellStyleToday:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_TODAY)
+},renderCellStyleSelected:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_SELECTED)
+},renderCellNotThisMonth:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_OOM);
+A.innerHTML=B.getDate();
+return YAHOO.widget.Calendar.STOP_RENDER
+},renderBodyCellRestricted:function(B,A){YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL);
+YAHOO.util.Dom.addClass(A,this.Style.CSS_CELL_RESTRICTED);
+A.innerHTML=B.getDate();
+return YAHOO.widget.Calendar.STOP_RENDER
+},addMonths:function(B){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
+this.cfg.setProperty(A,YAHOO.widget.DateMath.add(this.cfg.getProperty(A),YAHOO.widget.DateMath.MONTH,B));
+this.resetRenderers();
+this.changePageEvent.fire()
+},subtractMonths:function(B){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
+this.cfg.setProperty(A,YAHOO.widget.DateMath.subtract(this.cfg.getProperty(A),YAHOO.widget.DateMath.MONTH,B));
+this.resetRenderers();
+this.changePageEvent.fire()
+},addYears:function(B){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
+this.cfg.setProperty(A,YAHOO.widget.DateMath.add(this.cfg.getProperty(A),YAHOO.widget.DateMath.YEAR,B));
+this.resetRenderers();
+this.changePageEvent.fire()
+},subtractYears:function(B){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
+this.cfg.setProperty(A,YAHOO.widget.DateMath.subtract(this.cfg.getProperty(A),YAHOO.widget.DateMath.YEAR,B));
+this.resetRenderers();
+this.changePageEvent.fire()
+},nextMonth:function(){this.addMonths(1)
+},previousMonth:function(){this.subtractMonths(1)
+},nextYear:function(){this.addYears(1)
+},previousYear:function(){this.subtractYears(1)
+},reset:function(){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+this.cfg.resetProperty(A.SELECTED.key);
+this.cfg.resetProperty(A.PAGEDATE.key);
+this.resetEvent.fire()
+},clear:function(){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+this.cfg.setProperty(A.SELECTED.key,[]);
+this.cfg.setProperty(A.PAGEDATE.key,new Date(this.today.getTime()));
+this.clearEvent.fire()
+},select:function(C){var F=this._toFieldArray(C);
+var B=[];
+var E=[];
+var G=YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
+for(var A=0;
+A<F.length;
+++A){var D=F[A];
+if(!this.isDateOOB(this._toDate(D))){if(B.length===0){this.beforeSelectEvent.fire();
+E=this.cfg.getProperty(G)
+}B.push(D);
+if(this._indexOfSelectedFieldArray(D)==-1){E[E.length]=D
+}}}if(B.length>0){if(this.parent){this.parent.cfg.setProperty(G,E)
+}else{this.cfg.setProperty(G,E)
+}this.selectEvent.fire(B)
+}return this.getSelectedDates()
+},selectCell:function(D){var B=this.cells[D];
+var H=this.cellDates[D];
+var G=this._toDate(H);
+var C=YAHOO.util.Dom.hasClass(B,this.Style.CSS_CELL_SELECTABLE);
+if(C){this.beforeSelectEvent.fire();
+var F=YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
+var E=this.cfg.getProperty(F);
+var A=H.concat();
+if(this._indexOfSelectedFieldArray(A)==-1){E[E.length]=A
+}if(this.parent){this.parent.cfg.setProperty(F,E)
+}else{this.cfg.setProperty(F,E)
+}this.renderCellStyleSelected(G,B);
+this.selectEvent.fire([A]);
+this.doCellMouseOut.call(B,null,this)
+}return this.getSelectedDates()
+},deselect:function(E){var A=this._toFieldArray(E);
+var D=[];
+var G=[];
+var H=YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
+for(var B=0;
+B<A.length;
+++B){var F=A[B];
+if(!this.isDateOOB(this._toDate(F))){if(D.length===0){this.beforeDeselectEvent.fire();
+G=this.cfg.getProperty(H)
+}D.push(F);
+var C=this._indexOfSelectedFieldArray(F);
+if(C!=-1){G.splice(C,1)
+}}}if(D.length>0){if(this.parent){this.parent.cfg.setProperty(H,G)
+}else{this.cfg.setProperty(H,G)
+}this.deselectEvent.fire(D)
+}return this.getSelectedDates()
+},deselectCell:function(E){var H=this.cells[E];
+var B=this.cellDates[E];
+var F=this._indexOfSelectedFieldArray(B);
+var G=YAHOO.util.Dom.hasClass(H,this.Style.CSS_CELL_SELECTABLE);
+if(G){this.beforeDeselectEvent.fire();
+var I=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+var D=this.cfg.getProperty(I.SELECTED.key);
+var C=this._toDate(B);
+var A=B.concat();
+if(F>-1){if(this.cfg.getProperty(I.PAGEDATE.key).getMonth()==C.getMonth()&&this.cfg.getProperty(I.PAGEDATE.key).getFullYear()==C.getFullYear()){YAHOO.util.Dom.removeClass(H,this.Style.CSS_CELL_SELECTED)
+}D.splice(F,1)
+}if(this.parent){this.parent.cfg.setProperty(I.SELECTED.key,D)
+}else{this.cfg.setProperty(I.SELECTED.key,D)
+}this.deselectEvent.fire(A)
+}return this.getSelectedDates()
+},deselectAll:function(){this.beforeDeselectEvent.fire();
+var D=YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
+var A=this.cfg.getProperty(D);
+var B=A.length;
+var C=A.concat();
+if(this.parent){this.parent.cfg.setProperty(D,[])
+}else{this.cfg.setProperty(D,[])
+}if(B>0){this.deselectEvent.fire(C)
+}return this.getSelectedDates()
+},_toFieldArray:function(B){var A=[];
+if(B instanceof Date){A=[[B.getFullYear(),B.getMonth()+1,B.getDate()]]
+}else{if(YAHOO.lang.isString(B)){A=this._parseDates(B)
+}else{if(YAHOO.lang.isArray(B)){for(var C=0;
+C<B.length;
+++C){var D=B[C];
+A[A.length]=[D.getFullYear(),D.getMonth()+1,D.getDate()]
+}}}}return A
+},toDate:function(A){return this._toDate(A)
+},_toDate:function(A){if(A instanceof Date){return A
+}else{return YAHOO.widget.DateMath.getDate(A[0],A[1]-1,A[2])
+}},_fieldArraysAreEqual:function(C,B){var A=false;
+if(C[0]==B[0]&&C[1]==B[1]&&C[2]==B[2]){A=true
+}return A
+},_indexOfSelectedFieldArray:function(E){var D=-1;
+var A=this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key);
+for(var C=0;
+C<A.length;
+++C){var B=A[C];
+if(E[0]==B[0]&&E[1]==B[1]&&E[2]==B[2]){D=C;
+break
+}}return D
+},isDateOOM:function(A){return(A.getMonth()!=this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key).getMonth())
+},isDateOOB:function(D){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+var E=this.cfg.getProperty(A.MINDATE.key);
+var F=this.cfg.getProperty(A.MAXDATE.key);
+var C=YAHOO.widget.DateMath;
+if(E){E=C.clearTime(E)
+}if(F){F=C.clearTime(F)
+}var B=new Date(D.getTime());
+B=C.clearTime(B);
+return((E&&B.getTime()<E.getTime())||(F&&B.getTime()>F.getTime()))
+},_parsePageDate:function(B){var E;
+var A=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+if(B){if(B instanceof Date){E=YAHOO.widget.DateMath.findMonthStart(B)
+}else{var F,D,C;
+C=B.split(this.cfg.getProperty(A.DATE_FIELD_DELIMITER.key));
+F=parseInt(C[this.cfg.getProperty(A.MY_MONTH_POSITION.key)-1],10)-1;
+D=parseInt(C[this.cfg.getProperty(A.MY_YEAR_POSITION.key)-1],10);
+E=YAHOO.widget.DateMath.getDate(D,F,1)
+}}else{E=YAHOO.widget.DateMath.getDate(this.today.getFullYear(),this.today.getMonth(),1)
+}return E
+},onBeforeSelect:function(){if(this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.MULTI_SELECT.key)===false){if(this.parent){this.parent.callChildFunction("clearAllBodyCellStyles",this.Style.CSS_CELL_SELECTED);
+this.parent.deselectAll()
+}else{this.clearAllBodyCellStyles(this.Style.CSS_CELL_SELECTED);
+this.deselectAll()
+}}},onSelect:function(A){},onBeforeDeselect:function(){},onDeselect:function(A){},onChangePage:function(){this.render()
+},onRender:function(){},onReset:function(){this.render()
+},onClear:function(){this.render()
+},validate:function(){return true
+},_parseDate:function(C){var D=C.split(this.Locale.DATE_FIELD_DELIMITER);
+var A;
+if(D.length==2){A=[D[this.Locale.MD_MONTH_POSITION-1],D[this.Locale.MD_DAY_POSITION-1]];
+A.type=YAHOO.widget.Calendar.MONTH_DAY
+}else{A=[D[this.Locale.MDY_YEAR_POSITION-1],D[this.Locale.MDY_MONTH_POSITION-1],D[this.Locale.MDY_DAY_POSITION-1]];
+A.type=YAHOO.widget.Calendar.DATE
+}for(var B=0;
+B<A.length;
+B++){A[B]=parseInt(A[B],10)
+}return A
+},_parseDates:function(B){var I=[];
+var H=B.split(this.Locale.DATE_DELIMITER);
+for(var G=0;
+G<H.length;
+++G){var F=H[G];
+if(F.indexOf(this.Locale.DATE_RANGE_DELIMITER)!=-1){var A=F.split(this.Locale.DATE_RANGE_DELIMITER);
+var E=this._parseDate(A[0]);
+var J=this._parseDate(A[1]);
+var D=this._parseRange(E,J);
+I=I.concat(D)
+}else{var C=this._parseDate(F);
+I.push(C)
+}}return I
+},_parseRange:function(A,E){var B=YAHOO.widget.DateMath.add(YAHOO.widget.DateMath.getDate(A[0],A[1]-1,A[2]),YAHOO.widget.DateMath.DAY,1);
+var D=YAHOO.widget.DateMath.getDate(E[0],E[1]-1,E[2]);
+var C=[];
+C.push(A);
+while(B.getTime()<=D.getTime()){C.push([B.getFullYear(),B.getMonth()+1,B.getDate()]);
+B=YAHOO.widget.DateMath.add(B,YAHOO.widget.DateMath.DAY,1)
+}return C
+},resetRenderers:function(){this.renderStack=this._renderStack.concat()
+},removeRenderers:function(){this._renderStack=[];
+this.renderStack=[]
+},clearElement:function(A){A.innerHTML="&#160;";
+A.className=""
+},addRenderer:function(A,B){var D=this._parseDates(A);
+for(var C=0;
+C<D.length;
+++C){var E=D[C];
+if(E.length==2){if(E[0] instanceof Array){this._addRenderer(YAHOO.widget.Calendar.RANGE,E,B)
+}else{this._addRenderer(YAHOO.widget.Calendar.MONTH_DAY,E,B)
+}}else{if(E.length==3){this._addRenderer(YAHOO.widget.Calendar.DATE,E,B)
+}}}},_addRenderer:function(B,C,A){var D=[B,C,A];
+this.renderStack.unshift(D);
+this._renderStack=this.renderStack.concat()
+},addMonthRenderer:function(B,A){this._addRenderer(YAHOO.widget.Calendar.MONTH,[B],A)
+},addWeekdayRenderer:function(B,A){this._addRenderer(YAHOO.widget.Calendar.WEEKDAY,[B],A)
+},clearAllBodyCellStyles:function(A){for(var B=0;
+B<this.cells.length;
+++B){YAHOO.util.Dom.removeClass(this.cells[B],A)
+}},setMonth:function(C){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
+var B=this.cfg.getProperty(A);
+B.setMonth(parseInt(C,10));
+this.cfg.setProperty(A,B)
+},setYear:function(B){var A=YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
+var C=this.cfg.getProperty(A);
+C.setFullYear(parseInt(B,10));
+this.cfg.setProperty(A,C)
+},getSelectedDates:function(){var C=[];
+var B=this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key);
+for(var E=0;
+E<B.length;
+++E){var D=B[E];
+var A=YAHOO.widget.DateMath.getDate(D[0],D[1]-1,D[2]);
+C.push(A)
+}C.sort(function(G,F){return G-F
+});
+return C
+},hide:function(){if(this.beforeHideEvent.fire()){this.oDomContainer.style.display="none";
+this.hideEvent.fire()
+}},show:function(){if(this.beforeShowEvent.fire()){this.oDomContainer.style.display="block";
+this.showEvent.fire()
+}},browser:(function(){var A=navigator.userAgent.toLowerCase();
+if(A.indexOf("opera")!=-1){return"opera"
+}else{if(A.indexOf("msie 7")!=-1){return"ie7"
+}else{if(A.indexOf("msie")!=-1){return"ie"
+}else{if(A.indexOf("safari")!=-1){return"safari"
+}else{if(A.indexOf("gecko")!=-1){return"gecko"
+}else{return false
+}}}}}})(),toString:function(){return"Calendar "+this.id
+}};
+YAHOO.widget.Calendar_Core=YAHOO.widget.Calendar;
+YAHOO.widget.Cal_Core=YAHOO.widget.Calendar;
+YAHOO.widget.CalendarGroup=function(C,A,B){if(arguments.length>0){this.init.apply(this,arguments)
+}};
+YAHOO.widget.CalendarGroup.prototype={init:function(D,B,C){var A=this._parseArgs(arguments);
+D=A.id;
+B=A.container;
+C=A.config;
+this.oDomContainer=YAHOO.util.Dom.get(B);
+if(!this.oDomContainer.id){this.oDomContainer.id=YAHOO.util.Dom.generateId()
+}if(!D){D=this.oDomContainer.id+"_t"
+}this.id=D;
+this.containerId=this.oDomContainer.id;
+this.initEvents();
+this.initStyles();
+this.pages=[];
+YAHOO.util.Dom.addClass(this.oDomContainer,YAHOO.widget.CalendarGroup.CSS_CONTAINER);
+YAHOO.util.Dom.addClass(this.oDomContainer,YAHOO.widget.CalendarGroup.CSS_MULTI_UP);
+this.cfg=new YAHOO.util.Config(this);
+this.Options={};
+this.Locale={};
+this.setupConfig();
+if(C){this.cfg.applyConfig(C,true)
+}this.cfg.fireQueue();
+if(YAHOO.env.ua.opera){this.renderEvent.subscribe(this._fixWidth,this,true);
+this.showEvent.subscribe(this._fixWidth,this,true)
+}},setupConfig:function(){var A=YAHOO.widget.CalendarGroup._DEFAULT_CONFIG;
+this.cfg.addProperty(A.PAGES.key,{value:A.PAGES.value,validator:this.cfg.checkNumber,handler:this.configPages});
+this.cfg.addProperty(A.PAGEDATE.key,{value:new Date(),handler:this.configPageDate});
+this.cfg.addProperty(A.SELECTED.key,{value:[],handler:this.configSelected});
+this.cfg.addProperty(A.TITLE.key,{value:A.TITLE.value,handler:this.configTitle});
+this.cfg.addProperty(A.CLOSE.key,{value:A.CLOSE.value,handler:this.configClose});
+this.cfg.addProperty(A.IFRAME.key,{value:A.IFRAME.value,handler:this.configIframe,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.MINDATE.key,{value:A.MINDATE.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.MAXDATE.key,{value:A.MAXDATE.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.MULTI_SELECT.key,{value:A.MULTI_SELECT.value,handler:this.delegateConfig,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.START_WEEKDAY.key,{value:A.START_WEEKDAY.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.SHOW_WEEKDAYS.key,{value:A.SHOW_WEEKDAYS.value,handler:this.delegateConfig,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.SHOW_WEEK_HEADER.key,{value:A.SHOW_WEEK_HEADER.value,handler:this.delegateConfig,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.SHOW_WEEK_FOOTER.key,{value:A.SHOW_WEEK_FOOTER.value,handler:this.delegateConfig,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.HIDE_BLANK_WEEKS.key,{value:A.HIDE_BLANK_WEEKS.value,handler:this.delegateConfig,validator:this.cfg.checkBoolean});
+this.cfg.addProperty(A.NAV_ARROW_LEFT.key,{value:A.NAV_ARROW_LEFT.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.NAV_ARROW_RIGHT.key,{value:A.NAV_ARROW_RIGHT.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.MONTHS_SHORT.key,{value:A.MONTHS_SHORT.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.MONTHS_LONG.key,{value:A.MONTHS_LONG.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.WEEKDAYS_1CHAR.key,{value:A.WEEKDAYS_1CHAR.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.WEEKDAYS_SHORT.key,{value:A.WEEKDAYS_SHORT.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.WEEKDAYS_MEDIUM.key,{value:A.WEEKDAYS_MEDIUM.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.WEEKDAYS_LONG.key,{value:A.WEEKDAYS_LONG.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.LOCALE_MONTHS.key,{value:A.LOCALE_MONTHS.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.LOCALE_WEEKDAYS.key,{value:A.LOCALE_WEEKDAYS.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.DATE_DELIMITER.key,{value:A.DATE_DELIMITER.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.DATE_FIELD_DELIMITER.key,{value:A.DATE_FIELD_DELIMITER.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.DATE_RANGE_DELIMITER.key,{value:A.DATE_RANGE_DELIMITER.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.MY_MONTH_POSITION.key,{value:A.MY_MONTH_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MY_YEAR_POSITION.key,{value:A.MY_YEAR_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MD_MONTH_POSITION.key,{value:A.MD_MONTH_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MD_DAY_POSITION.key,{value:A.MD_DAY_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MDY_MONTH_POSITION.key,{value:A.MDY_MONTH_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MDY_DAY_POSITION.key,{value:A.MDY_DAY_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MDY_YEAR_POSITION.key,{value:A.MDY_YEAR_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MY_LABEL_MONTH_POSITION.key,{value:A.MY_LABEL_MONTH_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MY_LABEL_YEAR_POSITION.key,{value:A.MY_LABEL_YEAR_POSITION.value,handler:this.delegateConfig,validator:this.cfg.checkNumber});
+this.cfg.addProperty(A.MY_LABEL_MONTH_SUFFIX.key,{value:A.MY_LABEL_MONTH_SUFFIX.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.MY_LABEL_YEAR_SUFFIX.key,{value:A.MY_LABEL_YEAR_SUFFIX.value,handler:this.delegateConfig});
+this.cfg.addProperty(A.NAV.key,{value:A.NAV.value,handler:this.configNavigator})
+},initEvents:function(){var C=this;
+var E="Event";
+var B=function(G,J,F){for(var I=0;
+I<C.pages.length;
+++I){var H=C.pages[I];
+H[this.type+E].subscribe(G,J,F)
+}};
+var A=function(F,I){for(var H=0;
+H<C.pages.length;
+++H){var G=C.pages[H];
+G[this.type+E].unsubscribe(F,I)
+}};
+var D=YAHOO.widget.Calendar._EVENT_TYPES;
+this.beforeSelectEvent=new YAHOO.util.CustomEvent(D.BEFORE_SELECT);
+this.beforeSelectEvent.subscribe=B;
+this.beforeSelectEvent.unsubscribe=A;
+this.selectEvent=new YAHOO.util.CustomEvent(D.SELECT);
+this.selectEvent.subscribe=B;
+this.selectEvent.unsubscribe=A;
+this.beforeDeselectEvent=new YAHOO.util.CustomEvent(D.BEFORE_DESELECT);
+this.beforeDeselectEvent.subscribe=B;
+this.beforeDeselectEvent.unsubscribe=A;
+this.deselectEvent=new YAHOO.util.CustomEvent(D.DESELECT);
+this.deselectEvent.subscribe=B;
+this.deselectEvent.unsubscribe=A;
+this.changePageEvent=new YAHOO.util.CustomEvent(D.CHANGE_PAGE);
+this.changePageEvent.subscribe=B;
+this.changePageEvent.unsubscribe=A;
+this.beforeRenderEvent=new YAHOO.util.CustomEvent(D.BEFORE_RENDER);
+this.beforeRenderEvent.subscribe=B;
+this.beforeRenderEvent.unsubscribe=A;
+this.renderEvent=new YAHOO.util.CustomEvent(D.RENDER);
+this.renderEvent.subscribe=B;
+this.renderEvent.unsubscribe=A;
+this.resetEvent=new YAHOO.util.CustomEvent(D.RESET);
+this.resetEvent.subscribe=B;
+this.resetEvent.unsubscribe=A;
+this.clearEvent=new YAHOO.util.CustomEvent(D.CLEAR);
+this.clearEvent.subscribe=B;
+this.clearEvent.unsubscribe=A;
+this.beforeShowEvent=new YAHOO.util.CustomEvent(D.BEFORE_SHOW);
+this.showEvent=new YAHOO.util.CustomEvent(D.SHOW);
+this.beforeHideEvent=new YAHOO.util.CustomEvent(D.BEFORE_HIDE);
+this.hideEvent=new YAHOO.util.CustomEvent(D.HIDE);
+this.beforeShowNavEvent=new YAHOO.util.CustomEvent(D.BEFORE_SHOW_NAV);
+this.showNavEvent=new YAHOO.util.CustomEvent(D.SHOW_NAV);
+this.beforeHideNavEvent=new YAHOO.util.CustomEvent(D.BEFORE_HIDE_NAV);
+this.hideNavEvent=new YAHOO.util.CustomEvent(D.HIDE_NAV);
+this.beforeRenderNavEvent=new YAHOO.util.CustomEvent(D.BEFORE_RENDER_NAV);
+this.renderNavEvent=new YAHOO.util.CustomEvent(D.RENDER_NAV)
+},configPages:function(K,J,G){var E=J[0];
+var C=YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGEDATE.key;
+var O="_";
+var L="groupcal";
+var N="first-of-type";
+var D="last-of-type";
+for(var B=0;
+B<E;
+++B){var M=this.id+O+B;
+var I=this.containerId+O+B;
+var H=this.cfg.getConfig();
+H.close=false;
+H.title=false;
+H.navigator=null;
+var A=this.constructChild(M,I,H);
+var F=A.cfg.getProperty(C);
+this._setMonthOnDate(F,F.getMonth()+B);
+A.cfg.setProperty(C,F);
+YAHOO.util.Dom.removeClass(A.oDomContainer,this.Style.CSS_SINGLE);
+YAHOO.util.Dom.addClass(A.oDomContainer,L);
+if(B===0){YAHOO.util.Dom.addClass(A.oDomContainer,N)
+}if(B==(E-1)){YAHOO.util.Dom.addClass(A.oDomContainer,D)
+}A.parent=this;
+A.index=B;
+this.pages[this.pages.length]=A
+}},configPageDate:function(H,G,E){var C=G[0];
+var F;
+var D=YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGEDATE.key;
+for(var B=0;
+B<this.pages.length;
+++B){var A=this.pages[B];
+if(B===0){F=A._parsePageDate(C);
+A.cfg.setProperty(D,F)
+}else{var I=new Date(F);
+this._setMonthOnDate(I,I.getMonth()+B);
+A.cfg.setProperty(D,I)
+}}},configSelected:function(C,A,E){var D=YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.SELECTED.key;
+this.delegateConfig(C,A,E);
+var B=(this.pages.length>0)?this.pages[0].cfg.getProperty(D):[];
+this.cfg.setProperty(D,B,true)
+},delegateConfig:function(B,A,E){var F=A[0];
+var D;
+for(var C=0;
+C<this.pages.length;
+C++){D=this.pages[C];
+D.cfg.setProperty(B,F)
+}},setChildFunction:function(D,B){var A=this.cfg.getProperty(YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGES.key);
+for(var C=0;
+C<A;
+++C){this.pages[C][D]=B
+}},callChildFunction:function(F,B){var A=this.cfg.getProperty(YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGES.key);
+for(var E=0;
+E<A;
+++E){var D=this.pages[E];
+if(D[F]){var C=D[F];
+C.call(D,B)
+}}},constructChild:function(D,B,C){var A=document.getElementById(B);
+if(!A){A=document.createElement("div");
+A.id=B;
+this.oDomContainer.appendChild(A)
+}return new YAHOO.widget.Calendar(D,B,C)
+},setMonth:function(E){E=parseInt(E,10);
+var F;
+var B=YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGEDATE.key;
+for(var D=0;
+D<this.pages.length;
+++D){var C=this.pages[D];
+var A=C.cfg.getProperty(B);
+if(D===0){F=A.getFullYear()
+}else{A.setFullYear(F)
+}this._setMonthOnDate(A,E+D);
+C.cfg.setProperty(B,A)
+}},setYear:function(C){var B=YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGEDATE.key;
+C=parseInt(C,10);
+for(var E=0;
+E<this.pages.length;
+++E){var D=this.pages[E];
+var A=D.cfg.getProperty(B);
+if((A.getMonth()+1)==1&&E>0){C+=1
+}D.setYear(C)
+}},render:function(){this.renderHeader();
+for(var B=0;
+B<this.pages.length;
+++B){var A=this.pages[B];
+A.render()
+}this.renderFooter()
+},select:function(A){for(var C=0;
+C<this.pages.length;
+++C){var B=this.pages[C];
+B.select(A)
+}return this.getSelectedDates()
+},selectCell:function(A){for(var C=0;
+C<this.pages.length;
+++C){var B=this.pages[C];
+B.selectCell(A)
+}return this.getSelectedDates()
+},deselect:function(A){for(var C=0;
+C<this.pages.length;
+++C){var B=this.pages[C];
+B.deselect(A)
+}return this.getSelectedDates()
+},deselectAll:function(){for(var B=0;
+B<this.pages.length;
+++B){var A=this.pages[B];
+A.deselectAll()
+}return this.getSelectedDates()
+},deselectCell:function(A){for(var C=0;
+C<this.pages.length;
+++C){var B=this.pages[C];
+B.deselectCell(A)
+}return this.getSelectedDates()
+},reset:function(){for(var B=0;
+B<this.pages.length;
+++B){var A=this.pages[B];
+A.reset()
+}},clear:function(){for(var B=0;
+B<this.pages.length;
+++B){var A=this.pages[B];
+A.clear()
+}},nextMonth:function(){for(var B=0;
+B<this.pages.length;
+++B){var A=this.pages[B];
+A.nextMonth()
+}},previousMonth:function(){for(var B=this.pages.length-1;
+B>=0;
+--B){var A=this.pages[B];
+A.previousMonth()
+}},nextYear:function(){for(var B=0;
+B<this.pages.length;
+++B){var A=this.pages[B];
+A.nextYear()
+}},previousYear:function(){for(var B=0;
+B<this.pages.length;
+++B){var A=this.pages[B];
+A.previousYear()
+}},getSelectedDates:function(){var C=[];
+var B=this.cfg.getProperty(YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.SELECTED.key);
+for(var E=0;
+E<B.length;
+++E){var D=B[E];
+var A=YAHOO.widget.DateMath.getDate(D[0],D[1]-1,D[2]);
+C.push(A)
+}C.sort(function(G,F){return G-F
+});
+return C
+},addRenderer:function(A,B){for(var D=0;
+D<this.pages.length;
+++D){var C=this.pages[D];
+C.addRenderer(A,B)
+}},addMonthRenderer:function(D,A){for(var C=0;
+C<this.pages.length;
+++C){var B=this.pages[C];
+B.addMonthRenderer(D,A)
+}},addWeekdayRenderer:function(B,A){for(var D=0;
+D<this.pages.length;
+++D){var C=this.pages[D];
+C.addWeekdayRenderer(B,A)
+}},removeRenderers:function(){this.callChildFunction("removeRenderers")
+},renderHeader:function(){},renderFooter:function(){},addMonths:function(A){this.callChildFunction("addMonths",A)
+},subtractMonths:function(A){this.callChildFunction("subtractMonths",A)
+},addYears:function(A){this.callChildFunction("addYears",A)
+},subtractYears:function(A){this.callChildFunction("subtractYears",A)
+},getCalendarPage:function(D){var F=null;
+if(D){var G=D.getFullYear(),C=D.getMonth();
+var B=this.pages;
+for(var E=0;
+E<B.length;
+++E){var A=B[E].cfg.getProperty("pagedate");
+if(A.getFullYear()===G&&A.getMonth()===C){F=B[E];
+break
+}}}return F
+},_setMonthOnDate:function(C,D){if(YAHOO.env.ua.webkit&&YAHOO.env.ua.webkit<420&&(D<0||D>11)){var B=YAHOO.widget.DateMath;
+var A=B.add(C,B.MONTH,D-C.getMonth());
+C.setTime(A.getTime())
+}else{C.setMonth(D)
+}},_fixWidth:function(){var A=0;
+for(var C=0;
+C<this.pages.length;
+++C){var B=this.pages[C];
+A+=B.oDomContainer.offsetWidth
+}if(A>0){this.oDomContainer.style.width=A+"px"
+}},toString:function(){return"CalendarGroup "+this.id
+}};
+YAHOO.widget.CalendarGroup.CSS_CONTAINER="yui-calcontainer";
+YAHOO.widget.CalendarGroup.CSS_MULTI_UP="multi";
+YAHOO.widget.CalendarGroup.CSS_2UPTITLE="title";
+YAHOO.widget.CalendarGroup.CSS_2UPCLOSE="close-icon";
+YAHOO.lang.augmentProto(YAHOO.widget.CalendarGroup,YAHOO.widget.Calendar,"buildDayLabel","buildMonthLabel","renderOutOfBoundsDate","renderRowHeader","renderRowFooter","renderCellDefault","styleCellDefault","renderCellStyleHighlight1","renderCellStyleHighlight2","renderCellStyleHighlight3","renderCellStyleHighlight4","renderCellStyleToday","renderCellStyleSelected","renderCellNotThisMonth","renderBodyCellRestricted","initStyles","configTitle","configClose","configIframe","configNavigator","createTitleBar","createCloseButton","removeTitleBar","removeCloseButton","hide","show","toDate","_parseArgs","browser");
+YAHOO.widget.CalendarGroup._DEFAULT_CONFIG=YAHOO.widget.Calendar._DEFAULT_CONFIG;
+YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGES={key:"pages",value:2};
+YAHOO.widget.CalGrp=YAHOO.widget.CalendarGroup;
+YAHOO.widget.Calendar2up=function(C,A,B){this.init(C,A,B)
+};
+YAHOO.extend(YAHOO.widget.Calendar2up,YAHOO.widget.CalendarGroup);
+YAHOO.widget.Cal2up=YAHOO.widget.Calendar2up;
+YAHOO.widget.CalendarNavigator=function(A){this.init(A)
+};
+(function(){var A=YAHOO.widget.CalendarNavigator;
+A.CLASSES={NAV:"yui-cal-nav",NAV_VISIBLE:"yui-cal-nav-visible",MASK:"yui-cal-nav-mask",YEAR:"yui-cal-nav-y",MONTH:"yui-cal-nav-m",BUTTONS:"yui-cal-nav-b",BUTTON:"yui-cal-nav-btn",ERROR:"yui-cal-nav-e",YEAR_CTRL:"yui-cal-nav-yc",MONTH_CTRL:"yui-cal-nav-mc",INVALID:"yui-invalid",DEFAULT:"yui-default"};
+A._DEFAULT_CFG={strings:{month:"Month",year:"Year",submit:"Okay",cancel:"Cancel",invalidYear:"Year needs to be a number"},monthFormat:YAHOO.widget.Calendar.LONG,initialFocus:"year"};
+A.ID_SUFFIX="_nav";
+A.MONTH_SUFFIX="_month";
+A.YEAR_SUFFIX="_year";
+A.ERROR_SUFFIX="_error";
+A.CANCEL_SUFFIX="_cancel";
+A.SUBMIT_SUFFIX="_submit";
+A.YR_MAX_DIGITS=4;
+A.YR_MINOR_INC=1;
+A.YR_MAJOR_INC=10;
+A.UPDATE_DELAY=50;
+A.YR_PATTERN=/^\d+$/;
+A.TRIM=/^\s*(.*?)\s*$/
+})();
+YAHOO.widget.CalendarNavigator.prototype={id:null,cal:null,navEl:null,maskEl:null,yearEl:null,monthEl:null,errorEl:null,submitEl:null,cancelEl:null,firstCtrl:null,lastCtrl:null,_doc:null,_year:null,_month:0,__rendered:false,init:function(A){var C=A.oDomContainer;
+this.cal=A;
+this.id=C.id+YAHOO.widget.CalendarNavigator.ID_SUFFIX;
+this._doc=C.ownerDocument;
+var B=YAHOO.env.ua.ie;
+this.__isIEQuirks=(B&&((B<=6)||(B===7&&this._doc.compatMode=="BackCompat")))
+},show:function(){var A=YAHOO.widget.CalendarNavigator.CLASSES;
+if(this.cal.beforeShowNavEvent.fire()){if(!this.__rendered){this.render()
+}this.clearErrors();
+this._updateMonthUI();
+this._updateYearUI();
+this._show(this.navEl,true);
+this.setInitialFocus();
+this.showMask();
+YAHOO.util.Dom.addClass(this.cal.oDomContainer,A.NAV_VISIBLE);
+this.cal.showNavEvent.fire()
+}},hide:function(){var A=YAHOO.widget.CalendarNavigator.CLASSES;
+if(this.cal.beforeHideNavEvent.fire()){this._show(this.navEl,false);
+this.hideMask();
+YAHOO.util.Dom.removeClass(this.cal.oDomContainer,A.NAV_VISIBLE);
+this.cal.hideNavEvent.fire()
+}},showMask:function(){this._show(this.maskEl,true);
+if(this.__isIEQuirks){this._syncMask()
+}},hideMask:function(){this._show(this.maskEl,false)
+},getMonth:function(){return this._month
+},getYear:function(){return this._year
+},setMonth:function(A){if(A>=0&&A<12){this._month=A
+}this._updateMonthUI()
+},setYear:function(B){var A=YAHOO.widget.CalendarNavigator.YR_PATTERN;
+if(YAHOO.lang.isNumber(B)&&A.test(B+"")){this._year=B
+}this._updateYearUI()
+},render:function(){this.cal.beforeRenderNavEvent.fire();
+if(!this.__rendered){this.createNav();
+this.createMask();
+this.applyListeners();
+this.__rendered=true
+}this.cal.renderNavEvent.fire()
+},createNav:function(){var B=YAHOO.widget.CalendarNavigator;
+var C=this._doc;
+var D=C.createElement("div");
+D.className=B.CLASSES.NAV;
+var A=this.renderNavContents([]);
+D.innerHTML=A.join("");
+this.cal.oDomContainer.appendChild(D);
+this.navEl=D;
+this.yearEl=C.getElementById(this.id+B.YEAR_SUFFIX);
+this.monthEl=C.getElementById(this.id+B.MONTH_SUFFIX);
+this.errorEl=C.getElementById(this.id+B.ERROR_SUFFIX);
+this.submitEl=C.getElementById(this.id+B.SUBMIT_SUFFIX);
+this.cancelEl=C.getElementById(this.id+B.CANCEL_SUFFIX);
+if(YAHOO.env.ua.gecko&&this.yearEl&&this.yearEl.type=="text"){this.yearEl.setAttribute("autocomplete","off")
+}this._setFirstLastElements()
+},createMask:function(){var B=YAHOO.widget.CalendarNavigator.CLASSES;
+var A=this._doc.createElement("div");
+A.className=B.MASK;
+this.cal.oDomContainer.appendChild(A);
+this.maskEl=A
+},_syncMask:function(){var B=this.cal.oDomContainer;
+if(B&&this.maskEl){var A=YAHOO.util.Dom.getRegion(B);
+YAHOO.util.Dom.setStyle(this.maskEl,"width",A.right-A.left+"px");
+YAHOO.util.Dom.setStyle(this.maskEl,"height",A.bottom-A.top+"px")
+}},renderNavContents:function(A){var D=YAHOO.widget.CalendarNavigator,E=D.CLASSES,B=A;
+B[B.length]='<div class="'+E.MONTH+'">';
+this.renderMonth(B);
+B[B.length]="</div>";
+B[B.length]='<div class="'+E.YEAR+'">';
+this.renderYear(B);
+B[B.length]="</div>";
+B[B.length]='<div class="'+E.BUTTONS+'">';
+this.renderButtons(B);
+B[B.length]="</div>";
+B[B.length]='<div class="'+E.ERROR+'" id="'+this.id+D.ERROR_SUFFIX+'"></div>';
+return B
+},renderMonth:function(D){var G=YAHOO.widget.CalendarNavigator,H=G.CLASSES;
+var I=this.id+G.MONTH_SUFFIX,F=this.__getCfg("monthFormat"),A=this.cal.cfg.getProperty((F==YAHOO.widget.Calendar.SHORT)?"MONTHS_SHORT":"MONTHS_LONG"),E=D;
+if(A&&A.length>0){E[E.length]='<label for="'+I+'">';
+E[E.length]=this.__getCfg("month",true);
+E[E.length]="</label>";
+E[E.length]='<select name="'+I+'" id="'+I+'" class="'+H.MONTH_CTRL+'">';
+for(var B=0;
+B<A.length;
+B++){E[E.length]='<option value="'+B+'">';
+E[E.length]=A[B];
+E[E.length]="</option>"
+}E[E.length]="</select>"
+}return E
+},renderYear:function(B){var E=YAHOO.widget.CalendarNavigator,F=E.CLASSES;
+var G=this.id+E.YEAR_SUFFIX,A=E.YR_MAX_DIGITS,D=B;
+D[D.length]='<label for="'+G+'">';
+D[D.length]=this.__getCfg("year",true);
+D[D.length]="</label>";
+D[D.length]='<input type="text" name="'+G+'" id="'+G+'" class="'+F.YEAR_CTRL+'" maxlength="'+A+'"/>';
+return D
+},renderButtons:function(A){var D=YAHOO.widget.CalendarNavigator.CLASSES;
+var B=A;
+B[B.length]='<span class="'+D.BUTTON+" "+D.DEFAULT+'">';
+B[B.length]='<button type="button" id="'+this.id+'_submit">';
+B[B.length]=this.__getCfg("submit",true);
+B[B.length]="</button>";
+B[B.length]="</span>";
+B[B.length]='<span class="'+D.BUTTON+'">';
+B[B.length]='<button type="button" id="'+this.id+'_cancel">';
+B[B.length]=this.__getCfg("cancel",true);
+B[B.length]="</button>";
+B[B.length]="</span>";
+return B
+},applyListeners:function(){var B=YAHOO.util.Event;
+function A(){if(this.validate()){this.setYear(this._getYearFromUI())
+}}function C(){this.setMonth(this._getMonthFromUI())
+}B.on(this.submitEl,"click",this.submit,this,true);
+B.on(this.cancelEl,"click",this.cancel,this,true);
+B.on(this.yearEl,"blur",A,this,true);
+B.on(this.monthEl,"change",C,this,true);
+if(this.__isIEQuirks){YAHOO.util.Event.on(this.cal.oDomContainer,"resize",this._syncMask,this,true)
+}this.applyKeyListeners()
+},purgeListeners:function(){var A=YAHOO.util.Event;
+A.removeListener(this.submitEl,"click",this.submit);
+A.removeListener(this.cancelEl,"click",this.cancel);
+A.removeListener(this.yearEl,"blur");
+A.removeListener(this.monthEl,"change");
+if(this.__isIEQuirks){A.removeListener(this.cal.oDomContainer,"resize",this._syncMask)
+}this.purgeKeyListeners()
+},applyKeyListeners:function(){var D=YAHOO.util.Event;
+var A=YAHOO.env.ua;
+var C=(A.ie)?"keydown":"keypress";
+var B=(A.ie||A.opera)?"keydown":"keypress";
+D.on(this.yearEl,"keypress",this._handleEnterKey,this,true);
+D.on(this.yearEl,C,this._handleDirectionKeys,this,true);
+D.on(this.lastCtrl,B,this._handleTabKey,this,true);
+D.on(this.firstCtrl,B,this._handleShiftTabKey,this,true)
+},purgeKeyListeners:function(){var C=YAHOO.util.Event;
+var B=(YAHOO.env.ua.ie)?"keydown":"keypress";
+var A=(YAHOO.env.ua.ie||YAHOO.env.ua.opera)?"keydown":"keypress";
+C.removeListener(this.yearEl,"keypress",this._handleEnterKey);
+C.removeListener(this.yearEl,B,this._handleDirectionKeys);
+C.removeListener(this.lastCtrl,A,this._handleTabKey);
+C.removeListener(this.firstCtrl,A,this._handleShiftTabKey)
+},submit:function(){if(this.validate()){this.hide();
+this.setMonth(this._getMonthFromUI());
+this.setYear(this._getYearFromUI());
+var B=this.cal;
+var C=this;
+function D(){B.setYear(C.getYear());
+B.setMonth(C.getMonth());
+B.render()
+}var A=YAHOO.widget.CalendarNavigator.UPDATE_DELAY;
+if(A>0){window.setTimeout(D,A)
+}else{D()
+}}},cancel:function(){this.hide()
+},validate:function(){if(this._getYearFromUI()!==null){this.clearErrors();
+return true
+}else{this.setYearError();
+this.setError(this.__getCfg("invalidYear",true));
+return false
+}},setError:function(A){if(this.errorEl){this.errorEl.innerHTML=A;
+this._show(this.errorEl,true)
+}},clearError:function(){if(this.errorEl){this.errorEl.innerHTML="";
+this._show(this.errorEl,false)
+}},setYearError:function(){YAHOO.util.Dom.addClass(this.yearEl,YAHOO.widget.CalendarNavigator.CLASSES.INVALID)
+},clearYearError:function(){YAHOO.util.Dom.removeClass(this.yearEl,YAHOO.widget.CalendarNavigator.CLASSES.INVALID)
+},clearErrors:function(){this.clearError();
+this.clearYearError()
+},setInitialFocus:function(){var A=this.submitEl;
+var B=this.__getCfg("initialFocus");
+if(B&&B.toLowerCase){B=B.toLowerCase();
+if(B=="year"){A=this.yearEl;
+try{this.yearEl.select()
+}catch(C){}}else{if(B=="month"){A=this.monthEl
+}}}if(A&&YAHOO.lang.isFunction(A.focus)){try{A.focus()
+}catch(C){}}},erase:function(){if(this.__rendered){this.purgeListeners();
+this.yearEl=null;
+this.monthEl=null;
+this.errorEl=null;
+this.submitEl=null;
+this.cancelEl=null;
+this.firstCtrl=null;
+this.lastCtrl=null;
+if(this.navEl){this.navEl.innerHTML=""
+}var B=this.navEl.parentNode;
+if(B){B.removeChild(this.navEl)
+}this.navEl=null;
+var A=this.maskEl.parentNode;
+if(A){A.removeChild(this.maskEl)
+}this.maskEl=null;
+this.__rendered=false
+}},destroy:function(){this.erase();
+this._doc=null;
+this.cal=null;
+this.id=null
+},_show:function(B,A){if(B){YAHOO.util.Dom.setStyle(B,"display",(A)?"block":"none")
+}},_getMonthFromUI:function(){if(this.monthEl){return this.monthEl.selectedIndex
+}else{return 0
+}},_getYearFromUI:function(){var B=YAHOO.widget.CalendarNavigator;
+var A=null;
+if(this.yearEl){var C=this.yearEl.value;
+C=C.replace(B.TRIM,"$1");
+if(B.YR_PATTERN.test(C)){A=parseInt(C,10)
+}}return A
+},_updateYearUI:function(){if(this.yearEl&&this._year!==null){this.yearEl.value=this._year
+}},_updateMonthUI:function(){if(this.monthEl){this.monthEl.selectedIndex=this._month
+}},_setFirstLastElements:function(){this.firstCtrl=this.monthEl;
+this.lastCtrl=this.cancelEl;
+if(this.__isMac){if(YAHOO.env.ua.webkit&&YAHOO.env.ua.webkit<420){this.firstCtrl=this.monthEl;
+this.lastCtrl=this.yearEl
+}if(YAHOO.env.ua.gecko){this.firstCtrl=this.yearEl;
+this.lastCtrl=this.yearEl
+}}},_handleEnterKey:function(B){var A=YAHOO.util.KeyListener.KEY;
+if(YAHOO.util.Event.getCharCode(B)==A.ENTER){this.submit()
+}},_handleDirectionKeys:function(G){var F=YAHOO.util.Event;
+var A=YAHOO.util.KeyListener.KEY;
+var C=YAHOO.widget.CalendarNavigator;
+var D=(this.yearEl.value)?parseInt(this.yearEl.value,10):null;
+if(isFinite(D)){var B=false;
+switch(F.getCharCode(G)){case A.UP:this.yearEl.value=D+C.YR_MINOR_INC;
+B=true;
+break;
+case A.DOWN:this.yearEl.value=Math.max(D-C.YR_MINOR_INC,0);
+B=true;
+break;
+case A.PAGE_UP:this.yearEl.value=D+C.YR_MAJOR_INC;
+B=true;
+break;
+case A.PAGE_DOWN:this.yearEl.value=Math.max(D-C.YR_MAJOR_INC,0);
+B=true;
+break;
+default:break
+}if(B){F.preventDefault(G);
+try{this.yearEl.select()
+}catch(G){}}}},_handleTabKey:function(C){var B=YAHOO.util.Event;
+var A=YAHOO.util.KeyListener.KEY;
+if(B.getCharCode(C)==A.TAB&&!C.shiftKey){try{B.preventDefault(C);
+this.firstCtrl.focus()
+}catch(C){}}},_handleShiftTabKey:function(C){var B=YAHOO.util.Event;
+var A=YAHOO.util.KeyListener.KEY;
+if(C.shiftKey&&B.getCharCode(C)==A.TAB){try{B.preventDefault(C);
+this.lastCtrl.focus()
+}catch(C){}}},__getCfg:function(D,B){var C=YAHOO.widget.CalendarNavigator._DEFAULT_CFG;
+var A=this.cal.cfg.getProperty("navigator");
+if(B){return(A!==true&&A.strings&&A.strings[D])?A.strings[D]:C.strings[D]
+}else{return(A!==true&&A[D])?A[D]:C[D]
+}},__isMac:(navigator.userAgent.toLowerCase().indexOf("macintosh")!=-1)};
+YAHOO.register("calendar",YAHOO.widget.Calendar,{version:"2.4.1",build:"742"});
\ No newline at end of file
diff --git a/sonar-server/src/main/webapp/javascripts/calendar/calendar.js b/sonar-server/src/main/webapp/javascripts/calendar/calendar.js
deleted file mode 100644 (file)
index e76a332..0000000
+++ /dev/null
@@ -1,6759 +0,0 @@
-/*
-Copyright (c) 2007, Yahoo! Inc. All rights reserved.
-Code licensed under the BSD License:
-http://developer.yahoo.net/yui/license.txt
-version: 2.4.1
-*/
-(function () {
-
-    /**
-    * Config is a utility used within an Object to allow the implementer to
-    * maintain a list of local configuration properties and listen for changes 
-    * to those properties dynamically using CustomEvent. The initial values are 
-    * also maintained so that the configuration can be reset at any given point 
-    * to its initial state.
-    * @namespace YAHOO.util
-    * @class Config
-    * @constructor
-    * @param {Object} owner The owner Object to which this Config Object belongs
-    */
-    YAHOO.util.Config = function (owner) {
-
-        if (owner) {
-            this.init(owner);
-        }
-
-
-    };
-
-
-    var Lang = YAHOO.lang,
-        CustomEvent = YAHOO.util.CustomEvent,
-        Config = YAHOO.util.Config;
-
-
-    /**
-     * Constant representing the CustomEvent type for the config changed event.
-     * @property YAHOO.util.Config.CONFIG_CHANGED_EVENT
-     * @private
-     * @static
-     * @final
-     */
-    Config.CONFIG_CHANGED_EVENT = "configChanged";
-    
-    /**
-     * Constant representing the boolean type string
-     * @property YAHOO.util.Config.BOOLEAN_TYPE
-     * @private
-     * @static
-     * @final
-     */
-    Config.BOOLEAN_TYPE = "boolean";
-    
-    Config.prototype = {
-     
-        /**
-        * Object reference to the owner of this Config Object
-        * @property owner
-        * @type Object
-        */
-        owner: null,
-        
-        /**
-        * Boolean flag that specifies whether a queue is currently 
-        * being executed
-        * @property queueInProgress
-        * @type Boolean
-        */
-        queueInProgress: false,
-        
-        /**
-        * Maintains the local collection of configuration property objects and 
-        * their specified values
-        * @property config
-        * @private
-        * @type Object
-        */ 
-        config: null,
-        
-        /**
-        * Maintains the local collection of configuration property objects as 
-        * they were initially applied.
-        * This object is used when resetting a property.
-        * @property initialConfig
-        * @private
-        * @type Object
-        */ 
-        initialConfig: null,
-        
-        /**
-        * Maintains the local, normalized CustomEvent queue
-        * @property eventQueue
-        * @private
-        * @type Object
-        */ 
-        eventQueue: null,
-        
-        /**
-        * Custom Event, notifying subscribers when Config properties are set 
-        * (setProperty is called without the silent flag
-        * @event configChangedEvent
-        */
-        configChangedEvent: null,
-    
-        /**
-        * Initializes the configuration Object and all of its local members.
-        * @method init
-        * @param {Object} owner The owner Object to which this Config 
-        * Object belongs
-        */
-        init: function (owner) {
-    
-            this.owner = owner;
-    
-            this.configChangedEvent = 
-                this.createEvent(Config.CONFIG_CHANGED_EVENT);
-    
-            this.configChangedEvent.signature = CustomEvent.LIST;
-            this.queueInProgress = false;
-            this.config = {};
-            this.initialConfig = {};
-            this.eventQueue = [];
-        
-        },
-        
-        /**
-        * Validates that the value passed in is a Boolean.
-        * @method checkBoolean
-        * @param {Object} val The value to validate
-        * @return {Boolean} true, if the value is valid
-        */ 
-        checkBoolean: function (val) {
-            return (typeof val == Config.BOOLEAN_TYPE);
-        },
-        
-        /**
-        * Validates that the value passed in is a number.
-        * @method checkNumber
-        * @param {Object} val The value to validate
-        * @return {Boolean} true, if the value is valid
-        */
-        checkNumber: function (val) {
-            return (!isNaN(val));
-        },
-        
-        /**
-        * Fires a configuration property event using the specified value. 
-        * @method fireEvent
-        * @private
-        * @param {String} key The configuration property's name
-        * @param {value} Object The value of the correct type for the property
-        */ 
-        fireEvent: function ( key, value ) {
-            var property = this.config[key];
-        
-            if (property && property.event) {
-                property.event.fire(value);
-            } 
-        },
-        
-        /**
-        * Adds a property to the Config Object's private config hash.
-        * @method addProperty
-        * @param {String} key The configuration property's name
-        * @param {Object} propertyObject The Object containing all of this 
-        * property's arguments
-        */
-        addProperty: function ( key, propertyObject ) {
-            key = key.toLowerCase();
-        
-            this.config[key] = propertyObject;
-        
-            propertyObject.event = this.createEvent(key, { scope: this.owner });
-            propertyObject.event.signature = CustomEvent.LIST;
-            
-            
-            propertyObject.key = key;
-        
-            if (propertyObject.handler) {
-                propertyObject.event.subscribe(propertyObject.handler, 
-                    this.owner);
-            }
-        
-            this.setProperty(key, propertyObject.value, true);
-            
-            if (! propertyObject.suppressEvent) {
-                this.queueProperty(key, propertyObject.value);
-            }
-            
-        },
-        
-        /**
-        * Returns a key-value configuration map of the values currently set in  
-        * the Config Object.
-        * @method getConfig
-        * @return {Object} The current config, represented in a key-value map
-        */
-        getConfig: function () {
-        
-            var cfg = {},
-                prop,
-                property;
-                
-            for (prop in this.config) {
-                property = this.config[prop];
-                if (property && property.event) {
-                    cfg[prop] = property.value;
-                }
-            }
-            
-            return cfg;
-        },
-        
-        /**
-        * Returns the value of specified property.
-        * @method getProperty
-        * @param {String} key The name of the property
-        * @return {Object}  The value of the specified property
-        */
-        getProperty: function (key) {
-            var property = this.config[key.toLowerCase()];
-            if (property && property.event) {
-                return property.value;
-            } else {
-                return undefined;
-            }
-        },
-        
-        /**
-        * Resets the specified property's value to its initial value.
-        * @method resetProperty
-        * @param {String} key The name of the property
-        * @return {Boolean} True is the property was reset, false if not
-        */
-        resetProperty: function (key) {
-    
-            key = key.toLowerCase();
-        
-            var property = this.config[key];
-    
-            if (property && property.event) {
-    
-                if (this.initialConfig[key] && 
-                    !Lang.isUndefined(this.initialConfig[key])) {
-    
-                    this.setProperty(key, this.initialConfig[key]);
-
-                    return true;
-    
-                }
-    
-            } else {
-    
-                return false;
-            }
-    
-        },
-        
-        /**
-        * Sets the value of a property. If the silent property is passed as 
-        * true, the property's event will not be fired.
-        * @method setProperty
-        * @param {String} key The name of the property
-        * @param {String} value The value to set the property to
-        * @param {Boolean} silent Whether the value should be set silently, 
-        * without firing the property event.
-        * @return {Boolean} True, if the set was successful, false if it failed.
-        */
-        setProperty: function (key, value, silent) {
-        
-            var property;
-        
-            key = key.toLowerCase();
-        
-            if (this.queueInProgress && ! silent) {
-                // Currently running through a queue... 
-                this.queueProperty(key,value);
-                return true;
-    
-            } else {
-                property = this.config[key];
-                if (property && property.event) {
-                    if (property.validator && !property.validator(value)) {
-                        return false;
-                    } else {
-                        property.value = value;
-                        if (! silent) {
-                            this.fireEvent(key, value);
-                            this.configChangedEvent.fire([key, value]);
-                        }
-                        return true;
-                    }
-                } else {
-                    return false;
-                }
-            }
-        },
-        
-        /**
-        * Sets the value of a property and queues its event to execute. If the 
-        * event is already scheduled to execute, it is
-        * moved from its current position to the end of the queue.
-        * @method queueProperty
-        * @param {String} key The name of the property
-        * @param {String} value The value to set the property to
-        * @return {Boolean}  true, if the set was successful, false if 
-        * it failed.
-        */ 
-        queueProperty: function (key, value) {
-        
-            key = key.toLowerCase();
-        
-            var property = this.config[key],
-                foundDuplicate = false,
-                iLen,
-                queueItem,
-                queueItemKey,
-                queueItemValue,
-                sLen,
-                supercedesCheck,
-                qLen,
-                queueItemCheck,
-                queueItemCheckKey,
-                queueItemCheckValue,
-                i,
-                s,
-                q;
-                                
-            if (property && property.event) {
-    
-                if (!Lang.isUndefined(value) && property.validator && 
-                    !property.validator(value)) { // validator
-                    return false;
-                } else {
-        
-                    if (!Lang.isUndefined(value)) {
-                        property.value = value;
-                    } else {
-                        value = property.value;
-                    }
-        
-                    foundDuplicate = false;
-                    iLen = this.eventQueue.length;
-        
-                    for (i = 0; i < iLen; i++) {
-                        queueItem = this.eventQueue[i];
-        
-                        if (queueItem) {
-                            queueItemKey = queueItem[0];
-                            queueItemValue = queueItem[1];
-
-                            if (queueItemKey == key) {
-    
-                                /*
-                                    found a dupe... push to end of queue, null 
-                                    current item, and break
-                                */
-    
-                                this.eventQueue[i] = null;
-    
-                                this.eventQueue.push(
-                                    [key, (!Lang.isUndefined(value) ? 
-                                    value : queueItemValue)]);
-    
-                                foundDuplicate = true;
-                                break;
-                            }
-                        }
-                    }
-                    
-                    // this is a refire, or a new property in the queue
-    
-                    if (! foundDuplicate && !Lang.isUndefined(value)) { 
-                        this.eventQueue.push([key, value]);
-                    }
-                }
-        
-                if (property.supercedes) {
-
-                    sLen = property.supercedes.length;
-
-                    for (s = 0; s < sLen; s++) {
-
-                        supercedesCheck = property.supercedes[s];
-                        qLen = this.eventQueue.length;
-
-                        for (q = 0; q < qLen; q++) {
-                            queueItemCheck = this.eventQueue[q];
-
-                            if (queueItemCheck) {
-                                queueItemCheckKey = queueItemCheck[0];
-                                queueItemCheckValue = queueItemCheck[1];
-
-                                if (queueItemCheckKey == 
-                                    supercedesCheck.toLowerCase() ) {
-
-                                    this.eventQueue.push([queueItemCheckKey, 
-                                        queueItemCheckValue]);
-
-                                    this.eventQueue[q] = null;
-                                    break;
-
-                                }
-                            }
-                        }
-                    }
-                }
-
-
-                return true;
-            } else {
-                return false;
-            }
-        },
-        
-        /**
-        * Fires the event for a property using the property's current value.
-        * @method refireEvent
-        * @param {String} key The name of the property
-        */
-        refireEvent: function (key) {
-    
-            key = key.toLowerCase();
-        
-            var property = this.config[key];
-    
-            if (property && property.event && 
-    
-                !Lang.isUndefined(property.value)) {
-    
-                if (this.queueInProgress) {
-    
-                    this.queueProperty(key);
-    
-                } else {
-    
-                    this.fireEvent(key, property.value);
-    
-                }
-    
-            }
-        },
-        
-        /**
-        * Applies a key-value Object literal to the configuration, replacing  
-        * any existing values, and queueing the property events.
-        * Although the values will be set, fireQueue() must be called for their 
-        * associated events to execute.
-        * @method applyConfig
-        * @param {Object} userConfig The configuration Object literal
-        * @param {Boolean} init  When set to true, the initialConfig will 
-        * be set to the userConfig passed in, so that calling a reset will 
-        * reset the properties to the passed values.
-        */
-        applyConfig: function (userConfig, init) {
-        
-            var sKey,
-                oConfig;
-
-            if (init) {
-                oConfig = {};
-                for (sKey in userConfig) {
-                    if (Lang.hasOwnProperty(userConfig, sKey)) {
-                        oConfig[sKey.toLowerCase()] = userConfig[sKey];
-                    }
-                }
-                this.initialConfig = oConfig;
-            }
-
-            for (sKey in userConfig) {
-                if (Lang.hasOwnProperty(userConfig, sKey)) {
-                    this.queueProperty(sKey, userConfig[sKey]);
-                }
-            }
-        },
-        
-        /**
-        * Refires the events for all configuration properties using their 
-        * current values.
-        * @method refresh
-        */
-        refresh: function () {
-        
-            var prop;
-        
-            for (prop in this.config) {
-                this.refireEvent(prop);
-            }
-        },
-        
-        /**
-        * Fires the normalized list of queued property change events
-        * @method fireQueue
-        */
-        fireQueue: function () {
-        
-            var i, 
-                queueItem,
-                key,
-                value,
-                property;
-        
-            this.queueInProgress = true;
-            for (i = 0;i < this.eventQueue.length; i++) {
-                queueItem = this.eventQueue[i];
-                if (queueItem) {
-        
-                    key = queueItem[0];
-                    value = queueItem[1];
-                    property = this.config[key];
-        
-                    property.value = value;
-        
-                    this.fireEvent(key,value);
-                }
-            }
-            
-            this.queueInProgress = false;
-            this.eventQueue = [];
-        },
-        
-        /**
-        * Subscribes an external handler to the change event for any 
-        * given property. 
-        * @method subscribeToConfigEvent
-        * @param {String} key The property name
-        * @param {Function} handler The handler function to use subscribe to 
-        * the property's event
-        * @param {Object} obj The Object to use for scoping the event handler 
-        * (see CustomEvent documentation)
-        * @param {Boolean} override Optional. If true, will override "this"  
-        * within the handler to map to the scope Object passed into the method.
-        * @return {Boolean} True, if the subscription was successful, 
-        * otherwise false.
-        */ 
-        subscribeToConfigEvent: function (key, handler, obj, override) {
-    
-            var property = this.config[key.toLowerCase()];
-    
-            if (property && property.event) {
-                if (!Config.alreadySubscribed(property.event, handler, obj)) {
-                    property.event.subscribe(handler, obj, override);
-                }
-                return true;
-            } else {
-                return false;
-            }
-    
-        },
-        
-        /**
-        * Unsubscribes an external handler from the change event for any 
-        * given property. 
-        * @method unsubscribeFromConfigEvent
-        * @param {String} key The property name
-        * @param {Function} handler The handler function to use subscribe to 
-        * the property's event
-        * @param {Object} obj The Object to use for scoping the event 
-        * handler (see CustomEvent documentation)
-        * @return {Boolean} True, if the unsubscription was successful, 
-        * otherwise false.
-        */
-        unsubscribeFromConfigEvent: function (key, handler, obj) {
-            var property = this.config[key.toLowerCase()];
-            if (property && property.event) {
-                return property.event.unsubscribe(handler, obj);
-            } else {
-                return false;
-            }
-        },
-        
-        /**
-        * Returns a string representation of the Config object
-        * @method toString
-        * @return {String} The Config object in string format.
-        */
-        toString: function () {
-            var output = "Config";
-            if (this.owner) {
-                output += " [" + this.owner.toString() + "]";
-            }
-            return output;
-        },
-        
-        /**
-        * Returns a string representation of the Config object's current 
-        * CustomEvent queue
-        * @method outputEventQueue
-        * @return {String} The string list of CustomEvents currently queued 
-        * for execution
-        */
-        outputEventQueue: function () {
-
-            var output = "",
-                queueItem,
-                q,
-                nQueue = this.eventQueue.length;
-              
-            for (q = 0; q < nQueue; q++) {
-                queueItem = this.eventQueue[q];
-                if (queueItem) {
-                    output += queueItem[0] + "=" + queueItem[1] + ", ";
-                }
-            }
-            return output;
-        },
-
-        /**
-        * Sets all properties to null, unsubscribes all listeners from each 
-        * property's change event and all listeners from the configChangedEvent.
-        * @method destroy
-        */
-        destroy: function () {
-
-            var oConfig = this.config,
-                sProperty,
-                oProperty;
-
-
-            for (sProperty in oConfig) {
-            
-                if (Lang.hasOwnProperty(oConfig, sProperty)) {
-
-                    oProperty = oConfig[sProperty];
-
-                    oProperty.event.unsubscribeAll();
-                    oProperty.event = null;
-
-                }
-            
-            }
-            
-            this.configChangedEvent.unsubscribeAll();
-            
-            this.configChangedEvent = null;
-            this.owner = null;
-            this.config = null;
-            this.initialConfig = null;
-            this.eventQueue = null;
-        
-        }
-
-    };
-    
-    
-    
-    /**
-    * Checks to determine if a particular function/Object pair are already 
-    * subscribed to the specified CustomEvent
-    * @method YAHOO.util.Config.alreadySubscribed
-    * @static
-    * @param {YAHOO.util.CustomEvent} evt The CustomEvent for which to check 
-    * the subscriptions
-    * @param {Function} fn The function to look for in the subscribers list
-    * @param {Object} obj The execution scope Object for the subscription
-    * @return {Boolean} true, if the function/Object pair is already subscribed 
-    * to the CustomEvent passed in
-    */
-    Config.alreadySubscribed = function (evt, fn, obj) {
-    
-        var nSubscribers = evt.subscribers.length,
-            subsc,
-            i;
-
-        if (nSubscribers > 0) {
-            i = nSubscribers - 1;
-            do {
-                subsc = evt.subscribers[i];
-                if (subsc && subsc.obj == obj && subsc.fn == fn) {
-                    return true;
-                }
-            }
-            while (i--);
-        }
-
-        return false;
-
-    };
-
-    YAHOO.lang.augmentProto(Config, YAHOO.util.EventProvider);
-
-}());
-
-/**
-* YAHOO.widget.DateMath is used for simple date manipulation. The class is a static utility
-* used for adding, subtracting, and comparing dates.
-* @namespace YAHOO.widget
-* @class DateMath
-*/
-YAHOO.widget.DateMath = {
-       /**
-       * Constant field representing Day
-       * @property DAY
-       * @static
-       * @final
-       * @type String
-       */
-       DAY : "D",
-
-       /**
-       * Constant field representing Week
-       * @property WEEK
-       * @static
-       * @final
-       * @type String
-       */
-       WEEK : "W",
-
-       /**
-       * Constant field representing Year
-       * @property YEAR
-       * @static
-       * @final
-       * @type String
-       */
-       YEAR : "Y",
-
-       /**
-       * Constant field representing Month
-       * @property MONTH
-       * @static
-       * @final
-       * @type String
-       */
-       MONTH : "M",
-
-       /**
-       * Constant field representing one day, in milliseconds
-       * @property ONE_DAY_MS
-       * @static
-       * @final
-       * @type Number
-       */
-       ONE_DAY_MS : 1000*60*60*24,
-
-       /**
-       * Adds the specified amount of time to the this instance.
-       * @method add
-       * @param {Date} date    The JavaScript Date object to perform addition on
-       * @param {String} field The field constant to be used for performing addition.
-       * @param {Number} amount        The number of units (measured in the field constant) to add to the date.
-       * @return {Date} The resulting Date object
-       */
-       add : function(date, field, amount) {
-               var d = new Date(date.getTime());
-               switch (field) {
-                       case this.MONTH:
-                               var newMonth = date.getMonth() + amount;
-                               var years = 0;
-
-
-                               if (newMonth < 0) {
-                                       while (newMonth < 0) {
-                                               newMonth += 12;
-                                               years -= 1;
-                                       }
-                               } else if (newMonth > 11) {
-                                       while (newMonth > 11) {
-                                               newMonth -= 12;
-                                               years += 1;
-                                       }
-                               }
-                               
-                               d.setMonth(newMonth);
-                               d.setFullYear(date.getFullYear() + years);
-                               break;
-                       case this.DAY:
-                               d.setDate(date.getDate() + amount);
-                               break;
-                       case this.YEAR:
-                               d.setFullYear(date.getFullYear() + amount);
-                               break;
-                       case this.WEEK:
-                               d.setDate(date.getDate() + (amount * 7));
-                               break;
-               }
-               return d;
-       },
-
-       /**
-       * Subtracts the specified amount of time from the this instance.
-       * @method subtract
-       * @param {Date} date    The JavaScript Date object to perform subtraction on
-       * @param {Number} field The this field constant to be used for performing subtraction.
-       * @param {Number} amount        The number of units (measured in the field constant) to subtract from the date.
-       * @return {Date} The resulting Date object
-       */
-       subtract : function(date, field, amount) {
-               return this.add(date, field, (amount*-1));
-       },
-
-       /**
-       * Determines whether a given date is before another date on the calendar.
-       * @method before
-       * @param {Date} date            The Date object to compare with the compare argument
-       * @param {Date} compareTo       The Date object to use for the comparison
-       * @return {Boolean} true if the date occurs before the compared date; false if not.
-       */
-       before : function(date, compareTo) {
-               var ms = compareTo.getTime();
-               if (date.getTime() < ms) {
-                       return true;
-               } else {
-                       return false;
-               }
-       },
-
-       /**
-       * Determines whether a given date is after another date on the calendar.
-       * @method after
-       * @param {Date} date            The Date object to compare with the compare argument
-       * @param {Date} compareTo       The Date object to use for the comparison
-       * @return {Boolean} true if the date occurs after the compared date; false if not.
-       */
-       after : function(date, compareTo) {
-               var ms = compareTo.getTime();
-               if (date.getTime() > ms) {
-                       return true;
-               } else {
-                       return false;
-               }
-       },
-
-       /**
-       * Determines whether a given date is between two other dates on the calendar.
-       * @method between
-       * @param {Date} date            The date to check for
-       * @param {Date} dateBegin       The start of the range
-       * @param {Date} dateEnd         The end of the range
-       * @return {Boolean} true if the date occurs between the compared dates; false if not.
-       */
-       between : function(date, dateBegin, dateEnd) {
-               if (this.after(date, dateBegin) && this.before(date, dateEnd)) {
-                       return true;
-               } else {
-                       return false;
-               }
-       },
-       
-       /**
-       * Retrieves a JavaScript Date object representing January 1 of any given year.
-       * @method getJan1
-       * @param {Number} calendarYear          The calendar year for which to retrieve January 1
-       * @return {Date}        January 1 of the calendar year specified.
-       */
-       getJan1 : function(calendarYear) {
-               return this.getDate(calendarYear,0,1);
-       },
-
-       /**
-       * Calculates the number of days the specified date is from January 1 of the specified calendar year.
-       * Passing January 1 to this function would return an offset value of zero.
-       * @method getDayOffset
-       * @param {Date} date    The JavaScript date for which to find the offset
-       * @param {Number} calendarYear  The calendar year to use for determining the offset
-       * @return {Number}      The number of days since January 1 of the given year
-       */
-       getDayOffset : function(date, calendarYear) {
-               var beginYear = this.getJan1(calendarYear); // Find the start of the year. This will be in week 1.
-               
-               // Find the number of days the passed in date is away from the calendar year start
-               var dayOffset = Math.ceil((date.getTime()-beginYear.getTime()) / this.ONE_DAY_MS);
-               return dayOffset;
-       },
-
-       /**
-       * Calculates the week number for the given date. This function assumes that week 1 is the
-       * week in which January 1 appears, regardless of whether the week consists of a full 7 days.
-       * The calendar year can be specified to help find what a the week number would be for a given
-       * date if the date overlaps years. For instance, a week may be considered week 1 of 2005, or
-       * week 53 of 2004. Specifying the optional calendarYear allows one to make this distinction
-       * easily.
-       * @method getWeekNumber
-       * @param {Date} date    The JavaScript date for which to find the week number
-       * @param {Number} calendarYear  OPTIONAL - The calendar year to use for determining the week number. Default is
-       *                                                                                       the calendar year of parameter "date".
-       * @return {Number}      The week number of the given date.
-       */
-       getWeekNumber : function(date, calendarYear) {
-               date = this.clearTime(date);
-               var nearestThurs = new Date(date.getTime() + (4 * this.ONE_DAY_MS) - ((date.getDay()) * this.ONE_DAY_MS));
-
-               var jan1 = this.getDate(nearestThurs.getFullYear(),0,1);
-               var dayOfYear = ((nearestThurs.getTime() - jan1.getTime()) / this.ONE_DAY_MS) - 1;
-
-               var weekNum = Math.ceil((dayOfYear)/ 7);
-               return weekNum;
-       },
-
-       /**
-       * Determines if a given week overlaps two different years.
-       * @method isYearOverlapWeek
-       * @param {Date} weekBeginDate   The JavaScript Date representing the first day of the week.
-       * @return {Boolean}     true if the date overlaps two different years.
-       */
-       isYearOverlapWeek : function(weekBeginDate) {
-               var overlaps = false;
-               var nextWeek = this.add(weekBeginDate, this.DAY, 6);
-               if (nextWeek.getFullYear() != weekBeginDate.getFullYear()) {
-                       overlaps = true;
-               }
-               return overlaps;
-       },
-
-       /**
-       * Determines if a given week overlaps two different months.
-       * @method isMonthOverlapWeek
-       * @param {Date} weekBeginDate   The JavaScript Date representing the first day of the week.
-       * @return {Boolean}     true if the date overlaps two different months.
-       */
-       isMonthOverlapWeek : function(weekBeginDate) {
-               var overlaps = false;
-               var nextWeek = this.add(weekBeginDate, this.DAY, 6);
-               if (nextWeek.getMonth() != weekBeginDate.getMonth()) {
-                       overlaps = true;
-               }
-               return overlaps;
-       },
-
-       /**
-       * Gets the first day of a month containing a given date.
-       * @method findMonthStart
-       * @param {Date} date    The JavaScript Date used to calculate the month start
-       * @return {Date}                The JavaScript Date representing the first day of the month
-       */
-       findMonthStart : function(date) {
-               var start = this.getDate(date.getFullYear(), date.getMonth(), 1);
-               return start;
-       },
-
-       /**
-       * Gets the last day of a month containing a given date.
-       * @method findMonthEnd
-       * @param {Date} date    The JavaScript Date used to calculate the month end
-       * @return {Date}                The JavaScript Date representing the last day of the month
-       */
-       findMonthEnd : function(date) {
-               var start = this.findMonthStart(date);
-               var nextMonth = this.add(start, this.MONTH, 1);
-               var end = this.subtract(nextMonth, this.DAY, 1);
-               return end;
-       },
-
-       /**
-       * Clears the time fields from a given date, effectively setting the time to 12 noon.
-       * @method clearTime
-       * @param {Date} date    The JavaScript Date for which the time fields will be cleared
-       * @return {Date}                The JavaScript Date cleared of all time fields
-       */
-       clearTime : function(date) {
-               date.setHours(12,0,0,0);
-               return date;
-       },
-
-       /**
-        * Returns a new JavaScript Date object, representing the given year, month and date. Time fields (hr, min, sec, ms) on the new Date object
-        * are set to 0. The method allows Date instances to be created with the a year less than 100. "new Date(year, month, date)" implementations 
-        * set the year to 19xx if a year (xx) which is less than 100 is provided.
-        * <p>
-        * <em>NOTE:</em>Validation on argument values is not performed. It is the caller's responsibility to ensure
-        * arguments are valid as per the ECMAScript-262 Date object specification for the new Date(year, month[, date]) constructor.
-        * </p>
-        * @method getDate
-        * @param {Number} y Year.
-        * @param {Number} m Month index from 0 (Jan) to 11 (Dec).
-        * @param {Number} d (optional) Date from 1 to 31. If not provided, defaults to 1.
-        * @return {Date} The JavaScript date object with year, month, date set as provided.
-        */
-       getDate : function(y, m, d) {
-               var dt = null;
-               if (YAHOO.lang.isUndefined(d)) {
-                       d = 1;
-               }
-               if (y >= 100) {
-                       dt = new Date(y, m, d);
-               } else {
-                       dt = new Date();
-                       dt.setFullYear(y);
-                       dt.setMonth(m);
-                       dt.setDate(d);
-                       dt.setHours(0,0,0,0);
-               }
-               return dt;
-       }
-};
-
-/**
-* The Calendar component is a UI control that enables users to choose one or more dates from a graphical calendar presented in a one-month or
-* multi-month interface. Calendars are generated entirely via script and can be navigated without any page refreshes.
-* @module    calendar
-* @title    Calendar
-* @namespace  YAHOO.widget
-* @requires  yahoo,dom,event
-*/
-
-/**
-* Calendar is the base class for the Calendar widget. In its most basic
-* implementation, it has the ability to render a calendar widget on the page
-* that can be manipulated to select a single date, move back and forth between
-* months and years.
-* <p>To construct the placeholder for the calendar widget, the code is as
-* follows:
-*      <xmp>
-*              <div id="calContainer"></div>
-*      </xmp>
-* </p>
-* <p>
-* <strong>NOTE: As of 2.4.0, the constructor's ID argument is optional.</strong>
-* The Calendar can be constructed by simply providing a container ID string, 
-* or a reference to a container DIV HTMLElement (the element needs to exist 
-* in the document).
-* 
-* E.g.:
-*      <xmp>
-*              var c = new YAHOO.widget.Calendar("calContainer", configOptions);
-*      </xmp>
-* or:
-*   <xmp>
-*       var containerDiv = YAHOO.util.Dom.get("calContainer");
-*              var c = new YAHOO.widget.Calendar(containerDiv, configOptions);
-*      </xmp>
-* </p>
-* <p>
-* If not provided, the ID will be generated from the container DIV ID by adding an "_t" suffix.
-* For example if an ID is not provided, and the container's ID is "calContainer", the Calendar's ID will be set to "calContainer_t".
-* </p>
-* 
-* @namespace YAHOO.widget
-* @class Calendar
-* @constructor
-* @param {String} id optional The id of the table element that will represent the Calendar widget. As of 2.4.0, this argument is optional.
-* @param {String | HTMLElement} container The id of the container div element that will wrap the Calendar table, or a reference to a DIV element which exists in the document.
-* @param {Object} config optional The configuration object containing the initial configuration values for the Calendar.
-*/
-YAHOO.widget.Calendar = function(id, containerId, config) {
-       this.init.apply(this, arguments);
-};
-
-/**
-* The path to be used for images loaded for the Calendar
-* @property YAHOO.widget.Calendar.IMG_ROOT
-* @static
-* @deprecated  You can now customize images by overriding the calclose, calnavleft and calnavright default CSS classes for the close icon, left arrow and right arrow respectively
-* @type String
-*/
-YAHOO.widget.Calendar.IMG_ROOT = null;
-
-/**
-* Type constant used for renderers to represent an individual date (M/D/Y)
-* @property YAHOO.widget.Calendar.DATE
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.DATE = "D";
-
-/**
-* Type constant used for renderers to represent an individual date across any year (M/D)
-* @property YAHOO.widget.Calendar.MONTH_DAY
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.MONTH_DAY = "MD";
-
-/**
-* Type constant used for renderers to represent a weekday
-* @property YAHOO.widget.Calendar.WEEKDAY
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.WEEKDAY = "WD";
-
-/**
-* Type constant used for renderers to represent a range of individual dates (M/D/Y-M/D/Y)
-* @property YAHOO.widget.Calendar.RANGE
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.RANGE = "R";
-
-/**
-* Type constant used for renderers to represent a month across any year
-* @property YAHOO.widget.Calendar.MONTH
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.MONTH = "M";
-
-/**
-* Constant that represents the total number of date cells that are displayed in a given month
-* @property YAHOO.widget.Calendar.DISPLAY_DAYS
-* @static
-* @final
-* @type Number
-*/
-YAHOO.widget.Calendar.DISPLAY_DAYS = 42;
-
-/**
-* Constant used for halting the execution of the remainder of the render stack
-* @property YAHOO.widget.Calendar.STOP_RENDER
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.STOP_RENDER = "S";
-
-/**
-* Constant used to represent short date field string formats (e.g. Tu or Feb)
-* @property YAHOO.widget.Calendar.SHORT
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.SHORT = "short";
-
-/**
-* Constant used to represent long date field string formats (e.g. Monday or February)
-* @property YAHOO.widget.Calendar.LONG
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.LONG = "long";
-
-/**
-* Constant used to represent medium date field string formats (e.g. Mon)
-* @property YAHOO.widget.Calendar.MEDIUM
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.MEDIUM = "medium";
-
-/**
-* Constant used to represent single character date field string formats (e.g. M, T, W)
-* @property YAHOO.widget.Calendar.ONE_CHAR
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.Calendar.ONE_CHAR = "1char";
-
-/**
-* The set of default Config property keys and values for the Calendar
-* @property YAHOO.widget.Calendar._DEFAULT_CONFIG
-* @final
-* @static
-* @private
-* @type Object
-*/
-YAHOO.widget.Calendar._DEFAULT_CONFIG = {
-       // Default values for pagedate and selected are not class level constants - they are set during instance creation 
-       PAGEDATE : {key:"pagedate", value:null},
-       SELECTED : {key:"selected", value:null},
-       TITLE : {key:"title", value:""},
-       CLOSE : {key:"close", value:false},
-       IFRAME : {key:"iframe", value:(YAHOO.env.ua.ie && YAHOO.env.ua.ie <= 6) ? true : false},
-       MINDATE : {key:"mindate", value:null},
-       MAXDATE : {key:"maxdate", value:null},
-       MULTI_SELECT : {key:"multi_select", value:false},
-       START_WEEKDAY : {key:"start_weekday", value:0},
-       SHOW_WEEKDAYS : {key:"show_weekdays", value:true},
-       SHOW_WEEK_HEADER : {key:"show_week_header", value:false},
-       SHOW_WEEK_FOOTER : {key:"show_week_footer", value:false},
-       HIDE_BLANK_WEEKS : {key:"hide_blank_weeks", value:false},
-       NAV_ARROW_LEFT: {key:"nav_arrow_left", value:null} ,
-       NAV_ARROW_RIGHT : {key:"nav_arrow_right", value:null} ,
-       MONTHS_SHORT : {key:"months_short", value:["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]},
-       MONTHS_LONG: {key:"months_long", value:["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"]},
-       WEEKDAYS_1CHAR: {key:"weekdays_1char", value:["S", "M", "T", "W", "T", "F", "S"]},
-       WEEKDAYS_SHORT: {key:"weekdays_short", value:["Su", "Mo", "Tu", "We", "Th", "Fr", "Sa"]},
-       WEEKDAYS_MEDIUM: {key:"weekdays_medium", value:["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"]},
-       WEEKDAYS_LONG: {key:"weekdays_long", value:["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"]},
-       LOCALE_MONTHS:{key:"locale_months", value:"long"},
-       LOCALE_WEEKDAYS:{key:"locale_weekdays", value:"short"},
-       DATE_DELIMITER:{key:"date_delimiter", value:","},
-       DATE_FIELD_DELIMITER:{key:"date_field_delimiter", value:"/"},
-       DATE_RANGE_DELIMITER:{key:"date_range_delimiter", value:"-"},
-       MY_MONTH_POSITION:{key:"my_month_position", value:1},
-       MY_YEAR_POSITION:{key:"my_year_position", value:2},
-       MD_MONTH_POSITION:{key:"md_month_position", value:1},
-       MD_DAY_POSITION:{key:"md_day_position", value:2},
-       MDY_MONTH_POSITION:{key:"mdy_month_position", value:1},
-       MDY_DAY_POSITION:{key:"mdy_day_position", value:2},
-       MDY_YEAR_POSITION:{key:"mdy_year_position", value:3},
-       MY_LABEL_MONTH_POSITION:{key:"my_label_month_position", value:1},
-       MY_LABEL_YEAR_POSITION:{key:"my_label_year_position", value:2},
-       MY_LABEL_MONTH_SUFFIX:{key:"my_label_month_suffix", value:" "},
-       MY_LABEL_YEAR_SUFFIX:{key:"my_label_year_suffix", value:""},
-       NAV: {key:"navigator", value: null}
-};
-
-/**
-* The set of Custom Event types supported by the Calendar
-* @property YAHOO.widget.Calendar._EVENT_TYPES
-* @final
-* @static
-* @private
-* @type Object
-*/
-YAHOO.widget.Calendar._EVENT_TYPES = {
-       BEFORE_SELECT : "beforeSelect", 
-       SELECT : "select",
-       BEFORE_DESELECT : "beforeDeselect",
-       DESELECT : "deselect",
-       CHANGE_PAGE : "changePage",
-       BEFORE_RENDER : "beforeRender",
-       RENDER : "render",
-       RESET : "reset",
-       CLEAR : "clear",
-       BEFORE_HIDE : "beforeHide",
-       HIDE : "hide",
-       BEFORE_SHOW : "beforeShow",
-       SHOW : "show",
-       BEFORE_HIDE_NAV : "beforeHideNav",
-       HIDE_NAV : "hideNav",
-       BEFORE_SHOW_NAV : "beforeShowNav",
-       SHOW_NAV : "showNav",
-       BEFORE_RENDER_NAV : "beforeRenderNav",
-       RENDER_NAV : "renderNav"
-};
-
-/**
-* The set of default style constants for the Calendar
-* @property YAHOO.widget.Calendar._STYLES
-* @final
-* @static
-* @private
-* @type Object
-*/
-YAHOO.widget.Calendar._STYLES = {
-       CSS_ROW_HEADER: "calrowhead",
-       CSS_ROW_FOOTER: "calrowfoot",
-       CSS_CELL : "calcell",
-       CSS_CELL_SELECTOR : "selector",
-       CSS_CELL_SELECTED : "selected",
-       CSS_CELL_SELECTABLE : "selectable",
-       CSS_CELL_RESTRICTED : "restricted",
-       CSS_CELL_TODAY : "today",
-       CSS_CELL_OOM : "oom",
-       CSS_CELL_OOB : "previous",
-       CSS_HEADER : "calheader",
-       CSS_HEADER_TEXT : "calhead",
-       CSS_BODY : "calbody",
-       CSS_WEEKDAY_CELL : "calweekdaycell",
-       CSS_WEEKDAY_ROW : "calweekdayrow",
-       CSS_FOOTER : "calfoot",
-       CSS_CALENDAR : "yui-calendar",
-       CSS_SINGLE : "single",
-       CSS_CONTAINER : "yui-calcontainer",
-       CSS_NAV_LEFT : "calnavleft",
-       CSS_NAV_RIGHT : "calnavright",
-       CSS_NAV : "calnav",
-       CSS_CLOSE : "calclose",
-       CSS_CELL_TOP : "calcelltop",
-       CSS_CELL_LEFT : "calcellleft",
-       CSS_CELL_RIGHT : "calcellright",
-       CSS_CELL_BOTTOM : "calcellbottom",
-       CSS_CELL_HOVER : "calcellhover",
-       CSS_CELL_HIGHLIGHT1 : "highlight1",
-       CSS_CELL_HIGHLIGHT2 : "highlight2",
-       CSS_CELL_HIGHLIGHT3 : "highlight3",
-       CSS_CELL_HIGHLIGHT4 : "highlight4"
-};
-
-YAHOO.widget.Calendar.prototype = {
-
-       /**
-       * The configuration object used to set up the calendars various locale and style options.
-       * @property Config
-       * @private
-       * @deprecated Configuration properties should be set by calling Calendar.cfg.setProperty.
-       * @type Object
-       */
-       Config : null,
-
-       /**
-       * The parent CalendarGroup, only to be set explicitly by the parent group
-       * @property parent
-       * @type CalendarGroup
-       */      
-       parent : null,
-
-       /**
-       * The index of this item in the parent group
-       * @property index
-       * @type Number
-       */
-       index : -1,
-
-       /**
-       * The collection of calendar table cells
-       * @property cells
-       * @type HTMLTableCellElement[]
-       */
-       cells : null,
-
-       /**
-       * The collection of calendar cell dates that is parallel to the cells collection. The array contains dates field arrays in the format of [YYYY, M, D].
-       * @property cellDates
-       * @type Array[](Number[])
-       */
-       cellDates : null,
-
-       /**
-       * The id that uniquely identifies this Calendar.
-       * @property id
-       * @type String
-       */
-       id : null,
-
-       /**
-       * The unique id associated with the Calendar's container
-       * @property containerId
-       * @type String
-       */
-       containerId: null,
-
-       /**
-       * The DOM element reference that points to this calendar's container element. The calendar will be inserted into this element when the shell is rendered.
-       * @property oDomContainer
-       * @type HTMLElement
-       */
-       oDomContainer : null,
-
-       /**
-       * A Date object representing today's date.
-       * @property today
-       * @type Date
-       */
-       today : null,
-
-       /**
-       * The list of render functions, along with required parameters, used to render cells. 
-       * @property renderStack
-       * @type Array[]
-       */
-       renderStack : null,
-
-       /**
-       * A copy of the initial render functions created before rendering.
-       * @property _renderStack
-       * @private
-       * @type Array
-       */
-       _renderStack : null,
-
-       /**
-       * A reference to the CalendarNavigator instance created for this Calendar.
-       * Will be null if the "navigator" configuration property has not been set
-       * @property oNavigator
-       * @type CalendarNavigator
-       */
-       oNavigator : null,
-
-       /**
-       * The private list of initially selected dates.
-       * @property _selectedDates
-       * @private
-       * @type Array
-       */
-       _selectedDates : null,
-
-       /**
-       * A map of DOM event handlers to attach to cells associated with specific CSS class names
-       * @property domEventMap
-       * @type Object
-       */
-       domEventMap : null,
-
-       /**
-        * Protected helper used to parse Calendar constructor/init arguments.
-        *
-        * As of 2.4.0, Calendar supports a simpler constructor 
-        * signature. This method reconciles arguments
-        * received in the pre 2.4.0 and 2.4.0 formats.
-        * 
-        * @protected
-        * @method _parseArgs
-        * @param {Array} Function "arguments" array
-        * @return {Object} Object with id, container, config properties containing
-        * the reconciled argument values.
-        **/
-       _parseArgs : function(args) {
-               /*
-                  2.4.0 Constructors signatures
-
-                  new Calendar(String)
-                  new Calendar(HTMLElement)
-                  new Calendar(String, ConfigObject)
-                  new Calendar(HTMLElement, ConfigObject)
-
-                  Pre 2.4.0 Constructor signatures
-
-                  new Calendar(String, String)
-                  new Calendar(String, HTMLElement)
-                  new Calendar(String, String, ConfigObject)
-                  new Calendar(String, HTMLElement, ConfigObject)
-                */
-               var nArgs = {id:null, container:null, config:null};
-
-               if (args && args.length && args.length > 0) {
-                       switch (args.length) {
-                               case 1:
-                                       nArgs.id = null;
-                                       nArgs.container = args[0];
-                                       nArgs.config = null;
-                                       break;
-                               case 2:
-                                       if (YAHOO.lang.isObject(args[1]) && !args[1].tagName && !(args[1] instanceof String)) {
-                                               nArgs.id = null;
-                                               nArgs.container = args[0];
-                                               nArgs.config = args[1];
-                                       } else {
-                                               nArgs.id = args[0];
-                                               nArgs.container = args[1];
-                                               nArgs.config = null;
-                                       }
-                                       break;
-                               default: // 3+
-                                       nArgs.id = args[0];
-                                       nArgs.container = args[1];
-                                       nArgs.config = args[2];
-                                       break;
-                       }
-               } else {
-               }
-               return nArgs;
-       },
-
-       /**
-       * Initializes the Calendar widget.
-       * @method init
-       *
-       * @param {String} id optional The id of the table element that will represent the Calendar widget. As of 2.4.0, this argument is optional.
-       * @param {String | HTMLElement} container The id of the container div element that will wrap the Calendar table, or a reference to a DIV element which exists in the document.
-       * @param {Object} config optional The configuration object containing the initial configuration values for the Calendar.
-       */
-       init : function(id, container, config) {
-               // Normalize 2.4.0, pre 2.4.0 args
-               var nArgs = this._parseArgs(arguments);
-
-               id = nArgs.id;
-               container = nArgs.container;
-               config = nArgs.config;
-
-               this.oDomContainer = YAHOO.util.Dom.get(container);
-
-               if (!this.oDomContainer.id) {
-                       this.oDomContainer.id = YAHOO.util.Dom.generateId();
-               }
-               if (!id) {
-                       id = this.oDomContainer.id + "_t";
-               }
-
-               this.id = id;
-               this.containerId = this.oDomContainer.id;
-
-               this.initEvents();
-
-               this.today = new Date();
-               YAHOO.widget.DateMath.clearTime(this.today);
-
-               /**
-               * The Config object used to hold the configuration variables for the Calendar
-               * @property cfg
-               * @type YAHOO.util.Config
-               */
-               this.cfg = new YAHOO.util.Config(this);
-
-               /**
-               * The local object which contains the Calendar's options
-               * @property Options
-               * @type Object
-               */
-               this.Options = {};
-
-               /**
-               * The local object which contains the Calendar's locale settings
-               * @property Locale
-               * @type Object
-               */
-               this.Locale = {};
-
-               this.initStyles();
-
-               YAHOO.util.Dom.addClass(this.oDomContainer, this.Style.CSS_CONTAINER);
-               YAHOO.util.Dom.addClass(this.oDomContainer, this.Style.CSS_SINGLE);
-
-               this.cellDates = [];
-               this.cells = [];
-               this.renderStack = [];
-               this._renderStack = [];
-
-               this.setupConfig();
-
-               if (config) {
-                       this.cfg.applyConfig(config, true);
-               }
-
-               this.cfg.fireQueue();
-       },
-
-       /**
-       * Default Config listener for the iframe property. If the iframe config property is set to true, 
-       * renders the built-in IFRAME shim if the container is relatively or absolutely positioned.
-       * 
-       * @method configIframe
-       */
-       configIframe : function(type, args, obj) {
-               var useIframe = args[0];
-       
-               if (!this.parent) {
-                       if (YAHOO.util.Dom.inDocument(this.oDomContainer)) {
-                               if (useIframe) {
-                                       var pos = YAHOO.util.Dom.getStyle(this.oDomContainer, "position");
-                                       
-                                       if (pos == "absolute" || pos == "relative") {
-                                               
-                                               if (!YAHOO.util.Dom.inDocument(this.iframe)) {
-                                                       this.iframe = document.createElement("iframe");
-                                                       this.iframe.src = "javascript:false;";
-       
-                                                       YAHOO.util.Dom.setStyle(this.iframe, "opacity", "0");
-       
-                                                       if (YAHOO.env.ua.ie && YAHOO.env.ua.ie <= 6) {
-                                                               YAHOO.util.Dom.addClass(this.iframe, "fixedsize");
-                                                       }
-       
-                                                       this.oDomContainer.insertBefore(this.iframe, this.oDomContainer.firstChild);
-                                               }
-                                       }
-                               } else {
-                                       if (this.iframe) {
-                                               if (this.iframe.parentNode) {
-                                                       this.iframe.parentNode.removeChild(this.iframe);
-                                               }
-                                               this.iframe = null;
-                                       }
-                               }
-                       }
-               }
-       },
-
-       /**
-       * Default handler for the "title" property
-       * @method configTitle
-       */
-       configTitle : function(type, args, obj) {
-               var title = args[0];
-
-               // "" disables title bar
-               if (title) {
-                       this.createTitleBar(title);
-               } else {
-                       var close = this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.CLOSE.key);
-                       if (!close) {
-                               this.removeTitleBar();
-                       } else {
-                               this.createTitleBar("&#160;");
-                       }
-               }
-       },
-       
-       /**
-       * Default handler for the "close" property
-       * @method configClose
-       */
-       configClose : function(type, args, obj) {
-               var close = args[0],
-                       title = this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.TITLE.key);
-       
-               if (close) {
-                       if (!title) {
-                               this.createTitleBar("&#160;");
-                       }
-                       this.createCloseButton();
-               } else {
-                       this.removeCloseButton();
-                       if (!title) {
-                               this.removeTitleBar();
-                       }
-               }
-       },
-       
-       /**
-       * Initializes Calendar's built-in CustomEvents
-       * @method initEvents
-       */
-       initEvents : function() {
-       
-               var defEvents = YAHOO.widget.Calendar._EVENT_TYPES;
-       
-               /**
-               * Fired before a selection is made
-               * @event beforeSelectEvent
-               */
-               this.beforeSelectEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_SELECT); 
-       
-               /**
-               * Fired when a selection is made
-               * @event selectEvent
-               * @param {Array}        Array of Date field arrays in the format [YYYY, MM, DD].
-               */
-               this.selectEvent = new YAHOO.util.CustomEvent(defEvents.SELECT);
-       
-               /**
-               * Fired before a selection is made
-               * @event beforeDeselectEvent
-               */
-               this.beforeDeselectEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_DESELECT);
-       
-               /**
-               * Fired when a selection is made
-               * @event deselectEvent
-               * @param {Array}        Array of Date field arrays in the format [YYYY, MM, DD].
-               */
-               this.deselectEvent = new YAHOO.util.CustomEvent(defEvents.DESELECT);
-       
-               /**
-               * Fired when the Calendar page is changed
-               * @event changePageEvent
-               */
-               this.changePageEvent = new YAHOO.util.CustomEvent(defEvents.CHANGE_PAGE);
-       
-               /**
-               * Fired before the Calendar is rendered
-               * @event beforeRenderEvent
-               */
-               this.beforeRenderEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_RENDER);
-       
-               /**
-               * Fired when the Calendar is rendered
-               * @event renderEvent
-               */
-               this.renderEvent = new YAHOO.util.CustomEvent(defEvents.RENDER);
-       
-               /**
-               * Fired when the Calendar is reset
-               * @event resetEvent
-               */
-               this.resetEvent = new YAHOO.util.CustomEvent(defEvents.RESET);
-       
-               /**
-               * Fired when the Calendar is cleared
-               * @event clearEvent
-               */
-               this.clearEvent = new YAHOO.util.CustomEvent(defEvents.CLEAR);
-       
-               /**
-               * Fired just before the Calendar is to be shown
-               * @event beforeShowEvent
-               */
-               this.beforeShowEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_SHOW);
-       
-               /**
-               * Fired after the Calendar is shown
-               * @event showEvent
-               */
-               this.showEvent = new YAHOO.util.CustomEvent(defEvents.SHOW);
-       
-               /**
-               * Fired just before the Calendar is to be hidden
-               * @event beforeHideEvent
-               */
-               this.beforeHideEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_HIDE);
-       
-               /**
-               * Fired after the Calendar is hidden
-               * @event hideEvent
-               */
-               this.hideEvent = new YAHOO.util.CustomEvent(defEvents.HIDE);
-
-               /**
-               * Fired just before the CalendarNavigator is to be shown
-               * @event beforeShowNavEvent
-               */
-               this.beforeShowNavEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_SHOW_NAV);
-       
-               /**
-               * Fired after the CalendarNavigator is shown
-               * @event showNavEvent
-               */
-               this.showNavEvent = new YAHOO.util.CustomEvent(defEvents.SHOW_NAV);
-       
-               /**
-               * Fired just before the CalendarNavigator is to be hidden
-               * @event beforeHideNavEvent
-               */
-               this.beforeHideNavEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_HIDE_NAV);
-       
-               /**
-               * Fired after the CalendarNavigator is hidden
-               * @event hideNavEvent
-               */
-               this.hideNavEvent = new YAHOO.util.CustomEvent(defEvents.HIDE_NAV);
-
-               /**
-               * Fired just before the CalendarNavigator is to be rendered
-               * @event beforeRenderNavEvent
-               */
-               this.beforeRenderNavEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_RENDER_NAV);
-
-               /**
-               * Fired after the CalendarNavigator is rendered
-               * @event renderNavEvent
-               */
-               this.renderNavEvent = new YAHOO.util.CustomEvent(defEvents.RENDER_NAV);
-
-               this.beforeSelectEvent.subscribe(this.onBeforeSelect, this, true);
-               this.selectEvent.subscribe(this.onSelect, this, true);
-               this.beforeDeselectEvent.subscribe(this.onBeforeDeselect, this, true);
-               this.deselectEvent.subscribe(this.onDeselect, this, true);
-               this.changePageEvent.subscribe(this.onChangePage, this, true);
-               this.renderEvent.subscribe(this.onRender, this, true);
-               this.resetEvent.subscribe(this.onReset, this, true);
-               this.clearEvent.subscribe(this.onClear, this, true);
-       },
-       
-       /**
-       * The default event function that is attached to a date link within a calendar cell
-       * when the calendar is rendered.
-       * @method doSelectCell
-       * @param {DOMEvent} e   The event
-       * @param {Calendar} cal A reference to the calendar passed by the Event utility
-       */
-       doSelectCell : function(e, cal) {
-               var cell,index,d,date;
-
-               var target = YAHOO.util.Event.getTarget(e);
-               var tagName = target.tagName.toLowerCase();
-               var defSelector = false;
-
-               while (tagName != "td" && ! YAHOO.util.Dom.hasClass(target, cal.Style.CSS_CELL_SELECTABLE)) {
-
-                       if (!defSelector && tagName == "a" && YAHOO.util.Dom.hasClass(target, cal.Style.CSS_CELL_SELECTOR)) {
-                               defSelector = true;     
-                       }
-
-                       target = target.parentNode;
-                       tagName = target.tagName.toLowerCase();
-                       // TODO: No need to go all the way up to html.
-                       if (tagName == "html") {
-                               return;
-                       }
-               }
-
-               if (defSelector) {
-                       // Stop link href navigation for default renderer
-                       YAHOO.util.Event.preventDefault(e);
-               }
-       
-               cell = target;
-
-               if (YAHOO.util.Dom.hasClass(cell, cal.Style.CSS_CELL_SELECTABLE)) {
-                       index = cell.id.split("cell")[1];
-                       d = cal.cellDates[index];
-                       date = YAHOO.widget.DateMath.getDate(d[0],d[1]-1,d[2]);
-               
-                       var link;
-
-                       if (cal.Options.MULTI_SELECT) {
-                               link = cell.getElementsByTagName("a")[0];
-                               if (link) {
-                                       link.blur();
-                               }
-
-                               var cellDate = cal.cellDates[index];
-                               var cellDateIndex = cal._indexOfSelectedFieldArray(cellDate);
-
-                               if (cellDateIndex > -1) {       
-                                       cal.deselectCell(index);
-                               } else {
-                                       cal.selectCell(index);
-                               }       
-       
-                       } else {
-                               link = cell.getElementsByTagName("a")[0];
-                               if (link) {
-                                       link.blur();
-                               }
-                               cal.selectCell(index);
-                       }
-               }
-       },
-
-       /**
-       * The event that is executed when the user hovers over a cell
-       * @method doCellMouseOver
-       * @param {DOMEvent} e   The event
-       * @param {Calendar} cal A reference to the calendar passed by the Event utility
-       */
-       doCellMouseOver : function(e, cal) {
-               var target;
-               if (e) {
-                       target = YAHOO.util.Event.getTarget(e);
-               } else {
-                       target = this;
-               }
-
-               while (target.tagName && target.tagName.toLowerCase() != "td") {
-                       target = target.parentNode;
-                       if (!target.tagName || target.tagName.toLowerCase() == "html") {
-                               return;
-                       }
-               }
-
-               if (YAHOO.util.Dom.hasClass(target, cal.Style.CSS_CELL_SELECTABLE)) {
-                       YAHOO.util.Dom.addClass(target, cal.Style.CSS_CELL_HOVER);
-               }
-       },
-
-       /**
-       * The event that is executed when the user moves the mouse out of a cell
-       * @method doCellMouseOut
-       * @param {DOMEvent} e   The event
-       * @param {Calendar} cal A reference to the calendar passed by the Event utility
-       */
-       doCellMouseOut : function(e, cal) {
-               var target;
-               if (e) {
-                       target = YAHOO.util.Event.getTarget(e);
-               } else {
-                       target = this;
-               }
-
-               while (target.tagName && target.tagName.toLowerCase() != "td") {
-                       target = target.parentNode;
-                       if (!target.tagName || target.tagName.toLowerCase() == "html") {
-                               return;
-                       }
-               }
-
-               if (YAHOO.util.Dom.hasClass(target, cal.Style.CSS_CELL_SELECTABLE)) {
-                       YAHOO.util.Dom.removeClass(target, cal.Style.CSS_CELL_HOVER);
-               }
-       },
-       
-       setupConfig : function() {
-       
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-
-               /**
-               * The month/year representing the current visible Calendar date (mm/yyyy)
-               * @config pagedate
-               * @type String
-               * @default today's date
-               */
-               this.cfg.addProperty(defCfg.PAGEDATE.key, { value:new Date(), handler:this.configPageDate } );
-
-               /**
-               * The date or range of dates representing the current Calendar selection
-               * @config selected
-               * @type String
-               * @default []
-               */
-               this.cfg.addProperty(defCfg.SELECTED.key, { value:[], handler:this.configSelected } );
-
-               /**
-               * The title to display above the Calendar's month header
-               * @config title
-               * @type String
-               * @default ""
-               */
-               this.cfg.addProperty(defCfg.TITLE.key, { value:defCfg.TITLE.value, handler:this.configTitle } );
-
-               /**
-               * Whether or not a close button should be displayed for this Calendar
-               * @config close
-               * @type Boolean
-               * @default false
-               */
-               this.cfg.addProperty(defCfg.CLOSE.key, { value:defCfg.CLOSE.value, handler:this.configClose } );
-
-               /**
-               * Whether or not an iframe shim should be placed under the Calendar to prevent select boxes from bleeding through in Internet Explorer 6 and below.
-               * This property is enabled by default for IE6 and below. It is disabled by default for other browsers for performance reasons, but can be 
-               * enabled if required.
-               * 
-               * @config iframe
-               * @type Boolean
-               * @default true for IE6 and below, false for all other browsers
-               */
-               this.cfg.addProperty(defCfg.IFRAME.key, { value:defCfg.IFRAME.value, handler:this.configIframe, validator:this.cfg.checkBoolean } );
-
-               /**
-               * The minimum selectable date in the current Calendar (mm/dd/yyyy)
-               * @config mindate
-               * @type String
-               * @default null
-               */
-               this.cfg.addProperty(defCfg.MINDATE.key, { value:defCfg.MINDATE.value, handler:this.configMinDate } );
-
-               /**
-               * The maximum selectable date in the current Calendar (mm/dd/yyyy)
-               * @config maxdate
-               * @type String
-               * @default null
-               */
-               this.cfg.addProperty(defCfg.MAXDATE.key, { value:defCfg.MAXDATE.value, handler:this.configMaxDate } );
-       
-       
-               // Options properties
-       
-               /**
-               * True if the Calendar should allow multiple selections. False by default.
-               * @config MULTI_SELECT
-               * @type Boolean
-               * @default false
-               */
-               this.cfg.addProperty(defCfg.MULTI_SELECT.key,   { value:defCfg.MULTI_SELECT.value, handler:this.configOptions, validator:this.cfg.checkBoolean } );
-       
-               /**
-               * The weekday the week begins on. Default is 0 (Sunday).
-               * @config START_WEEKDAY
-               * @type number
-               * @default 0
-               */
-               this.cfg.addProperty(defCfg.START_WEEKDAY.key,  { value:defCfg.START_WEEKDAY.value, handler:this.configOptions, validator:this.cfg.checkNumber  } );
-       
-               /**
-               * True if the Calendar should show weekday labels. True by default.
-               * @config SHOW_WEEKDAYS
-               * @type Boolean
-               * @default true
-               */
-               this.cfg.addProperty(defCfg.SHOW_WEEKDAYS.key,  { value:defCfg.SHOW_WEEKDAYS.value, handler:this.configOptions, validator:this.cfg.checkBoolean  } );
-       
-               /**
-               * True if the Calendar should show week row headers. False by default.
-               * @config SHOW_WEEK_HEADER
-               * @type Boolean
-               * @default false
-               */
-               this.cfg.addProperty(defCfg.SHOW_WEEK_HEADER.key, { value:defCfg.SHOW_WEEK_HEADER.value, handler:this.configOptions, validator:this.cfg.checkBoolean } );
-       
-               /**
-               * True if the Calendar should show week row footers. False by default.
-               * @config SHOW_WEEK_FOOTER
-               * @type Boolean
-               * @default false
-               */      
-               this.cfg.addProperty(defCfg.SHOW_WEEK_FOOTER.key,{ value:defCfg.SHOW_WEEK_FOOTER.value, handler:this.configOptions, validator:this.cfg.checkBoolean } );
-       
-               /**
-               * True if the Calendar should suppress weeks that are not a part of the current month. False by default.
-               * @config HIDE_BLANK_WEEKS
-               * @type Boolean
-               * @default false
-               */      
-               this.cfg.addProperty(defCfg.HIDE_BLANK_WEEKS.key, { value:defCfg.HIDE_BLANK_WEEKS.value, handler:this.configOptions, validator:this.cfg.checkBoolean } );
-               
-               /**
-               * The image that should be used for the left navigation arrow.
-               * @config NAV_ARROW_LEFT
-               * @type String
-               * @deprecated   You can customize the image by overriding the default CSS class for the left arrow - "calnavleft"  
-               * @default null
-               */      
-               this.cfg.addProperty(defCfg.NAV_ARROW_LEFT.key, { value:defCfg.NAV_ARROW_LEFT.value, handler:this.configOptions } );
-       
-               /**
-               * The image that should be used for the right navigation arrow.
-               * @config NAV_ARROW_RIGHT
-               * @type String
-               * @deprecated   You can customize the image by overriding the default CSS class for the right arrow - "calnavright"
-               * @default null
-               */      
-               this.cfg.addProperty(defCfg.NAV_ARROW_RIGHT.key, { value:defCfg.NAV_ARROW_RIGHT.value, handler:this.configOptions } );
-       
-               // Locale properties
-       
-               /**
-               * The short month labels for the current locale.
-               * @config MONTHS_SHORT
-               * @type String[]
-               * @default ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
-               */
-               this.cfg.addProperty(defCfg.MONTHS_SHORT.key,   { value:defCfg.MONTHS_SHORT.value, handler:this.configLocale } );
-               
-               /**
-               * The long month labels for the current locale.
-               * @config MONTHS_LONG
-               * @type String[]
-               * @default ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"
-               */      
-               this.cfg.addProperty(defCfg.MONTHS_LONG.key,            { value:defCfg.MONTHS_LONG.value, handler:this.configLocale } );
-
-               /**
-               * The 1-character weekday labels for the current locale.
-               * @config WEEKDAYS_1CHAR
-               * @type String[]
-               * @default ["S", "M", "T", "W", "T", "F", "S"]
-               */      
-               this.cfg.addProperty(defCfg.WEEKDAYS_1CHAR.key, { value:defCfg.WEEKDAYS_1CHAR.value, handler:this.configLocale } );
-               
-               /**
-               * The short weekday labels for the current locale.
-               * @config WEEKDAYS_SHORT
-               * @type String[]
-               * @default ["Su", "Mo", "Tu", "We", "Th", "Fr", "Sa"]
-               */      
-               this.cfg.addProperty(defCfg.WEEKDAYS_SHORT.key, { value:defCfg.WEEKDAYS_SHORT.value, handler:this.configLocale } );
-               
-               /**
-               * The medium weekday labels for the current locale.
-               * @config WEEKDAYS_MEDIUM
-               * @type String[]
-               * @default ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"]
-               */      
-               this.cfg.addProperty(defCfg.WEEKDAYS_MEDIUM.key,        { value:defCfg.WEEKDAYS_MEDIUM.value, handler:this.configLocale } );
-               
-               /**
-               * The long weekday labels for the current locale.
-               * @config WEEKDAYS_LONG
-               * @type String[]
-               * @default ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"]
-               */      
-               this.cfg.addProperty(defCfg.WEEKDAYS_LONG.key,  { value:defCfg.WEEKDAYS_LONG.value, handler:this.configLocale } );
-       
-               /**
-               * Refreshes the locale values used to build the Calendar.
-               * @method refreshLocale
-               * @private
-               */
-               var refreshLocale = function() {
-                       this.cfg.refireEvent(defCfg.LOCALE_MONTHS.key);
-                       this.cfg.refireEvent(defCfg.LOCALE_WEEKDAYS.key);
-               };
-       
-               this.cfg.subscribeToConfigEvent(defCfg.START_WEEKDAY.key, refreshLocale, this, true);
-               this.cfg.subscribeToConfigEvent(defCfg.MONTHS_SHORT.key, refreshLocale, this, true);
-               this.cfg.subscribeToConfigEvent(defCfg.MONTHS_LONG.key, refreshLocale, this, true);
-               this.cfg.subscribeToConfigEvent(defCfg.WEEKDAYS_1CHAR.key, refreshLocale, this, true);
-               this.cfg.subscribeToConfigEvent(defCfg.WEEKDAYS_SHORT.key, refreshLocale, this, true);
-               this.cfg.subscribeToConfigEvent(defCfg.WEEKDAYS_MEDIUM.key, refreshLocale, this, true);
-               this.cfg.subscribeToConfigEvent(defCfg.WEEKDAYS_LONG.key, refreshLocale, this, true);
-               
-               /**
-               * The setting that determines which length of month labels should be used. Possible values are "short" and "long".
-               * @config LOCALE_MONTHS
-               * @type String
-               * @default "long"
-               */      
-               this.cfg.addProperty(defCfg.LOCALE_MONTHS.key,  { value:defCfg.LOCALE_MONTHS.value, handler:this.configLocaleValues } );
-               
-               /**
-               * The setting that determines which length of weekday labels should be used. Possible values are "1char", "short", "medium", and "long".
-               * @config LOCALE_WEEKDAYS
-               * @type String
-               * @default "short"
-               */      
-               this.cfg.addProperty(defCfg.LOCALE_WEEKDAYS.key,        { value:defCfg.LOCALE_WEEKDAYS.value, handler:this.configLocaleValues } );
-       
-               /**
-               * The value used to delimit individual dates in a date string passed to various Calendar functions.
-               * @config DATE_DELIMITER
-               * @type String
-               * @default ","
-               */      
-               this.cfg.addProperty(defCfg.DATE_DELIMITER.key,         { value:defCfg.DATE_DELIMITER.value, handler:this.configLocale } );
-       
-               /**
-               * The value used to delimit date fields in a date string passed to various Calendar functions.
-               * @config DATE_FIELD_DELIMITER
-               * @type String
-               * @default "/"
-               */      
-               this.cfg.addProperty(defCfg.DATE_FIELD_DELIMITER.key, { value:defCfg.DATE_FIELD_DELIMITER.value, handler:this.configLocale } );
-       
-               /**
-               * The value used to delimit date ranges in a date string passed to various Calendar functions.
-               * @config DATE_RANGE_DELIMITER
-               * @type String
-               * @default "-"
-               */
-               this.cfg.addProperty(defCfg.DATE_RANGE_DELIMITER.key, { value:defCfg.DATE_RANGE_DELIMITER.value, handler:this.configLocale } );
-       
-               /**
-               * The position of the month in a month/year date string
-               * @config MY_MONTH_POSITION
-               * @type Number
-               * @default 1
-               */
-               this.cfg.addProperty(defCfg.MY_MONTH_POSITION.key,      { value:defCfg.MY_MONTH_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the year in a month/year date string
-               * @config MY_YEAR_POSITION
-               * @type Number
-               * @default 2
-               */
-               this.cfg.addProperty(defCfg.MY_YEAR_POSITION.key,       { value:defCfg.MY_YEAR_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the month in a month/day date string
-               * @config MD_MONTH_POSITION
-               * @type Number
-               * @default 1
-               */
-               this.cfg.addProperty(defCfg.MD_MONTH_POSITION.key,      { value:defCfg.MD_MONTH_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the day in a month/year date string
-               * @config MD_DAY_POSITION
-               * @type Number
-               * @default 2
-               */
-               this.cfg.addProperty(defCfg.MD_DAY_POSITION.key,                { value:defCfg.MD_DAY_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the month in a month/day/year date string
-               * @config MDY_MONTH_POSITION
-               * @type Number
-               * @default 1
-               */
-               this.cfg.addProperty(defCfg.MDY_MONTH_POSITION.key,     { value:defCfg.MDY_MONTH_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the day in a month/day/year date string
-               * @config MDY_DAY_POSITION
-               * @type Number
-               * @default 2
-               */
-               this.cfg.addProperty(defCfg.MDY_DAY_POSITION.key,       { value:defCfg.MDY_DAY_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the year in a month/day/year date string
-               * @config MDY_YEAR_POSITION
-               * @type Number
-               * @default 3
-               */
-               this.cfg.addProperty(defCfg.MDY_YEAR_POSITION.key,      { value:defCfg.MDY_YEAR_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The position of the month in the month year label string used as the Calendar header
-               * @config MY_LABEL_MONTH_POSITION
-               * @type Number
-               * @default 1
-               */
-               this.cfg.addProperty(defCfg.MY_LABEL_MONTH_POSITION.key,        { value:defCfg.MY_LABEL_MONTH_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the year in the month year label string used as the Calendar header
-               * @config MY_LABEL_YEAR_POSITION
-               * @type Number
-               * @default 2
-               */
-               this.cfg.addProperty(defCfg.MY_LABEL_YEAR_POSITION.key, { value:defCfg.MY_LABEL_YEAR_POSITION.value, handler:this.configLocale, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The suffix used after the month when rendering the Calendar header
-               * @config MY_LABEL_MONTH_SUFFIX
-               * @type String
-               * @default " "
-               */
-               this.cfg.addProperty(defCfg.MY_LABEL_MONTH_SUFFIX.key,  { value:defCfg.MY_LABEL_MONTH_SUFFIX.value, handler:this.configLocale } );
-               
-               /**
-               * The suffix used after the year when rendering the Calendar header
-               * @config MY_LABEL_YEAR_SUFFIX
-               * @type String
-               * @default ""
-               */
-               this.cfg.addProperty(defCfg.MY_LABEL_YEAR_SUFFIX.key, { value:defCfg.MY_LABEL_YEAR_SUFFIX.value, handler:this.configLocale } );
-
-               /**
-               * Configuration for the Month/Year CalendarNavigator UI which allows the user to jump directly to a 
-               * specific Month/Year without having to scroll sequentially through months.
-               * <p>
-               * Setting this property to null (default value) or false, will disable the CalendarNavigator UI.
-               * </p>
-               * <p>
-               * Setting this property to true will enable the CalendarNavigatior UI with the default CalendarNavigator configuration values.
-               * </p>
-               * <p>
-               * This property can also be set to an object literal containing configuration properties for the CalendarNavigator UI.
-               * The configuration object expects the the following case-sensitive properties, with the "strings" property being a nested object.
-               * Any properties which are not provided will use the default values (defined in the CalendarNavigator class).
-               * </p>
-               * <dl>
-               * <dt>strings</dt>
-               * <dd><em>Object</em> :  An object with the properties shown below, defining the string labels to use in the Navigator's UI
-               *     <dl>
-               *         <dt>month</dt><dd><em>String</em> : The string to use for the month label. Defaults to "Month".</dd>
-               *         <dt>year</dt><dd><em>String</em> : The string to use for the year label. Defaults to "Year".</dd>
-               *         <dt>submit</dt><dd><em>String</em> : The string to use for the submit button label. Defaults to "Okay".</dd>
-               *         <dt>cancel</dt><dd><em>String</em> : The string to use for the cancel button label. Defaults to "Cancel".</dd>
-               *         <dt>invalidYear</dt><dd><em>String</em> : The string to use for invalid year values. Defaults to "Year needs to be a number".</dd>
-               *     </dl>
-               * </dd>
-               * <dt>monthFormat</dt><dd><em>String</em> : The month format to use. Either YAHOO.widget.Calendar.LONG, or YAHOO.widget.Calendar.SHORT. Defaults to YAHOO.widget.Calendar.LONG</dd>
-               * <dt>initialFocus</dt><dd><em>String</em> : Either "year" or "month" specifying which input control should get initial focus. Defaults to "year"</dd>
-               * </dl>
-               * <p>E.g.</p>
-               * <pre>
-               * var navConfig = {
-               *         strings: {
-               *                 month:"Calendar Month",
-               *                 year:"Calendar Year",
-               *                 submit: "Submit",
-               *                 cancel: "Cancel",
-               *                 invalidYear: "Please enter a valid year"
-               *         },
-               *         monthFormat: YAHOO.widget.Calendar.SHORT,
-               *         initialFocus: "month"
-               * }
-               * </pre>
-               * @config navigator
-               * @type {Object|Boolean}
-               * @default null
-               */
-               this.cfg.addProperty(defCfg.NAV.key, { value:defCfg.NAV.value, handler:this.configNavigator } );
-       },
-
-       /**
-       * The default handler for the "pagedate" property
-       * @method configPageDate
-       */
-       configPageDate : function(type, args, obj) {
-               this.cfg.setProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key, this._parsePageDate(args[0]), true);
-       },
-
-       /**
-       * The default handler for the "mindate" property
-       * @method configMinDate
-       */
-       configMinDate : function(type, args, obj) {
-               var val = args[0];
-               if (YAHOO.lang.isString(val)) {
-                       val = this._parseDate(val);
-                       this.cfg.setProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.MINDATE.key, YAHOO.widget.DateMath.getDate(val[0],(val[1]-1),val[2]));
-               }
-       },
-
-       /**
-       * The default handler for the "maxdate" property
-       * @method configMaxDate
-       */
-       configMaxDate : function(type, args, obj) {
-               var val = args[0];
-               if (YAHOO.lang.isString(val)) {
-                       val = this._parseDate(val);
-                       this.cfg.setProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.MAXDATE.key, YAHOO.widget.DateMath.getDate(val[0],(val[1]-1),val[2]));
-               }
-       },
-       
-       /**
-       * The default handler for the "selected" property
-       * @method configSelected
-       */
-       configSelected : function(type, args, obj) {
-               var selected = args[0];
-               var cfgSelected = YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
-               
-               if (selected) {
-                       if (YAHOO.lang.isString(selected)) {
-                               this.cfg.setProperty(cfgSelected, this._parseDates(selected), true);
-                       } 
-               }
-               if (! this._selectedDates) {
-                       this._selectedDates = this.cfg.getProperty(cfgSelected);
-               }
-       },
-       
-       /**
-       * The default handler for all configuration options properties
-       * @method configOptions
-       */
-       configOptions : function(type, args, obj) {
-               this.Options[type.toUpperCase()] = args[0];
-       },
-       
-       /**
-       * The default handler for all configuration locale properties
-       * @method configLocale
-       */
-       configLocale : function(type, args, obj) {
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-               this.Locale[type.toUpperCase()] = args[0];
-       
-               this.cfg.refireEvent(defCfg.LOCALE_MONTHS.key);
-               this.cfg.refireEvent(defCfg.LOCALE_WEEKDAYS.key);
-       },
-       
-       /**
-       * The default handler for all configuration locale field length properties
-       * @method configLocaleValues
-       */
-       configLocaleValues : function(type, args, obj) {
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG; 
-       
-               type = type.toLowerCase();
-               var val = args[0];
-       
-               switch (type) {
-                       case defCfg.LOCALE_MONTHS.key:
-                               switch (val) {
-                                       case YAHOO.widget.Calendar.SHORT:
-                                               this.Locale.LOCALE_MONTHS = this.cfg.getProperty(defCfg.MONTHS_SHORT.key).concat();
-                                               break;
-                                       case YAHOO.widget.Calendar.LONG:
-                                               this.Locale.LOCALE_MONTHS = this.cfg.getProperty(defCfg.MONTHS_LONG.key).concat();
-                                               break;
-                               }
-                               break;
-                       case defCfg.LOCALE_WEEKDAYS.key:
-                               switch (val) {
-                                       case YAHOO.widget.Calendar.ONE_CHAR:
-                                               this.Locale.LOCALE_WEEKDAYS = this.cfg.getProperty(defCfg.WEEKDAYS_1CHAR.key).concat();
-                                               break;
-                                       case YAHOO.widget.Calendar.SHORT:
-                                               this.Locale.LOCALE_WEEKDAYS = this.cfg.getProperty(defCfg.WEEKDAYS_SHORT.key).concat();
-                                               break;
-                                       case YAHOO.widget.Calendar.MEDIUM:
-                                               this.Locale.LOCALE_WEEKDAYS = this.cfg.getProperty(defCfg.WEEKDAYS_MEDIUM.key).concat();
-                                               break;
-                                       case YAHOO.widget.Calendar.LONG:
-                                               this.Locale.LOCALE_WEEKDAYS = this.cfg.getProperty(defCfg.WEEKDAYS_LONG.key).concat();
-                                               break;
-                               }
-                               
-                               var START_WEEKDAY = this.cfg.getProperty(defCfg.START_WEEKDAY.key);
-       
-                               if (START_WEEKDAY > 0) {
-                                       for (var w=0;w<START_WEEKDAY;++w) {
-                                               this.Locale.LOCALE_WEEKDAYS.push(this.Locale.LOCALE_WEEKDAYS.shift());
-                                       }
-                               }
-                               break;
-               }
-       },
-
-       /**
-        * The default handler for the "navigator" property
-        * @method configNavigator
-        */
-       configNavigator : function(type, args, obj) {
-               var val = args[0];
-               if (YAHOO.widget.CalendarNavigator && (val === true || YAHOO.lang.isObject(val))) {
-                       if (!this.oNavigator) {
-                               this.oNavigator = new YAHOO.widget.CalendarNavigator(this);
-                               // Cleanup DOM Refs/Events before innerHTML is removed.
-                               function erase() {
-                                       if (!this.pages) {
-                                               this.oNavigator.erase();
-                                       }
-                               }
-                               this.beforeRenderEvent.subscribe(erase, this, true);
-                       }
-               } else {
-                       if (this.oNavigator) {
-                               this.oNavigator.destroy();
-                               this.oNavigator = null;
-                       }
-               }
-       },
-
-       /**
-       * Defines the style constants for the Calendar
-       * @method initStyles
-       */
-       initStyles : function() {
-
-               var defStyle = YAHOO.widget.Calendar._STYLES;
-
-               this.Style = {
-                       /**
-                       * @property Style.CSS_ROW_HEADER
-                       */
-                       CSS_ROW_HEADER: defStyle.CSS_ROW_HEADER,
-                       /**
-                       * @property Style.CSS_ROW_FOOTER
-                       */
-                       CSS_ROW_FOOTER: defStyle.CSS_ROW_FOOTER,
-                       /**
-                       * @property Style.CSS_CELL
-                       */
-                       CSS_CELL : defStyle.CSS_CELL,
-                       /**
-                       * @property Style.CSS_CELL_SELECTOR
-                       */
-                       CSS_CELL_SELECTOR : defStyle.CSS_CELL_SELECTOR,
-                       /**
-                       * @property Style.CSS_CELL_SELECTED
-                       */
-                       CSS_CELL_SELECTED : defStyle.CSS_CELL_SELECTED,
-                       /**
-                       * @property Style.CSS_CELL_SELECTABLE
-                       */
-                       CSS_CELL_SELECTABLE : defStyle.CSS_CELL_SELECTABLE,
-                       /**
-                       * @property Style.CSS_CELL_RESTRICTED
-                       */
-                       CSS_CELL_RESTRICTED : defStyle.CSS_CELL_RESTRICTED,
-                       /**
-                       * @property Style.CSS_CELL_TODAY
-                       */
-                       CSS_CELL_TODAY : defStyle.CSS_CELL_TODAY,
-                       /**
-                       * @property Style.CSS_CELL_OOM
-                       */
-                       CSS_CELL_OOM : defStyle.CSS_CELL_OOM,
-                       /**
-                       * @property Style.CSS_CELL_OOB
-                       */
-                       CSS_CELL_OOB : defStyle.CSS_CELL_OOB,
-                       /**
-                       * @property Style.CSS_HEADER
-                       */
-                       CSS_HEADER : defStyle.CSS_HEADER,
-                       /**
-                       * @property Style.CSS_HEADER_TEXT
-                       */
-                       CSS_HEADER_TEXT : defStyle.CSS_HEADER_TEXT,
-                       /**
-                       * @property Style.CSS_BODY
-                       */
-                       CSS_BODY : defStyle.CSS_BODY,
-                       /**
-                       * @property Style.CSS_WEEKDAY_CELL
-                       */
-                       CSS_WEEKDAY_CELL : defStyle.CSS_WEEKDAY_CELL,
-                       /**
-                       * @property Style.CSS_WEEKDAY_ROW
-                       */
-                       CSS_WEEKDAY_ROW : defStyle.CSS_WEEKDAY_ROW,
-                       /**
-                       * @property Style.CSS_FOOTER
-                       */
-                       CSS_FOOTER : defStyle.CSS_FOOTER,
-                       /**
-                       * @property Style.CSS_CALENDAR
-                       */
-                       CSS_CALENDAR : defStyle.CSS_CALENDAR,
-                       /**
-                       * @property Style.CSS_SINGLE
-                       */
-                       CSS_SINGLE : defStyle.CSS_SINGLE,
-                       /**
-                       * @property Style.CSS_CONTAINER
-                       */
-                       CSS_CONTAINER : defStyle.CSS_CONTAINER,
-                       /**
-                       * @property Style.CSS_NAV_LEFT
-                       */
-                       CSS_NAV_LEFT : defStyle.CSS_NAV_LEFT,
-                       /**
-                       * @property Style.CSS_NAV_RIGHT
-                       */
-                       CSS_NAV_RIGHT : defStyle.CSS_NAV_RIGHT,
-                       /**
-                       * @property Style.CSS_NAV
-                       */
-                       CSS_NAV : defStyle.CSS_NAV,
-                       /**
-                       * @property Style.CSS_CLOSE
-                       */
-                       CSS_CLOSE : defStyle.CSS_CLOSE,
-                       /**
-                       * @property Style.CSS_CELL_TOP
-                       */
-                       CSS_CELL_TOP : defStyle.CSS_CELL_TOP,
-                       /**
-                       * @property Style.CSS_CELL_LEFT
-                       */
-                       CSS_CELL_LEFT : defStyle.CSS_CELL_LEFT,
-                       /**
-                       * @property Style.CSS_CELL_RIGHT
-                       */
-                       CSS_CELL_RIGHT : defStyle.CSS_CELL_RIGHT,
-                       /**
-                       * @property Style.CSS_CELL_BOTTOM
-                       */
-                       CSS_CELL_BOTTOM : defStyle.CSS_CELL_BOTTOM,
-                       /**
-                       * @property Style.CSS_CELL_HOVER
-                       */
-                       CSS_CELL_HOVER : defStyle.CSS_CELL_HOVER,
-                       /**
-                       * @property Style.CSS_CELL_HIGHLIGHT1
-                       */
-                       CSS_CELL_HIGHLIGHT1 : defStyle.CSS_CELL_HIGHLIGHT1,
-                       /**
-                       * @property Style.CSS_CELL_HIGHLIGHT2
-                       */
-                       CSS_CELL_HIGHLIGHT2 : defStyle.CSS_CELL_HIGHLIGHT2,
-                       /**
-                       * @property Style.CSS_CELL_HIGHLIGHT3
-                       */
-                       CSS_CELL_HIGHLIGHT3 : defStyle.CSS_CELL_HIGHLIGHT3,
-                       /**
-                       * @property Style.CSS_CELL_HIGHLIGHT4
-                       */
-                       CSS_CELL_HIGHLIGHT4 : defStyle.CSS_CELL_HIGHLIGHT4
-               };
-       },
-       
-       /**
-       * Builds the date label that will be displayed in the calendar header or
-       * footer, depending on configuration.
-       * @method buildMonthLabel
-       * @return       {String}        The formatted calendar month label
-       */
-       buildMonthLabel : function() {
-               var pageDate = this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key);
-       
-               var monthLabel  = this.Locale.LOCALE_MONTHS[pageDate.getMonth()] + this.Locale.MY_LABEL_MONTH_SUFFIX;
-               var yearLabel = pageDate.getFullYear() + this.Locale.MY_LABEL_YEAR_SUFFIX;
-
-               if (this.Locale.MY_LABEL_MONTH_POSITION == 2 || this.Locale.MY_LABEL_YEAR_POSITION == 1) {
-                       return yearLabel + monthLabel;
-               } else {
-                       return monthLabel + yearLabel;
-               }
-       },
-       
-       /**
-       * Builds the date digit that will be displayed in calendar cells
-       * @method buildDayLabel
-       * @param {Date} workingDate     The current working date
-       * @return       {String}        The formatted day label
-       */
-       buildDayLabel : function(workingDate) {
-               return workingDate.getDate();
-       },
-       
-       /**
-        * Creates the title bar element and adds it to Calendar container DIV
-        * 
-        * @method createTitleBar
-        * @param {String} strTitle The title to display in the title bar
-        * @return The title bar element
-        */
-       createTitleBar : function(strTitle) {
-               var tDiv = YAHOO.util.Dom.getElementsByClassName(YAHOO.widget.CalendarGroup.CSS_2UPTITLE, "div", this.oDomContainer)[0] || document.createElement("div");
-               tDiv.className = YAHOO.widget.CalendarGroup.CSS_2UPTITLE;
-               tDiv.innerHTML = strTitle;
-               this.oDomContainer.insertBefore(tDiv, this.oDomContainer.firstChild);
-       
-               YAHOO.util.Dom.addClass(this.oDomContainer, "withtitle");
-       
-               return tDiv;
-       },
-       
-       /**
-        * Removes the title bar element from the DOM
-        * 
-        * @method removeTitleBar
-        */
-       removeTitleBar : function() {
-               var tDiv = YAHOO.util.Dom.getElementsByClassName(YAHOO.widget.CalendarGroup.CSS_2UPTITLE, "div", this.oDomContainer)[0] || null;
-               if (tDiv) {
-                       YAHOO.util.Event.purgeElement(tDiv);
-                       this.oDomContainer.removeChild(tDiv);
-               }
-               YAHOO.util.Dom.removeClass(this.oDomContainer, "withtitle");
-       },
-       
-       /**
-        * Creates the close button HTML element and adds it to Calendar container DIV
-        * 
-        * @method createCloseButton
-        * @return The close HTML element created
-        */
-       createCloseButton : function() {
-               var Dom = YAHOO.util.Dom,
-                       Event = YAHOO.util.Event,
-                       cssClose = YAHOO.widget.CalendarGroup.CSS_2UPCLOSE,
-                       DEPR_CLOSE_PATH = "us/my/bn/x_d.gif";
-       
-               var lnk = Dom.getElementsByClassName("link-close", "a", this.oDomContainer)[0];
-       
-               if (!lnk) {
-                       lnk = document.createElement("a");  
-                       Event.addListener(lnk, "click", function(e, cal) {
-                               cal.hide(); 
-                               Event.preventDefault(e);
-                       }, this);        
-               }
-       
-               lnk.href = "#";
-               lnk.className = "link-close";
-       
-               if (YAHOO.widget.Calendar.IMG_ROOT !== null) {
-                       var img = Dom.getElementsByClassName(cssClose, "img", lnk)[0] || document.createElement("img");
-                       img.src = YAHOO.widget.Calendar.IMG_ROOT + DEPR_CLOSE_PATH;
-                       img.className = cssClose;
-                       lnk.appendChild(img);
-               } else {
-                       lnk.innerHTML = '<span class="' + cssClose + ' ' + this.Style.CSS_CLOSE + '"></span>';
-               }
-               this.oDomContainer.appendChild(lnk);
-       
-               return lnk;
-       },
-       
-       /**
-        * Removes the close button HTML element from the DOM
-        * 
-        * @method removeCloseButton
-        */
-       removeCloseButton : function() {
-               var btn = YAHOO.util.Dom.getElementsByClassName("link-close", "a", this.oDomContainer)[0] || null;
-               if (btn) {
-                       YAHOO.util.Event.purgeElement(btn);
-                       this.oDomContainer.removeChild(btn);
-               }
-       },
-
-       /**
-       * Renders the calendar header.
-       * @method renderHeader
-       * @param {Array}        html    The current working HTML array
-       * @return {Array} The current working HTML array
-       */
-       renderHeader : function(html) {
-               var colSpan = 7;
-               
-               var DEPR_NAV_LEFT = "us/tr/callt.gif";
-               var DEPR_NAV_RIGHT = "us/tr/calrt.gif"; 
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-               
-               if (this.cfg.getProperty(defCfg.SHOW_WEEK_HEADER.key)) {
-                       colSpan += 1;
-               }
-       
-               if (this.cfg.getProperty(defCfg.SHOW_WEEK_FOOTER.key)) {
-                       colSpan += 1;
-               }
-       
-               html[html.length] = "<thead>";
-               html[html.length] =             "<tr>";
-               html[html.length] =                     '<th colspan="' + colSpan + '" class="' + this.Style.CSS_HEADER_TEXT + '">';
-               html[html.length] =                             '<div class="' + this.Style.CSS_HEADER + '">';
-       
-               var renderLeft, renderRight = false;
-       
-               if (this.parent) {
-                       if (this.index === 0) {
-                               renderLeft = true;
-                       }
-                       if (this.index == (this.parent.cfg.getProperty("pages") -1)) {
-                               renderRight = true;
-                       }
-               } else {
-                       renderLeft = true;
-                       renderRight = true;
-               }
-       
-               if (renderLeft) {
-                       var leftArrow = this.cfg.getProperty(defCfg.NAV_ARROW_LEFT.key);
-                       // Check for deprecated customization - If someone set IMG_ROOT, but didn't set NAV_ARROW_LEFT, then set NAV_ARROW_LEFT to the old deprecated value
-                       if (leftArrow === null && YAHOO.widget.Calendar.IMG_ROOT !== null) {
-                               leftArrow = YAHOO.widget.Calendar.IMG_ROOT + DEPR_NAV_LEFT;
-                       }
-                       var leftStyle = (leftArrow === null) ? "" : ' style="background-image:url(' + leftArrow + ')"';
-                       html[html.length] = '<a class="' + this.Style.CSS_NAV_LEFT + '"' + leftStyle + ' >&#160;</a>';
-               }
-
-               var lbl = this.buildMonthLabel();
-               var cal = this.parent || this;
-               if (cal.cfg.getProperty("navigator")) {
-                       lbl = "<a class=\"" + this.Style.CSS_NAV + "\" href=\"#\">" + lbl + "</a>";
-               }
-               html[html.length] = lbl;
-
-               if (renderRight) {
-                       var rightArrow = this.cfg.getProperty(defCfg.NAV_ARROW_RIGHT.key);
-                       if (rightArrow === null && YAHOO.widget.Calendar.IMG_ROOT !== null) {
-                               rightArrow = YAHOO.widget.Calendar.IMG_ROOT + DEPR_NAV_RIGHT;
-                       }
-                       var rightStyle = (rightArrow === null) ? "" : ' style="background-image:url(' + rightArrow + ')"';
-                       html[html.length] = '<a class="' + this.Style.CSS_NAV_RIGHT + '"' + rightStyle + ' >&#160;</a>';
-               }
-
-               html[html.length] =     '</div>\n</th>\n</tr>';
-
-               if (this.cfg.getProperty(defCfg.SHOW_WEEKDAYS.key)) {
-                       html = this.buildWeekdays(html);
-               }
-               
-               html[html.length] = '</thead>';
-       
-               return html;
-       },
-       
-       /**
-       * Renders the Calendar's weekday headers.
-       * @method buildWeekdays
-       * @param {Array}        html    The current working HTML array
-       * @return {Array} The current working HTML array
-       */
-       buildWeekdays : function(html) {
-       
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-       
-               html[html.length] = '<tr class="' + this.Style.CSS_WEEKDAY_ROW + '">';
-       
-               if (this.cfg.getProperty(defCfg.SHOW_WEEK_HEADER.key)) {
-                       html[html.length] = '<th>&#160;</th>';
-               }
-       
-               for(var i=0;i<this.Locale.LOCALE_WEEKDAYS.length;++i) {
-                       html[html.length] = '<th class="calweekdaycell">' + this.Locale.LOCALE_WEEKDAYS[i] + '</th>';
-               }
-       
-               if (this.cfg.getProperty(defCfg.SHOW_WEEK_FOOTER.key)) {
-                       html[html.length] = '<th>&#160;</th>';
-               }
-       
-               html[html.length] = '</tr>';
-       
-               return html;
-       },
-       
-       /**
-       * Renders the calendar body.
-       * @method renderBody
-       * @param {Date} workingDate     The current working Date being used for the render process
-       * @param {Array}        html    The current working HTML array
-       * @return {Array} The current working HTML array
-       */
-       renderBody : function(workingDate, html) {
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-       
-               var startDay = this.cfg.getProperty(defCfg.START_WEEKDAY.key);
-       
-               this.preMonthDays = workingDate.getDay();
-               if (startDay > 0) {
-                       this.preMonthDays -= startDay;
-               }
-               if (this.preMonthDays < 0) {
-                       this.preMonthDays += 7;
-               }
-               
-               this.monthDays = YAHOO.widget.DateMath.findMonthEnd(workingDate).getDate();
-               this.postMonthDays = YAHOO.widget.Calendar.DISPLAY_DAYS-this.preMonthDays-this.monthDays;
-               
-               workingDate = YAHOO.widget.DateMath.subtract(workingDate, YAHOO.widget.DateMath.DAY, this.preMonthDays);
-       
-               var weekNum,weekClass;
-               var weekPrefix = "w";
-               var cellPrefix = "_cell";
-               var workingDayPrefix = "wd";
-               var dayPrefix = "d";
-               
-               var cellRenderers;
-               var renderer;
-               
-               var todayYear = this.today.getFullYear();
-               var todayMonth = this.today.getMonth();
-               var todayDate = this.today.getDate();
-               
-               var useDate = this.cfg.getProperty(defCfg.PAGEDATE.key);
-               var hideBlankWeeks = this.cfg.getProperty(defCfg.HIDE_BLANK_WEEKS.key);
-               var showWeekFooter = this.cfg.getProperty(defCfg.SHOW_WEEK_FOOTER.key);
-               var showWeekHeader = this.cfg.getProperty(defCfg.SHOW_WEEK_HEADER.key);
-               var mindate = this.cfg.getProperty(defCfg.MINDATE.key);
-               var maxdate = this.cfg.getProperty(defCfg.MAXDATE.key);
-       
-               if (mindate) {
-                       mindate = YAHOO.widget.DateMath.clearTime(mindate);
-               }
-               if (maxdate) {
-                       maxdate = YAHOO.widget.DateMath.clearTime(maxdate);
-               }
-               
-               html[html.length] = '<tbody class="m' + (useDate.getMonth()+1) + ' ' + this.Style.CSS_BODY + '">';
-               
-               var i = 0;
-       
-               var tempDiv = document.createElement("div");
-               var cell = document.createElement("td");
-               tempDiv.appendChild(cell);
-       
-               var cal = this.parent || this;
-       
-               for (var r=0;r<6;r++) {
-       
-                       weekNum = YAHOO.widget.DateMath.getWeekNumber(workingDate, useDate.getFullYear(), startDay);
-                       weekClass = weekPrefix + weekNum;
-       
-                       // Local OOM check for performance, since we already have pagedate
-                       if (r !== 0 && hideBlankWeeks === true && workingDate.getMonth() != useDate.getMonth()) {
-                               break;
-                       } else {
-       
-                               html[html.length] = '<tr class="' + weekClass + '">';
-                               
-                               if (showWeekHeader) { html = this.renderRowHeader(weekNum, html); }
-                               
-                               for (var d=0;d<7;d++){ // Render actual days
-       
-                                       cellRenderers = [];
-       
-                                       this.clearElement(cell);
-                                       cell.className = this.Style.CSS_CELL;
-                                       cell.id = this.id + cellPrefix + i;
-
-                                       if (workingDate.getDate()               == todayDate && 
-                                               workingDate.getMonth()          == todayMonth &&
-                                               workingDate.getFullYear()       == todayYear) {
-                                               cellRenderers[cellRenderers.length]=cal.renderCellStyleToday;
-                                       }
-                                       
-                                       var workingArray = [workingDate.getFullYear(),workingDate.getMonth()+1,workingDate.getDate()];
-                                       this.cellDates[this.cellDates.length] = workingArray; // Add this date to cellDates
-                                       
-                                       // Local OOM check for performance, since we already have pagedate
-                                       if (workingDate.getMonth() != useDate.getMonth()) {
-                                               cellRenderers[cellRenderers.length]=cal.renderCellNotThisMonth;
-                                       } else {
-                                               YAHOO.util.Dom.addClass(cell, workingDayPrefix + workingDate.getDay());
-                                               YAHOO.util.Dom.addClass(cell, dayPrefix + workingDate.getDate());
-                                       
-                                               for (var s=0;s<this.renderStack.length;++s) {
-       
-                                                       renderer = null;
-       
-                                                       var rArray = this.renderStack[s];
-                                                       var type = rArray[0];
-                                                       
-                                                       var month;
-                                                       var day;
-                                                       var year;
-                                                       
-                                                       switch (type) {
-                                                               case YAHOO.widget.Calendar.DATE:
-                                                                       month = rArray[1][1];
-                                                                       day = rArray[1][2];
-                                                                       year = rArray[1][0];
-       
-                                                                       if (workingDate.getMonth()+1 == month && workingDate.getDate() == day && workingDate.getFullYear() == year) {
-                                                                               renderer = rArray[2];
-                                                                               this.renderStack.splice(s,1);
-                                                                       }
-                                                                       break;
-                                                               case YAHOO.widget.Calendar.MONTH_DAY:
-                                                                       month = rArray[1][0];
-                                                                       day = rArray[1][1];
-                                                                       
-                                                                       if (workingDate.getMonth()+1 == month && workingDate.getDate() == day) {
-                                                                               renderer = rArray[2];
-                                                                               this.renderStack.splice(s,1);
-                                                                       }
-                                                                       break;
-                                                               case YAHOO.widget.Calendar.RANGE:
-                                                                       var date1 = rArray[1][0];
-                                                                       var date2 = rArray[1][1];
-       
-                                                                       var d1month = date1[1];
-                                                                       var d1day = date1[2];
-                                                                       var d1year = date1[0];
-                                                                       
-                                                                       var d1 = YAHOO.widget.DateMath.getDate(d1year, d1month-1, d1day);
-       
-                                                                       var d2month = date2[1];
-                                                                       var d2day = date2[2];
-                                                                       var d2year = date2[0];
-       
-                                                                       var d2 = YAHOO.widget.DateMath.getDate(d2year, d2month-1, d2day);
-       
-                                                                       if (workingDate.getTime() >= d1.getTime() && workingDate.getTime() <= d2.getTime()) {
-                                                                               renderer = rArray[2];
-       
-                                                                               if (workingDate.getTime()==d2.getTime()) { 
-                                                                                       this.renderStack.splice(s,1);
-                                                                               }
-                                                                       }
-                                                                       break;
-                                                               case YAHOO.widget.Calendar.WEEKDAY:
-                                                                       
-                                                                       var weekday = rArray[1][0];
-                                                                       if (workingDate.getDay()+1 == weekday) {
-                                                                               renderer = rArray[2];
-                                                                       }
-                                                                       break;
-                                                               case YAHOO.widget.Calendar.MONTH:
-                                                                       
-                                                                       month = rArray[1][0];
-                                                                       if (workingDate.getMonth()+1 == month) {
-                                                                               renderer = rArray[2];
-                                                                       }
-                                                                       break;
-                                                       }
-                                                       
-                                                       if (renderer) {
-                                                               cellRenderers[cellRenderers.length]=renderer;
-                                                       }
-                                               }
-       
-                                       }
-       
-                                       if (this._indexOfSelectedFieldArray(workingArray) > -1) {
-                                               cellRenderers[cellRenderers.length]=cal.renderCellStyleSelected; 
-                                       }
-       
-                                       if ((mindate && (workingDate.getTime() < mindate.getTime())) ||
-                                               (maxdate && (workingDate.getTime() > maxdate.getTime()))
-                                       ) {
-                                               cellRenderers[cellRenderers.length]=cal.renderOutOfBoundsDate;
-                                       } else {
-                                               cellRenderers[cellRenderers.length]=cal.styleCellDefault;
-                                               cellRenderers[cellRenderers.length]=cal.renderCellDefault;      
-                                       }
-                                       
-                                       for (var x=0; x < cellRenderers.length; ++x) {
-                                               if (cellRenderers[x].call(cal, workingDate, cell) == YAHOO.widget.Calendar.STOP_RENDER) {
-                                                       break;
-                                               }
-                                       }
-       
-                                       workingDate.setTime(workingDate.getTime() + YAHOO.widget.DateMath.ONE_DAY_MS);
-       
-                                       if (i >= 0 && i <= 6) {
-                                               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_TOP);
-                                       }
-                                       if ((i % 7) === 0) {
-                                               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_LEFT);
-                                       }
-                                       if (((i+1) % 7) === 0) {
-                                               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_RIGHT);
-                                       }
-                                       
-                                       var postDays = this.postMonthDays; 
-                                       if (hideBlankWeeks && postDays >= 7) {
-                                               var blankWeeks = Math.floor(postDays/7);
-                                               for (var p=0;p<blankWeeks;++p) {
-                                                       postDays -= 7;
-                                               }
-                                       }
-                                       
-                                       if (i >= ((this.preMonthDays+postDays+this.monthDays)-7)) {
-                                               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_BOTTOM);
-                                       }
-       
-                                       html[html.length] = tempDiv.innerHTML;
-                                       i++;
-                               }
-       
-                               if (showWeekFooter) { html = this.renderRowFooter(weekNum, html); }
-       
-                               html[html.length] = '</tr>';
-                       }
-               }
-       
-               html[html.length] = '</tbody>';
-       
-               return html;
-       },
-       
-       /**
-       * Renders the calendar footer. In the default implementation, there is
-       * no footer.
-       * @method renderFooter
-       * @param {Array}        html    The current working HTML array
-       * @return {Array} The current working HTML array
-       */
-       renderFooter : function(html) { return html; },
-       
-       /**
-       * Renders the calendar after it has been configured. The render() method has a specific call chain that will execute
-       * when the method is called: renderHeader, renderBody, renderFooter.
-       * Refer to the documentation for those methods for information on 
-       * individual render tasks.
-       * @method render
-       */
-       render : function() {
-               this.beforeRenderEvent.fire();
-       
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-       
-               // Find starting day of the current month
-               var workingDate = YAHOO.widget.DateMath.findMonthStart(this.cfg.getProperty(defCfg.PAGEDATE.key));
-       
-               this.resetRenderers();
-               this.cellDates.length = 0;
-
-               YAHOO.util.Event.purgeElement(this.oDomContainer, true);
-
-               var html = [];
-       
-               html[html.length] = '<table cellSpacing="0" class="' + this.Style.CSS_CALENDAR + ' y' + workingDate.getFullYear() + '" id="' + this.id + '">';
-               html = this.renderHeader(html);
-               html = this.renderBody(workingDate, html);
-               html = this.renderFooter(html);
-               html[html.length] = '</table>';
-
-               this.oDomContainer.innerHTML = html.join("\n");
-
-               this.applyListeners();
-               this.cells = this.oDomContainer.getElementsByTagName("td");
-       
-               this.cfg.refireEvent(defCfg.TITLE.key);
-               this.cfg.refireEvent(defCfg.CLOSE.key);
-               this.cfg.refireEvent(defCfg.IFRAME.key);
-       
-               this.renderEvent.fire();
-       },
-
-       /**
-       * Applies the Calendar's DOM listeners to applicable elements.
-       * @method applyListeners
-       */
-       applyListeners : function() {
-               var root = this.oDomContainer;
-               var cal = this.parent || this;
-               var anchor = "a";
-               var mousedown = "mousedown";
-
-               var linkLeft = YAHOO.util.Dom.getElementsByClassName(this.Style.CSS_NAV_LEFT, anchor, root);
-               var linkRight = YAHOO.util.Dom.getElementsByClassName(this.Style.CSS_NAV_RIGHT, anchor, root);
-       
-               if (linkLeft && linkLeft.length > 0) {
-                       this.linkLeft = linkLeft[0];
-                       YAHOO.util.Event.addListener(this.linkLeft, mousedown, cal.previousMonth, cal, true);
-               }
-
-               if (linkRight && linkRight.length > 0) {
-                       this.linkRight = linkRight[0];
-                       YAHOO.util.Event.addListener(this.linkRight, mousedown, cal.nextMonth, cal, true);
-               }
-
-               if (cal.cfg.getProperty("navigator") !== null) {
-                       this.applyNavListeners();
-               }
-
-               if (this.domEventMap) {
-                       var el,elements;
-                       for (var cls in this.domEventMap) {     
-                               if (YAHOO.lang.hasOwnProperty(this.domEventMap, cls)) {
-                                       var items = this.domEventMap[cls];
-       
-                                       if (! (items instanceof Array)) {
-                                               items = [items];
-                                       }
-       
-                                       for (var i=0;i<items.length;i++)        {
-                                               var item = items[i];
-                                               elements = YAHOO.util.Dom.getElementsByClassName(cls, item.tag, this.oDomContainer);
-       
-                                               for (var c=0;c<elements.length;c++) {
-                                                       el = elements[c];
-                                                        YAHOO.util.Event.addListener(el, item.event, item.handler, item.scope, item.correct );
-                                               }
-                                       }
-                               }
-                       }
-               }
-       
-               YAHOO.util.Event.addListener(this.oDomContainer, "click", this.doSelectCell, this);
-               YAHOO.util.Event.addListener(this.oDomContainer, "mouseover", this.doCellMouseOver, this);
-               YAHOO.util.Event.addListener(this.oDomContainer, "mouseout", this.doCellMouseOut, this);
-       },
-
-       applyNavListeners : function() {
-
-               var E = YAHOO.util.Event;
-
-               var calParent = this.parent || this;
-               var cal = this;
-
-               var navBtns = YAHOO.util.Dom.getElementsByClassName(this.Style.CSS_NAV, "a", this.oDomContainer);
-
-               if (navBtns.length > 0) {
-
-                       function show(e, obj) {
-                               var target = E.getTarget(e);
-                               // this == navBtn
-                               if (this === target || YAHOO.util.Dom.isAncestor(this, target)) {
-                                       E.preventDefault(e);
-                               }
-                               var navigator = calParent.oNavigator;
-                               if (navigator) {
-                                       var pgdate = cal.cfg.getProperty("pagedate");
-                                       navigator.setYear(pgdate.getFullYear());
-                                       navigator.setMonth(pgdate.getMonth());
-                                       navigator.show();
-                               }
-                       }
-                       E.addListener(navBtns, "click", show);
-               }
-       },
-
-       /**
-       * Retrieves the Date object for the specified Calendar cell
-       * @method getDateByCellId
-       * @param {String}       id      The id of the cell
-       * @return {Date} The Date object for the specified Calendar cell
-       */
-       getDateByCellId : function(id) {
-               var date = this.getDateFieldsByCellId(id);
-               return YAHOO.widget.DateMath.getDate(date[0],date[1]-1,date[2]);
-       },
-       
-       /**
-       * Retrieves the Date object for the specified Calendar cell
-       * @method getDateFieldsByCellId
-       * @param {String}       id      The id of the cell
-       * @return {Array}       The array of Date fields for the specified Calendar cell
-       */
-       getDateFieldsByCellId : function(id) {
-               id = id.toLowerCase().split("_cell")[1];
-               id = parseInt(id, 10);
-               return this.cellDates[id];
-       },
-       
-       /**
-        * Find the Calendar's cell index for a given date.
-        * If the date is not found, the method returns -1.
-        * <p>
-        * The returned index can be used to lookup the cell HTMLElement  
-        * using the Calendar's cells array or passed to selectCell to select 
-        * cells by index. 
-        * </p>
-        *
-        * See <a href="#cells">cells</a>, <a href="#selectCell">selectCell</a>.
-        *
-        * @method getCellIndex
-        * @param {Date} date JavaScript Date object, for which to find a cell index.
-        * @return {Number} The index of the date in Calendars cellDates/cells arrays, or -1 if the date 
-        * is not on the curently rendered Calendar page.
-        */
-       getCellIndex : function(date) {
-               var idx = -1;
-               if (date) {
-                       var m = date.getMonth(),
-                               y = date.getFullYear(),
-                               d = date.getDate(),
-                               dates = this.cellDates;
-
-                       for (var i = 0; i < dates.length; ++i) {
-                               var cellDate = dates[i];
-                               if (cellDate[0] === y && cellDate[1] === m+1 && cellDate[2] === d) {
-                                       idx = i;
-                                       break;
-                               }
-                       }
-               }
-               return idx;
-       },
-       
-       // BEGIN BUILT-IN TABLE CELL RENDERERS
-       
-       /**
-       * Renders a cell that falls before the minimum date or after the maximum date.
-       * widget class.
-       * @method renderOutOfBoundsDate
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       * @return {String} YAHOO.widget.Calendar.STOP_RENDER if rendering should stop with this style, null or nothing if rendering
-       *                       should not be terminated
-       */
-       renderOutOfBoundsDate : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_OOB);
-               cell.innerHTML = workingDate.getDate();
-               return YAHOO.widget.Calendar.STOP_RENDER;
-       },
-       
-       /**
-       * Renders the row header for a week.
-       * @method renderRowHeader
-       * @param {Number}       weekNum The week number of the current row
-       * @param {Array}        cell    The current working HTML array
-       */
-       renderRowHeader : function(weekNum, html) {
-               html[html.length] = '<th class="calrowhead">' + weekNum + '</th>';
-               return html;
-       },
-       
-       /**
-       * Renders the row footer for a week.
-       * @method renderRowFooter
-       * @param {Number}       weekNum The week number of the current row
-       * @param {Array}        cell    The current working HTML array
-       */
-       renderRowFooter : function(weekNum, html) {
-               html[html.length] = '<th class="calrowfoot">' + weekNum + '</th>';
-               return html;
-       },
-       
-       /**
-       * Renders a single standard calendar cell in the calendar widget table.
-       * All logic for determining how a standard default cell will be rendered is 
-       * encapsulated in this method, and must be accounted for when extending the
-       * widget class.
-       * @method renderCellDefault
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       */
-       renderCellDefault : function(workingDate, cell) {
-               cell.innerHTML = '<a href="#" class="' + this.Style.CSS_CELL_SELECTOR + '">' + this.buildDayLabel(workingDate) + "</a>";
-       },
-       
-       /**
-       * Styles a selectable cell.
-       * @method styleCellDefault
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       */
-       styleCellDefault : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_SELECTABLE);
-       },
-       
-       
-       /**
-       * Renders a single standard calendar cell using the CSS hightlight1 style
-       * @method renderCellStyleHighlight1
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       */
-       renderCellStyleHighlight1 : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_HIGHLIGHT1);
-       },
-       
-       /**
-       * Renders a single standard calendar cell using the CSS hightlight2 style
-       * @method renderCellStyleHighlight2
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       */
-       renderCellStyleHighlight2 : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_HIGHLIGHT2);
-       },
-       
-       /**
-       * Renders a single standard calendar cell using the CSS hightlight3 style
-       * @method renderCellStyleHighlight3
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       */
-       renderCellStyleHighlight3 : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_HIGHLIGHT3);
-       },
-       
-       /**
-       * Renders a single standard calendar cell using the CSS hightlight4 style
-       * @method renderCellStyleHighlight4
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       */
-       renderCellStyleHighlight4 : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_HIGHLIGHT4);
-       },
-       
-       /**
-       * Applies the default style used for rendering today's date to the current calendar cell
-       * @method renderCellStyleToday
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       */
-       renderCellStyleToday : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_TODAY);
-       },
-       
-       /**
-       * Applies the default style used for rendering selected dates to the current calendar cell
-       * @method renderCellStyleSelected
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       * @return {String} YAHOO.widget.Calendar.STOP_RENDER if rendering should stop with this style, null or nothing if rendering
-       *                       should not be terminated
-       */
-       renderCellStyleSelected : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_SELECTED);
-       },
-       
-       /**
-       * Applies the default style used for rendering dates that are not a part of the current
-       * month (preceding or trailing the cells for the current month)
-       * @method renderCellNotThisMonth
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       * @return {String} YAHOO.widget.Calendar.STOP_RENDER if rendering should stop with this style, null or nothing if rendering
-       *                       should not be terminated
-       */
-       renderCellNotThisMonth : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_OOM);
-               cell.innerHTML=workingDate.getDate();
-               return YAHOO.widget.Calendar.STOP_RENDER;
-       },
-       
-       /**
-       * Renders the current calendar cell as a non-selectable "black-out" date using the default
-       * restricted style.
-       * @method renderBodyCellRestricted
-       * @param {Date}                                 workingDate             The current working Date object being used to generate the calendar
-       * @param {HTMLTableCellElement} cell                    The current working cell in the calendar
-       * @return {String} YAHOO.widget.Calendar.STOP_RENDER if rendering should stop with this style, null or nothing if rendering
-       *                       should not be terminated
-       */
-       renderBodyCellRestricted : function(workingDate, cell) {
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL);
-               YAHOO.util.Dom.addClass(cell, this.Style.CSS_CELL_RESTRICTED);
-               cell.innerHTML=workingDate.getDate();
-               return YAHOO.widget.Calendar.STOP_RENDER;
-       },
-       
-       // END BUILT-IN TABLE CELL RENDERERS
-       
-       // BEGIN MONTH NAVIGATION METHODS
-       
-       /**
-       * Adds the designated number of months to the current calendar month, and sets the current
-       * calendar page date to the new month.
-       * @method addMonths
-       * @param {Number}       count   The number of months to add to the current calendar
-       */
-       addMonths : function(count) {
-               var cfgPageDate = YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
-               this.cfg.setProperty(cfgPageDate, YAHOO.widget.DateMath.add(this.cfg.getProperty(cfgPageDate), YAHOO.widget.DateMath.MONTH, count));
-               this.resetRenderers();
-               this.changePageEvent.fire();
-       },
-       
-       /**
-       * Subtracts the designated number of months from the current calendar month, and sets the current
-       * calendar page date to the new month.
-       * @method subtractMonths
-       * @param {Number}       count   The number of months to subtract from the current calendar
-       */
-       subtractMonths : function(count) {
-               var cfgPageDate = YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
-               this.cfg.setProperty(cfgPageDate, YAHOO.widget.DateMath.subtract(this.cfg.getProperty(cfgPageDate), YAHOO.widget.DateMath.MONTH, count));
-               this.resetRenderers();
-               this.changePageEvent.fire();
-       },
-       
-       /**
-       * Adds the designated number of years to the current calendar, and sets the current
-       * calendar page date to the new month.
-       * @method addYears
-       * @param {Number}       count   The number of years to add to the current calendar
-       */
-       addYears : function(count) {
-               var cfgPageDate = YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
-               this.cfg.setProperty(cfgPageDate, YAHOO.widget.DateMath.add(this.cfg.getProperty(cfgPageDate), YAHOO.widget.DateMath.YEAR, count));
-               this.resetRenderers();
-               this.changePageEvent.fire();
-       },
-       
-       /**
-       * Subtcats the designated number of years from the current calendar, and sets the current
-       * calendar page date to the new month.
-       * @method subtractYears
-       * @param {Number}       count   The number of years to subtract from the current calendar
-       */
-       subtractYears : function(count) {
-               var cfgPageDate = YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
-               this.cfg.setProperty(cfgPageDate, YAHOO.widget.DateMath.subtract(this.cfg.getProperty(cfgPageDate), YAHOO.widget.DateMath.YEAR, count));
-               this.resetRenderers();
-               this.changePageEvent.fire();
-       },
-       
-       /**
-       * Navigates to the next month page in the calendar widget.
-       * @method nextMonth
-       */
-       nextMonth : function() {
-               this.addMonths(1);
-       },
-       
-       /**
-       * Navigates to the previous month page in the calendar widget.
-       * @method previousMonth
-       */
-       previousMonth : function() {
-               this.subtractMonths(1);
-       },
-       
-       /**
-       * Navigates to the next year in the currently selected month in the calendar widget.
-       * @method nextYear
-       */
-       nextYear : function() {
-               this.addYears(1);
-       },
-       
-       /**
-       * Navigates to the previous year in the currently selected month in the calendar widget.
-       * @method previousYear
-       */
-       previousYear : function() {
-               this.subtractYears(1);
-       },
-       
-       // END MONTH NAVIGATION METHODS
-       
-       // BEGIN SELECTION METHODS
-       
-       /**
-       * Resets the calendar widget to the originally selected month and year, and 
-       * sets the calendar to the initial selection(s).
-       * @method reset
-       */
-       reset : function() {
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-               this.cfg.resetProperty(defCfg.SELECTED.key);
-               this.cfg.resetProperty(defCfg.PAGEDATE.key);
-               this.resetEvent.fire();
-       },
-       
-       /**
-       * Clears the selected dates in the current calendar widget and sets the calendar
-       * to the current month and year.
-       * @method clear
-       */
-       clear : function() {
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-               this.cfg.setProperty(defCfg.SELECTED.key, []);
-               this.cfg.setProperty(defCfg.PAGEDATE.key, new Date(this.today.getTime()));
-               this.clearEvent.fire();
-       },
-       
-       /**
-       * Selects a date or a collection of dates on the current calendar. This method, by default,
-       * does not call the render method explicitly. Once selection has completed, render must be 
-       * called for the changes to be reflected visually.
-       *
-       * Any dates which are OOB (out of bounds, not selectable) will not be selected and the array of 
-       * selected dates passed to the selectEvent will not contain OOB dates.
-       * 
-       * If all dates are OOB, the no state change will occur; beforeSelect and select events will not be fired.
-       *
-       * @method select
-       * @param        {String/Date/Date[]}    date    The date string of dates to select in the current calendar. Valid formats are
-       *                                                               individual date(s) (12/24/2005,12/26/2005) or date range(s) (12/24/2005-1/1/2006).
-       *                                                               Multiple comma-delimited dates can also be passed to this method (12/24/2005,12/11/2005-12/13/2005).
-       *                                                               This method can also take a JavaScript Date object or an array of Date objects.
-       * @return       {Date[]}                        Array of JavaScript Date objects representing all individual dates that are currently selected.
-       */
-       select : function(date) {
-       
-               var aToBeSelected = this._toFieldArray(date);
-       
-               // Filtered array of valid dates
-               var validDates = [];
-               var selected = [];
-               var cfgSelected = YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
-               
-               for (var a=0; a < aToBeSelected.length; ++a) {
-                       var toSelect = aToBeSelected[a];
-       
-                       if (!this.isDateOOB(this._toDate(toSelect))) {
-                               
-                               if (validDates.length === 0) {
-                                       this.beforeSelectEvent.fire();
-                                       selected = this.cfg.getProperty(cfgSelected);
-                               }
-       
-                               validDates.push(toSelect);
-                               
-                               if (this._indexOfSelectedFieldArray(toSelect) == -1) { 
-                                       selected[selected.length] = toSelect;
-                               }
-                       }
-               }
-               
-       
-               if (validDates.length > 0) {
-                       if (this.parent) {
-                               this.parent.cfg.setProperty(cfgSelected, selected);
-                       } else {
-                               this.cfg.setProperty(cfgSelected, selected);
-                       }
-                       this.selectEvent.fire(validDates);
-               }
-       
-               return this.getSelectedDates();
-       },
-       
-       /**
-       * Selects a date on the current calendar by referencing the index of the cell that should be selected.
-       * This method is used to easily select a single cell (usually with a mouse click) without having to do
-       * a full render. The selected style is applied to the cell directly.
-       *
-       * If the cell is not marked with the CSS_CELL_SELECTABLE class (as is the case by default for out of month 
-       * or out of bounds cells), it will not be selected and in such a case beforeSelect and select events will not be fired.
-       * 
-       * @method selectCell
-       * @param        {Number}        cellIndex       The index of the cell to select in the current calendar. 
-       * @return       {Date[]}        Array of JavaScript Date objects representing all individual dates that are currently selected.
-       */
-       selectCell : function(cellIndex) {
-       
-               var cell = this.cells[cellIndex];
-               var cellDate = this.cellDates[cellIndex];
-               var dCellDate = this._toDate(cellDate);
-               
-               var selectable = YAHOO.util.Dom.hasClass(cell, this.Style.CSS_CELL_SELECTABLE);
-       
-               if (selectable) {
-       
-                       this.beforeSelectEvent.fire();
-       
-                       var cfgSelected = YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
-                       var selected = this.cfg.getProperty(cfgSelected);
-       
-                       var selectDate = cellDate.concat();
-       
-                       if (this._indexOfSelectedFieldArray(selectDate) == -1) {
-                               selected[selected.length] = selectDate;
-                       }
-                       if (this.parent) {
-                               this.parent.cfg.setProperty(cfgSelected, selected);
-                       } else {
-                               this.cfg.setProperty(cfgSelected, selected);
-                       }
-                       this.renderCellStyleSelected(dCellDate,cell);
-                       this.selectEvent.fire([selectDate]);
-       
-                       this.doCellMouseOut.call(cell, null, this);             
-               }
-       
-               return this.getSelectedDates();
-       },
-       
-       /**
-       * Deselects a date or a collection of dates on the current calendar. This method, by default,
-       * does not call the render method explicitly. Once deselection has completed, render must be 
-       * called for the changes to be reflected visually.
-       * 
-       * The method will not attempt to deselect any dates which are OOB (out of bounds, and hence not selectable) 
-       * and the array of deselected dates passed to the deselectEvent will not contain any OOB dates.
-       * 
-       * If all dates are OOB, beforeDeselect and deselect events will not be fired.
-       * 
-       * @method deselect
-       * @param        {String/Date/Date[]}    date    The date string of dates to deselect in the current calendar. Valid formats are
-       *                                                               individual date(s) (12/24/2005,12/26/2005) or date range(s) (12/24/2005-1/1/2006).
-       *                                                               Multiple comma-delimited dates can also be passed to this method (12/24/2005,12/11/2005-12/13/2005).
-       *                                                               This method can also take a JavaScript Date object or an array of Date objects. 
-       * @return       {Date[]}                        Array of JavaScript Date objects representing all individual dates that are currently selected.
-       */
-       deselect : function(date) {
-       
-               var aToBeDeselected = this._toFieldArray(date);
-       
-               var validDates = [];
-               var selected = [];
-               var cfgSelected = YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
-       
-               for (var a=0; a < aToBeDeselected.length; ++a) {
-                       var toDeselect = aToBeDeselected[a];
-       
-                       if (!this.isDateOOB(this._toDate(toDeselect))) {
-       
-                               if (validDates.length === 0) {
-                                       this.beforeDeselectEvent.fire();
-                                       selected = this.cfg.getProperty(cfgSelected);
-                               }
-       
-                               validDates.push(toDeselect);
-       
-                               var index = this._indexOfSelectedFieldArray(toDeselect);
-                               if (index != -1) {      
-                                       selected.splice(index,1);
-                               }
-                       }
-               }
-       
-       
-               if (validDates.length > 0) {
-                       if (this.parent) {
-                               this.parent.cfg.setProperty(cfgSelected, selected);
-                       } else {
-                               this.cfg.setProperty(cfgSelected, selected);
-                       }
-                       this.deselectEvent.fire(validDates);
-               }
-       
-               return this.getSelectedDates();
-       },
-       
-       /**
-       * Deselects a date on the current calendar by referencing the index of the cell that should be deselected.
-       * This method is used to easily deselect a single cell (usually with a mouse click) without having to do
-       * a full render. The selected style is removed from the cell directly.
-       * 
-       * If the cell is not marked with the CSS_CELL_SELECTABLE class (as is the case by default for out of month 
-       * or out of bounds cells), the method will not attempt to deselect it and in such a case, beforeDeselect and 
-       * deselect events will not be fired.
-       * 
-       * @method deselectCell
-       * @param        {Number}        cellIndex       The index of the cell to deselect in the current calendar. 
-       * @return       {Date[]}        Array of JavaScript Date objects representing all individual dates that are currently selected.
-       */
-       deselectCell : function(cellIndex) {
-               var cell = this.cells[cellIndex];
-               var cellDate = this.cellDates[cellIndex];
-               var cellDateIndex = this._indexOfSelectedFieldArray(cellDate);
-               
-               var selectable = YAHOO.util.Dom.hasClass(cell, this.Style.CSS_CELL_SELECTABLE);
-       
-               if (selectable) {
-       
-                       this.beforeDeselectEvent.fire();
-       
-                       var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-                       var selected = this.cfg.getProperty(defCfg.SELECTED.key);
-       
-                       var dCellDate = this._toDate(cellDate);
-                       var selectDate = cellDate.concat();
-       
-                       if (cellDateIndex > -1) {
-                               if (this.cfg.getProperty(defCfg.PAGEDATE.key).getMonth() == dCellDate.getMonth() &&
-                                       this.cfg.getProperty(defCfg.PAGEDATE.key).getFullYear() == dCellDate.getFullYear()) {
-                                       YAHOO.util.Dom.removeClass(cell, this.Style.CSS_CELL_SELECTED);
-                               }
-                               selected.splice(cellDateIndex, 1);
-                       }
-       
-                       if (this.parent) {
-                               this.parent.cfg.setProperty(defCfg.SELECTED.key, selected);
-                       } else {
-                               this.cfg.setProperty(defCfg.SELECTED.key, selected);
-                       }
-       
-                       this.deselectEvent.fire(selectDate);
-               }
-       
-               return this.getSelectedDates();
-       },
-       
-       /**
-       * Deselects all dates on the current calendar.
-       * @method deselectAll
-       * @return {Date[]}              Array of JavaScript Date objects representing all individual dates that are currently selected.
-       *                                               Assuming that this function executes properly, the return value should be an empty array.
-       *                                               However, the empty array is returned for the sake of being able to check the selection status
-       *                                               of the calendar.
-       */
-       deselectAll : function() {
-               this.beforeDeselectEvent.fire();
-               
-               var cfgSelected = YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key;
-       
-               var selected = this.cfg.getProperty(cfgSelected);
-               var count = selected.length;
-               var sel = selected.concat();
-       
-               if (this.parent) {
-                       this.parent.cfg.setProperty(cfgSelected, []);
-               } else {
-                       this.cfg.setProperty(cfgSelected, []);
-               }
-               
-               if (count > 0) {
-                       this.deselectEvent.fire(sel);
-               }
-       
-               return this.getSelectedDates();
-       },
-       
-       // END SELECTION METHODS
-       
-       // BEGIN TYPE CONVERSION METHODS
-       
-       /**
-       * Converts a date (either a JavaScript Date object, or a date string) to the internal data structure
-       * used to represent dates: [[yyyy,mm,dd],[yyyy,mm,dd]].
-       * @method _toFieldArray
-       * @private
-       * @param        {String/Date/Date[]}    date    The date string of dates to deselect in the current calendar. Valid formats are
-       *                                                               individual date(s) (12/24/2005,12/26/2005) or date range(s) (12/24/2005-1/1/2006).
-       *                                                               Multiple comma-delimited dates can also be passed to this method (12/24/2005,12/11/2005-12/13/2005).
-       *                                                               This method can also take a JavaScript Date object or an array of Date objects. 
-       * @return {Array[](Number[])}   Array of date field arrays
-       */
-       _toFieldArray : function(date) {
-               var returnDate = [];
-       
-               if (date instanceof Date) {
-                       returnDate = [[date.getFullYear(), date.getMonth()+1, date.getDate()]];
-               } else if (YAHOO.lang.isString(date)) {
-                       returnDate = this._parseDates(date);
-               } else if (YAHOO.lang.isArray(date)) {
-                       for (var i=0;i<date.length;++i) {
-                               var d = date[i];
-                               returnDate[returnDate.length] = [d.getFullYear(),d.getMonth()+1,d.getDate()];
-                       }
-               }
-               
-               return returnDate;
-       },
-       
-       /**
-       * Converts a date field array [yyyy,mm,dd] to a JavaScript Date object. The date field array
-       * is the format in which dates are as provided as arguments to selectEvent and deselectEvent listeners.
-       * 
-       * @method toDate
-       * @param        {Number[]}      dateFieldArray  The date field array to convert to a JavaScript Date.
-       * @return       {Date}  JavaScript Date object representing the date field array.
-       */
-       toDate : function(dateFieldArray) {
-               return this._toDate(dateFieldArray);
-       },
-       
-       /**
-       * Converts a date field array [yyyy,mm,dd] to a JavaScript Date object.
-       * @method _toDate
-       * @private
-       * @deprecated Made public, toDate 
-       * @param        {Number[]}              dateFieldArray  The date field array to convert to a JavaScript Date.
-       * @return       {Date}  JavaScript Date object representing the date field array
-       */
-       _toDate : function(dateFieldArray) {
-               if (dateFieldArray instanceof Date) {
-                       return dateFieldArray;
-               } else {
-                       return YAHOO.widget.DateMath.getDate(dateFieldArray[0],dateFieldArray[1]-1,dateFieldArray[2]);
-               }
-       },
-       
-       // END TYPE CONVERSION METHODS 
-       
-       // BEGIN UTILITY METHODS
-       
-       /**
-       * Converts a date field array [yyyy,mm,dd] to a JavaScript Date object.
-       * @method _fieldArraysAreEqual
-       * @private
-       * @param        {Number[]}      array1  The first date field array to compare
-       * @param        {Number[]}      array2  The first date field array to compare
-       * @return       {Boolean}       The boolean that represents the equality of the two arrays
-       */
-       _fieldArraysAreEqual : function(array1, array2) {
-               var match = false;
-       
-               if (array1[0]==array2[0]&&array1[1]==array2[1]&&array1[2]==array2[2]) {
-                       match=true;     
-               }
-       
-               return match;
-       },
-       
-       /**
-       * Gets the index of a date field array [yyyy,mm,dd] in the current list of selected dates.
-       * @method       _indexOfSelectedFieldArray
-       * @private
-       * @param        {Number[]}              find    The date field array to search for
-       * @return       {Number}                        The index of the date field array within the collection of selected dates.
-       *                                                               -1 will be returned if the date is not found.
-       */
-       _indexOfSelectedFieldArray : function(find) {
-               var selected = -1;
-               var seldates = this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key);
-       
-               for (var s=0;s<seldates.length;++s) {
-                       var sArray = seldates[s];
-                       if (find[0]==sArray[0]&&find[1]==sArray[1]&&find[2]==sArray[2]) {
-                               selected = s;
-                               break;
-                       }
-               }
-       
-               return selected;
-       },
-       
-       /**
-       * Determines whether a given date is OOM (out of month).
-       * @method       isDateOOM
-       * @param        {Date}  date    The JavaScript Date object for which to check the OOM status
-       * @return       {Boolean}       true if the date is OOM
-       */
-       isDateOOM : function(date) {
-               return (date.getMonth() != this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key).getMonth());
-       },
-       
-       /**
-       * Determines whether a given date is OOB (out of bounds - less than the mindate or more than the maxdate).
-       *
-       * @method       isDateOOB
-       * @param        {Date}  date    The JavaScript Date object for which to check the OOB status
-       * @return       {Boolean}       true if the date is OOB
-       */
-       isDateOOB : function(date) {
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-               
-               var minDate = this.cfg.getProperty(defCfg.MINDATE.key);
-               var maxDate = this.cfg.getProperty(defCfg.MAXDATE.key);
-               var dm = YAHOO.widget.DateMath;
-               
-               if (minDate) {
-                       minDate = dm.clearTime(minDate);
-               } 
-               if (maxDate) {
-                       maxDate = dm.clearTime(maxDate);
-               }
-       
-               var clearedDate = new Date(date.getTime());
-               clearedDate = dm.clearTime(clearedDate);
-       
-               return ((minDate && clearedDate.getTime() < minDate.getTime()) || (maxDate && clearedDate.getTime() > maxDate.getTime()));
-       },
-       
-       /**
-        * Parses a pagedate configuration property value. The value can either be specified as a string of form "mm/yyyy" or a Date object 
-        * and is parsed into a Date object normalized to the first day of the month. If no value is passed in, the month and year from today's date are used to create the Date object 
-        * @method      _parsePageDate
-        * @private
-        * @param {Date|String} date    Pagedate value which needs to be parsed
-        * @return {Date}       The Date object representing the pagedate
-        */
-       _parsePageDate : function(date) {
-               var parsedDate;
-               
-               var defCfg = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-       
-               if (date) {
-                       if (date instanceof Date) {
-                               parsedDate = YAHOO.widget.DateMath.findMonthStart(date);
-                       } else {
-                               var month, year, aMonthYear;
-                               aMonthYear = date.split(this.cfg.getProperty(defCfg.DATE_FIELD_DELIMITER.key));
-                               month = parseInt(aMonthYear[this.cfg.getProperty(defCfg.MY_MONTH_POSITION.key)-1], 10)-1;
-                               year = parseInt(aMonthYear[this.cfg.getProperty(defCfg.MY_YEAR_POSITION.key)-1], 10);
-
-                               parsedDate = YAHOO.widget.DateMath.getDate(year, month, 1);
-                       }
-               } else {
-                       parsedDate = YAHOO.widget.DateMath.getDate(this.today.getFullYear(), this.today.getMonth(), 1);
-               }
-               return parsedDate;
-       },
-       
-       // END UTILITY METHODS
-       
-       // BEGIN EVENT HANDLERS
-       
-       /**
-       * Event executed before a date is selected in the calendar widget.
-       * @deprecated Event handlers for this event should be susbcribed to beforeSelectEvent.
-       */
-       onBeforeSelect : function() {
-               if (this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.MULTI_SELECT.key) === false) {
-                       if (this.parent) {
-                               this.parent.callChildFunction("clearAllBodyCellStyles", this.Style.CSS_CELL_SELECTED);
-                               this.parent.deselectAll();
-                       } else {
-                               this.clearAllBodyCellStyles(this.Style.CSS_CELL_SELECTED);
-                               this.deselectAll();
-                       }
-               }
-       },
-       
-       /**
-       * Event executed when a date is selected in the calendar widget.
-       * @param        {Array} selected        An array of date field arrays representing which date or dates were selected. Example: [ [2006,8,6],[2006,8,7],[2006,8,8] ]
-       * @deprecated Event handlers for this event should be susbcribed to selectEvent.
-       */
-       onSelect : function(selected) { },
-       
-       /**
-       * Event executed before a date is deselected in the calendar widget.
-       * @deprecated Event handlers for this event should be susbcribed to beforeDeselectEvent.
-       */
-       onBeforeDeselect : function() { },
-       
-       /**
-       * Event executed when a date is deselected in the calendar widget.
-       * @param        {Array} selected        An array of date field arrays representing which date or dates were deselected. Example: [ [2006,8,6],[2006,8,7],[2006,8,8] ]
-       * @deprecated Event handlers for this event should be susbcribed to deselectEvent.
-       */
-       onDeselect : function(deselected) { },
-       
-       /**
-       * Event executed when the user navigates to a different calendar page.
-       * @deprecated Event handlers for this event should be susbcribed to changePageEvent.
-       */
-       onChangePage : function() {
-               this.render();
-       },
-       
-       /**
-       * Event executed when the calendar widget is rendered.
-       * @deprecated Event handlers for this event should be susbcribed to renderEvent.
-       */
-       onRender : function() { },
-       
-       /**
-       * Event executed when the calendar widget is reset to its original state.
-       * @deprecated Event handlers for this event should be susbcribed to resetEvemt.
-       */
-       onReset : function() { this.render(); },
-       
-       /**
-       * Event executed when the calendar widget is completely cleared to the current month with no selections.
-       * @deprecated Event handlers for this event should be susbcribed to clearEvent.
-       */
-       onClear : function() { this.render(); },
-       
-       /**
-       * Validates the calendar widget. This method has no default implementation
-       * and must be extended by subclassing the widget.
-       * @return       Should return true if the widget validates, and false if
-       * it doesn't.
-       * @type Boolean
-       */
-       validate : function() { return true; },
-       
-       // END EVENT HANDLERS
-       
-       // BEGIN DATE PARSE METHODS
-       
-       /**
-       * Converts a date string to a date field array
-       * @private
-       * @param        {String}        sDate                   Date string. Valid formats are mm/dd and mm/dd/yyyy.
-       * @return                               A date field array representing the string passed to the method
-       * @type Array[](Number[])
-       */
-       _parseDate : function(sDate) {
-               var aDate = sDate.split(this.Locale.DATE_FIELD_DELIMITER);
-               var rArray;
-       
-               if (aDate.length == 2) {
-                       rArray = [aDate[this.Locale.MD_MONTH_POSITION-1],aDate[this.Locale.MD_DAY_POSITION-1]];
-                       rArray.type = YAHOO.widget.Calendar.MONTH_DAY;
-               } else {
-                       rArray = [aDate[this.Locale.MDY_YEAR_POSITION-1],aDate[this.Locale.MDY_MONTH_POSITION-1],aDate[this.Locale.MDY_DAY_POSITION-1]];
-                       rArray.type = YAHOO.widget.Calendar.DATE;
-               }
-       
-               for (var i=0;i<rArray.length;i++) {
-                       rArray[i] = parseInt(rArray[i], 10);
-               }
-       
-               return rArray;
-       },
-       
-       /**
-       * Converts a multi or single-date string to an array of date field arrays
-       * @private
-       * @param        {String}        sDates          Date string with one or more comma-delimited dates. Valid formats are mm/dd, mm/dd/yyyy, mm/dd/yyyy-mm/dd/yyyy
-       * @return                                                       An array of date field arrays
-       * @type Array[](Number[])
-       */
-       _parseDates : function(sDates) {
-               var aReturn = [];
-       
-               var aDates = sDates.split(this.Locale.DATE_DELIMITER);
-               
-               for (var d=0;d<aDates.length;++d) {
-                       var sDate = aDates[d];
-       
-                       if (sDate.indexOf(this.Locale.DATE_RANGE_DELIMITER) != -1) {
-                               // This is a range
-                               var aRange = sDate.split(this.Locale.DATE_RANGE_DELIMITER);
-       
-                               var dateStart = this._parseDate(aRange[0]);
-                               var dateEnd = this._parseDate(aRange[1]);
-       
-                               var fullRange = this._parseRange(dateStart, dateEnd);
-                               aReturn = aReturn.concat(fullRange);
-                       } else {
-                               // This is not a range
-                               var aDate = this._parseDate(sDate);
-                               aReturn.push(aDate);
-                       }
-               }
-               return aReturn;
-       },
-       
-       /**
-       * Converts a date range to the full list of included dates
-       * @private
-       * @param        {Number[]}      startDate       Date field array representing the first date in the range
-       * @param        {Number[]}      endDate         Date field array representing the last date in the range
-       * @return                                                       An array of date field arrays
-       * @type Array[](Number[])
-       */
-       _parseRange : function(startDate, endDate) {
-               var dCurrent = YAHOO.widget.DateMath.add(YAHOO.widget.DateMath.getDate(startDate[0],startDate[1]-1,startDate[2]),YAHOO.widget.DateMath.DAY,1);
-               var dEnd     = YAHOO.widget.DateMath.getDate(endDate[0],  endDate[1]-1,  endDate[2]);
-       
-               var results = [];
-               results.push(startDate);
-               while (dCurrent.getTime() <= dEnd.getTime()) {
-                       results.push([dCurrent.getFullYear(),dCurrent.getMonth()+1,dCurrent.getDate()]);
-                       dCurrent = YAHOO.widget.DateMath.add(dCurrent,YAHOO.widget.DateMath.DAY,1);
-               }
-               return results;
-       },
-       
-       // END DATE PARSE METHODS
-       
-       // BEGIN RENDERER METHODS
-       
-       /**
-       * Resets the render stack of the current calendar to its original pre-render value.
-       */
-       resetRenderers : function() {
-               this.renderStack = this._renderStack.concat();
-       },
-       
-       /**
-        * Removes all custom renderers added to the Calendar through the addRenderer, addMonthRenderer and 
-        * addWeekdayRenderer methods. Calendar's render method needs to be called after removing renderers 
-        * to re-render the Calendar without custom renderers applied.
-        */
-       removeRenderers : function() {
-               this._renderStack = [];
-               this.renderStack = [];
-       },
-
-       /**
-       * Clears the inner HTML, CSS class and style information from the specified cell.
-       * @method clearElement
-       * @param        {HTMLTableCellElement} cell The cell to clear
-       */ 
-       clearElement : function(cell) {
-               cell.innerHTML = "&#160;";
-               cell.className="";
-       },
-       
-       /**
-       * Adds a renderer to the render stack. The function reference passed to this method will be executed
-       * when a date cell matches the conditions specified in the date string for this renderer.
-       * @method addRenderer
-       * @param        {String}        sDates          A date string to associate with the specified renderer. Valid formats
-       *                                                                       include date (12/24/2005), month/day (12/24), and range (12/1/2004-1/1/2005)
-       * @param        {Function}      fnRender        The function executed to render cells that match the render rules for this renderer.
-       */
-       addRenderer : function(sDates, fnRender) {
-               var aDates = this._parseDates(sDates);
-               for (var i=0;i<aDates.length;++i) {
-                       var aDate = aDates[i];
-               
-                       if (aDate.length == 2) { // this is either a range or a month/day combo
-                               if (aDate[0] instanceof Array) { // this is a range
-                                       this._addRenderer(YAHOO.widget.Calendar.RANGE,aDate,fnRender);
-                               } else { // this is a month/day combo
-                                       this._addRenderer(YAHOO.widget.Calendar.MONTH_DAY,aDate,fnRender);
-                               }
-                       } else if (aDate.length == 3) {
-                               this._addRenderer(YAHOO.widget.Calendar.DATE,aDate,fnRender);
-                       }
-               }
-       },
-       
-       /**
-       * The private method used for adding cell renderers to the local render stack.
-       * This method is called by other methods that set the renderer type prior to the method call.
-       * @method _addRenderer
-       * @private
-       * @param        {String}        type            The type string that indicates the type of date renderer being added.
-       *                                                                       Values are YAHOO.widget.Calendar.DATE, YAHOO.widget.Calendar.MONTH_DAY, YAHOO.widget.Calendar.WEEKDAY,
-       *                                                                       YAHOO.widget.Calendar.RANGE, YAHOO.widget.Calendar.MONTH
-       * @param        {Array}         aDates          An array of dates used to construct the renderer. The format varies based
-       *                                                                       on the renderer type
-       * @param        {Function}      fnRender        The function executed to render cells that match the render rules for this renderer.
-       */
-       _addRenderer : function(type, aDates, fnRender) {
-               var add = [type,aDates,fnRender];
-               this.renderStack.unshift(add);  
-               this._renderStack = this.renderStack.concat();
-       },
-
-       /**
-       * Adds a month to the render stack. The function reference passed to this method will be executed
-       * when a date cell matches the month passed to this method.
-       * @method addMonthRenderer
-       * @param        {Number}        month           The month (1-12) to associate with this renderer
-       * @param        {Function}      fnRender        The function executed to render cells that match the render rules for this renderer.
-       */
-       addMonthRenderer : function(month, fnRender) {
-               this._addRenderer(YAHOO.widget.Calendar.MONTH,[month],fnRender);
-       },
-
-       /**
-       * Adds a weekday to the render stack. The function reference passed to this method will be executed
-       * when a date cell matches the weekday passed to this method.
-       * @method addWeekdayRenderer
-       * @param        {Number}        weekday         The weekday (0-6) to associate with this renderer
-       * @param        {Function}      fnRender        The function executed to render cells that match the render rules for this renderer.
-       */
-       addWeekdayRenderer : function(weekday, fnRender) {
-               this._addRenderer(YAHOO.widget.Calendar.WEEKDAY,[weekday],fnRender);
-       },
-       
-       // END RENDERER METHODS
-       
-       // BEGIN CSS METHODS
-       
-       /**
-       * Removes all styles from all body cells in the current calendar table.
-       * @method clearAllBodyCellStyles
-       * @param        {style} style The CSS class name to remove from all calendar body cells
-       */
-       clearAllBodyCellStyles : function(style) {
-               for (var c=0;c<this.cells.length;++c) {
-                       YAHOO.util.Dom.removeClass(this.cells[c],style);
-               }
-       },
-       
-       // END CSS METHODS
-       
-       // BEGIN GETTER/SETTER METHODS
-       /**
-       * Sets the calendar's month explicitly
-       * @method setMonth
-       * @param {Number}       month           The numeric month, from 0 (January) to 11 (December)
-       */
-       setMonth : function(month) {
-               var cfgPageDate = YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
-               var current = this.cfg.getProperty(cfgPageDate);
-               current.setMonth(parseInt(month, 10));
-               this.cfg.setProperty(cfgPageDate, current);
-       },
-
-       /**
-       * Sets the calendar's year explicitly.
-       * @method setYear
-       * @param {Number}       year            The numeric 4-digit year
-       */
-       setYear : function(year) {
-               var cfgPageDate = YAHOO.widget.Calendar._DEFAULT_CONFIG.PAGEDATE.key;
-               var current = this.cfg.getProperty(cfgPageDate);
-               current.setFullYear(parseInt(year, 10));
-               this.cfg.setProperty(cfgPageDate, current);
-       },
-
-       /**
-       * Gets the list of currently selected dates from the calendar.
-       * @method getSelectedDates
-       * @return {Date[]} An array of currently selected JavaScript Date objects.
-       */
-       getSelectedDates : function() {
-               var returnDates = [];
-               var selected = this.cfg.getProperty(YAHOO.widget.Calendar._DEFAULT_CONFIG.SELECTED.key);
-
-               for (var d=0;d<selected.length;++d) {
-                       var dateArray = selected[d];
-
-                       var date = YAHOO.widget.DateMath.getDate(dateArray[0],dateArray[1]-1,dateArray[2]);
-                       returnDates.push(date);
-               }
-
-               returnDates.sort( function(a,b) { return a-b; } );
-               return returnDates;
-       },
-
-       /// END GETTER/SETTER METHODS ///
-       
-       /**
-       * Hides the Calendar's outer container from view.
-       * @method hide
-       */
-       hide : function() {
-               if (this.beforeHideEvent.fire()) {
-                       this.oDomContainer.style.display = "none";
-                       this.hideEvent.fire();
-               }
-       },
-
-       /**
-       * Shows the Calendar's outer container.
-       * @method show
-       */
-       show : function() {
-               if (this.beforeShowEvent.fire()) {
-                       this.oDomContainer.style.display = "block";
-                       this.showEvent.fire();
-               }
-       },
-
-       /**
-       * Returns a string representing the current browser.
-       * @deprecated As of 2.3.0, environment information is available in YAHOO.env.ua
-       * @see YAHOO.env.ua
-       * @property browser
-       * @type String
-       */
-       browser : (function() {
-                               var ua = navigator.userAgent.toLowerCase();
-                                         if (ua.indexOf('opera')!=-1) { // Opera (check first in case of spoof)
-                                                return 'opera';
-                                         } else if (ua.indexOf('msie 7')!=-1) { // IE7
-                                                return 'ie7';
-                                         } else if (ua.indexOf('msie') !=-1) { // IE
-                                                return 'ie';
-                                         } else if (ua.indexOf('safari')!=-1) { // Safari (check before Gecko because it includes "like Gecko")
-                                                return 'safari';
-                                         } else if (ua.indexOf('gecko') != -1) { // Gecko
-                                                return 'gecko';
-                                         } else {
-                                                return false;
-                                         }
-                               })(),
-       /**
-       * Returns a string representation of the object.
-       * @method toString
-       * @return {String}      A string representation of the Calendar object.
-       */
-       toString : function() {
-               return "Calendar " + this.id;
-       }
-};
-
-/**
-* @namespace YAHOO.widget
-* @class Calendar_Core
-* @extends YAHOO.widget.Calendar
-* @deprecated The old Calendar_Core class is no longer necessary.
-*/
-YAHOO.widget.Calendar_Core = YAHOO.widget.Calendar;
-
-YAHOO.widget.Cal_Core = YAHOO.widget.Calendar;
-
-/**
-* YAHOO.widget.CalendarGroup is a special container class for YAHOO.widget.Calendar. This class facilitates
-* the ability to have multi-page calendar views that share a single dataset and are
-* dependent on each other.
-*
-* The calendar group instance will refer to each of its elements using a 0-based index.
-* For example, to construct the placeholder for a calendar group widget with id "cal1" and
-* containerId of "cal1Container", the markup would be as follows:
-*      <xmp>
-*              <div id="cal1Container_0"></div>
-*              <div id="cal1Container_1"></div>
-*      </xmp>
-* The tables for the calendars ("cal1_0" and "cal1_1") will be inserted into those containers.
-* 
-* <p>
-* <strong>NOTE: As of 2.4.0, the constructor's ID argument is optional.</strong>
-* The CalendarGroup can be constructed by simply providing a container ID string, 
-* or a reference to a container DIV HTMLElement (the element needs to exist 
-* in the document).
-* 
-* E.g.:
-*      <xmp>
-*              var c = new YAHOO.widget.CalendarGroup("calContainer", configOptions);
-*      </xmp>
-* or:
-*   <xmp>
-*       var containerDiv = YAHOO.util.Dom.get("calContainer");
-*              var c = new YAHOO.widget.CalendarGroup(containerDiv, configOptions);
-*      </xmp>
-* </p>
-* <p>
-* If not provided, the ID will be generated from the container DIV ID by adding an "_t" suffix.
-* For example if an ID is not provided, and the container's ID is "calContainer", the CalendarGroup's ID will be set to "calContainer_t".
-* </p>
-* 
-* @namespace YAHOO.widget
-* @class CalendarGroup
-* @constructor
-* @param {String} id optional The id of the table element that will represent the CalendarGroup widget. As of 2.4.0, this argument is optional.
-* @param {String | HTMLElement} container The id of the container div element that will wrap the CalendarGroup table, or a reference to a DIV element which exists in the document.
-* @param {Object} config optional The configuration object containing the initial configuration values for the CalendarGroup.
-*/
-YAHOO.widget.CalendarGroup = function(id, containerId, config) {
-       if (arguments.length > 0) {
-               this.init.apply(this, arguments);
-       }
-};
-
-YAHOO.widget.CalendarGroup.prototype = {
-
-       /**
-       * Initializes the calendar group. All subclasses must call this method in order for the
-       * group to be initialized properly.
-       * @method init
-       * @param {String} id optional The id of the table element that will represent the CalendarGroup widget. As of 2.4.0, this argument is optional.
-       * @param {String | HTMLElement} container The id of the container div element that will wrap the CalendarGroup table, or a reference to a DIV element which exists in the document.
-       * @param {Object} config optional The configuration object containing the initial configuration values for the CalendarGroup.
-       */
-       init : function(id, container, config) {
-
-               // Normalize 2.4.0, pre 2.4.0 args
-               var nArgs = this._parseArgs(arguments);
-
-               id = nArgs.id;
-               container = nArgs.container;
-               config = nArgs.config;
-
-               this.oDomContainer = YAHOO.util.Dom.get(container);
-
-               if (!this.oDomContainer.id) {
-                       this.oDomContainer.id = YAHOO.util.Dom.generateId();
-               }
-               if (!id) {
-                       id = this.oDomContainer.id + "_t";
-               }
-
-               /**
-               * The unique id associated with the CalendarGroup
-               * @property id
-               * @type String
-               */
-               this.id = id;
-
-               /**
-               * The unique id associated with the CalendarGroup container
-               * @property containerId
-               * @type String
-               */
-               this.containerId = this.oDomContainer.id;
-
-               this.initEvents();
-               this.initStyles();
-
-               /**
-               * The collection of Calendar pages contained within the CalendarGroup
-               * @property pages
-               * @type YAHOO.widget.Calendar[]
-               */
-               this.pages = [];
-
-               YAHOO.util.Dom.addClass(this.oDomContainer, YAHOO.widget.CalendarGroup.CSS_CONTAINER);
-               YAHOO.util.Dom.addClass(this.oDomContainer, YAHOO.widget.CalendarGroup.CSS_MULTI_UP);
-
-               /**
-               * The Config object used to hold the configuration variables for the CalendarGroup
-               * @property cfg
-               * @type YAHOO.util.Config
-               */
-               this.cfg = new YAHOO.util.Config(this);
-
-               /**
-               * The local object which contains the CalendarGroup's options
-               * @property Options
-               * @type Object
-               */
-               this.Options = {};
-
-               /**
-               * The local object which contains the CalendarGroup's locale settings
-               * @property Locale
-               * @type Object
-               */
-               this.Locale = {};
-
-               this.setupConfig();
-
-               if (config) {
-                       this.cfg.applyConfig(config, true);
-               }
-
-               this.cfg.fireQueue();
-
-               // OPERA HACK FOR MISWRAPPED FLOATS
-               if (YAHOO.env.ua.opera){
-                       this.renderEvent.subscribe(this._fixWidth, this, true);
-                       this.showEvent.subscribe(this._fixWidth, this, true);
-               }
-
-       },
-
-       setupConfig : function() {
-
-               var defCfg = YAHOO.widget.CalendarGroup._DEFAULT_CONFIG;
-
-               /**
-               * The number of pages to include in the CalendarGroup. This value can only be set once, in the CalendarGroup's constructor arguments.
-               * @config pages
-               * @type Number
-               * @default 2
-               */
-               this.cfg.addProperty(defCfg.PAGES.key, { value:defCfg.PAGES.value, validator:this.cfg.checkNumber, handler:this.configPages } );
-
-               /**
-               * The month/year representing the current visible Calendar date (mm/yyyy)
-               * @config pagedate
-               * @type String
-               * @default today's date
-               */
-               this.cfg.addProperty(defCfg.PAGEDATE.key, { value:new Date(), handler:this.configPageDate } );
-
-               /**
-               * The date or range of dates representing the current Calendar selection
-               * @config selected
-               * @type String
-               * @default []
-               */
-               this.cfg.addProperty(defCfg.SELECTED.key, { value:[], handler:this.configSelected } );
-
-               /**
-               * The title to display above the CalendarGroup's month header
-               * @config title
-               * @type String
-               * @default ""
-               */
-               this.cfg.addProperty(defCfg.TITLE.key, { value:defCfg.TITLE.value, handler:this.configTitle } );
-
-               /**
-               * Whether or not a close button should be displayed for this CalendarGroup
-               * @config close
-               * @type Boolean
-               * @default false
-               */
-               this.cfg.addProperty(defCfg.CLOSE.key, { value:defCfg.CLOSE.value, handler:this.configClose } );
-
-               /**
-               * Whether or not an iframe shim should be placed under the Calendar to prevent select boxes from bleeding through in Internet Explorer 6 and below.
-               * This property is enabled by default for IE6 and below. It is disabled by default for other browsers for performance reasons, but can be 
-               * enabled if required.
-               * 
-               * @config iframe
-               * @type Boolean
-               * @default true for IE6 and below, false for all other browsers
-               */
-               this.cfg.addProperty(defCfg.IFRAME.key, { value:defCfg.IFRAME.value, handler:this.configIframe, validator:this.cfg.checkBoolean } );
-       
-               /**
-               * The minimum selectable date in the current Calendar (mm/dd/yyyy)
-               * @config mindate
-               * @type String
-               * @default null
-               */
-               this.cfg.addProperty(defCfg.MINDATE.key, { value:defCfg.MINDATE.value, handler:this.delegateConfig } );
-       
-               /**
-               * The maximum selectable date in the current Calendar (mm/dd/yyyy)
-               * @config maxdate
-               * @type String
-               * @default null
-               */      
-               this.cfg.addProperty(defCfg.MAXDATE.key, { value:defCfg.MAXDATE.value, handler:this.delegateConfig  } );
-       
-               // Options properties
-       
-               /**
-               * True if the Calendar should allow multiple selections. False by default.
-               * @config MULTI_SELECT
-               * @type Boolean
-               * @default false
-               */
-               this.cfg.addProperty(defCfg.MULTI_SELECT.key,   { value:defCfg.MULTI_SELECT.value, handler:this.delegateConfig, validator:this.cfg.checkBoolean } );
-       
-               /**
-               * The weekday the week begins on. Default is 0 (Sunday).
-               * @config START_WEEKDAY
-               * @type number
-               * @default 0
-               */      
-               this.cfg.addProperty(defCfg.START_WEEKDAY.key,  { value:defCfg.START_WEEKDAY.value, handler:this.delegateConfig, validator:this.cfg.checkNumber  } );
-               
-               /**
-               * True if the Calendar should show weekday labels. True by default.
-               * @config SHOW_WEEKDAYS
-               * @type Boolean
-               * @default true
-               */      
-               this.cfg.addProperty(defCfg.SHOW_WEEKDAYS.key,  { value:defCfg.SHOW_WEEKDAYS.value, handler:this.delegateConfig, validator:this.cfg.checkBoolean } );
-               
-               /**
-               * True if the Calendar should show week row headers. False by default.
-               * @config SHOW_WEEK_HEADER
-               * @type Boolean
-               * @default false
-               */      
-               this.cfg.addProperty(defCfg.SHOW_WEEK_HEADER.key,{ value:defCfg.SHOW_WEEK_HEADER.value, handler:this.delegateConfig, validator:this.cfg.checkBoolean } );
-               
-               /**
-               * True if the Calendar should show week row footers. False by default.
-               * @config SHOW_WEEK_FOOTER
-               * @type Boolean
-               * @default false
-               */
-               this.cfg.addProperty(defCfg.SHOW_WEEK_FOOTER.key,{ value:defCfg.SHOW_WEEK_FOOTER.value, handler:this.delegateConfig, validator:this.cfg.checkBoolean } );
-               
-               /**
-               * True if the Calendar should suppress weeks that are not a part of the current month. False by default.
-               * @config HIDE_BLANK_WEEKS
-               * @type Boolean
-               * @default false
-               */              
-               this.cfg.addProperty(defCfg.HIDE_BLANK_WEEKS.key,{ value:defCfg.HIDE_BLANK_WEEKS.value, handler:this.delegateConfig, validator:this.cfg.checkBoolean } );
-               
-               /**
-               * The image that should be used for the left navigation arrow.
-               * @config NAV_ARROW_LEFT
-               * @type String
-               * @deprecated   You can customize the image by overriding the default CSS class for the left arrow - "calnavleft"
-               * @default null
-               */              
-               this.cfg.addProperty(defCfg.NAV_ARROW_LEFT.key, { value:defCfg.NAV_ARROW_LEFT.value, handler:this.delegateConfig } );
-               
-               /**
-               * The image that should be used for the right navigation arrow.
-               * @config NAV_ARROW_RIGHT
-               * @type String
-               * @deprecated   You can customize the image by overriding the default CSS class for the right arrow - "calnavright"
-               * @default null
-               */              
-               this.cfg.addProperty(defCfg.NAV_ARROW_RIGHT.key,        { value:defCfg.NAV_ARROW_RIGHT.value, handler:this.delegateConfig } );
-       
-               // Locale properties
-               
-               /**
-               * The short month labels for the current locale.
-               * @config MONTHS_SHORT
-               * @type String[]
-               * @default ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
-               */
-               this.cfg.addProperty(defCfg.MONTHS_SHORT.key,   { value:defCfg.MONTHS_SHORT.value, handler:this.delegateConfig } );
-               
-               /**
-               * The long month labels for the current locale.
-               * @config MONTHS_LONG
-               * @type String[]
-               * @default ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"
-               */              
-               this.cfg.addProperty(defCfg.MONTHS_LONG.key,            { value:defCfg.MONTHS_LONG.value, handler:this.delegateConfig } );
-               
-               /**
-               * The 1-character weekday labels for the current locale.
-               * @config WEEKDAYS_1CHAR
-               * @type String[]
-               * @default ["S", "M", "T", "W", "T", "F", "S"]
-               */              
-               this.cfg.addProperty(defCfg.WEEKDAYS_1CHAR.key, { value:defCfg.WEEKDAYS_1CHAR.value, handler:this.delegateConfig } );
-               
-               /**
-               * The short weekday labels for the current locale.
-               * @config WEEKDAYS_SHORT
-               * @type String[]
-               * @default ["Su", "Mo", "Tu", "We", "Th", "Fr", "Sa"]
-               */              
-               this.cfg.addProperty(defCfg.WEEKDAYS_SHORT.key, { value:defCfg.WEEKDAYS_SHORT.value, handler:this.delegateConfig } );
-               
-               /**
-               * The medium weekday labels for the current locale.
-               * @config WEEKDAYS_MEDIUM
-               * @type String[]
-               * @default ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"]
-               */              
-               this.cfg.addProperty(defCfg.WEEKDAYS_MEDIUM.key,        { value:defCfg.WEEKDAYS_MEDIUM.value, handler:this.delegateConfig } );
-               
-               /**
-               * The long weekday labels for the current locale.
-               * @config WEEKDAYS_LONG
-               * @type String[]
-               * @default ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"]
-               */              
-               this.cfg.addProperty(defCfg.WEEKDAYS_LONG.key,  { value:defCfg.WEEKDAYS_LONG.value, handler:this.delegateConfig } );
-       
-               /**
-               * The setting that determines which length of month labels should be used. Possible values are "short" and "long".
-               * @config LOCALE_MONTHS
-               * @type String
-               * @default "long"
-               */
-               this.cfg.addProperty(defCfg.LOCALE_MONTHS.key,  { value:defCfg.LOCALE_MONTHS.value, handler:this.delegateConfig } );
-       
-               /**
-               * The setting that determines which length of weekday labels should be used. Possible values are "1char", "short", "medium", and "long".
-               * @config LOCALE_WEEKDAYS
-               * @type String
-               * @default "short"
-               */      
-               this.cfg.addProperty(defCfg.LOCALE_WEEKDAYS.key,        { value:defCfg.LOCALE_WEEKDAYS.value, handler:this.delegateConfig } );
-       
-               /**
-               * The value used to delimit individual dates in a date string passed to various Calendar functions.
-               * @config DATE_DELIMITER
-               * @type String
-               * @default ","
-               */
-               this.cfg.addProperty(defCfg.DATE_DELIMITER.key,         { value:defCfg.DATE_DELIMITER.value, handler:this.delegateConfig } );
-       
-               /**
-               * The value used to delimit date fields in a date string passed to various Calendar functions.
-               * @config DATE_FIELD_DELIMITER
-               * @type String
-               * @default "/"
-               */      
-               this.cfg.addProperty(defCfg.DATE_FIELD_DELIMITER.key,{ value:defCfg.DATE_FIELD_DELIMITER.value, handler:this.delegateConfig } );
-       
-               /**
-               * The value used to delimit date ranges in a date string passed to various Calendar functions.
-               * @config DATE_RANGE_DELIMITER
-               * @type String
-               * @default "-"
-               */
-               this.cfg.addProperty(defCfg.DATE_RANGE_DELIMITER.key,{ value:defCfg.DATE_RANGE_DELIMITER.value, handler:this.delegateConfig } );
-       
-               /**
-               * The position of the month in a month/year date string
-               * @config MY_MONTH_POSITION
-               * @type Number
-               * @default 1
-               */
-               this.cfg.addProperty(defCfg.MY_MONTH_POSITION.key,      { value:defCfg.MY_MONTH_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The position of the year in a month/year date string
-               * @config MY_YEAR_POSITION
-               * @type Number
-               * @default 2
-               */      
-               this.cfg.addProperty(defCfg.MY_YEAR_POSITION.key,       { value:defCfg.MY_YEAR_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The position of the month in a month/day date string
-               * @config MD_MONTH_POSITION
-               * @type Number
-               * @default 1
-               */      
-               this.cfg.addProperty(defCfg.MD_MONTH_POSITION.key,      { value:defCfg.MD_MONTH_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The position of the day in a month/year date string
-               * @config MD_DAY_POSITION
-               * @type Number
-               * @default 2
-               */      
-               this.cfg.addProperty(defCfg.MD_DAY_POSITION.key,                { value:defCfg.MD_DAY_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The position of the month in a month/day/year date string
-               * @config MDY_MONTH_POSITION
-               * @type Number
-               * @default 1
-               */      
-               this.cfg.addProperty(defCfg.MDY_MONTH_POSITION.key,     { value:defCfg.MDY_MONTH_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The position of the day in a month/day/year date string
-               * @config MDY_DAY_POSITION
-               * @type Number
-               * @default 2
-               */      
-               this.cfg.addProperty(defCfg.MDY_DAY_POSITION.key,       { value:defCfg.MDY_DAY_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The position of the year in a month/day/year date string
-               * @config MDY_YEAR_POSITION
-               * @type Number
-               * @default 3
-               */      
-               this.cfg.addProperty(defCfg.MDY_YEAR_POSITION.key,      { value:defCfg.MDY_YEAR_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the month in the month year label string used as the Calendar header
-               * @config MY_LABEL_MONTH_POSITION
-               * @type Number
-               * @default 1
-               */
-               this.cfg.addProperty(defCfg.MY_LABEL_MONTH_POSITION.key,        { value:defCfg.MY_LABEL_MONTH_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-       
-               /**
-               * The position of the year in the month year label string used as the Calendar header
-               * @config MY_LABEL_YEAR_POSITION
-               * @type Number
-               * @default 2
-               */
-               this.cfg.addProperty(defCfg.MY_LABEL_YEAR_POSITION.key, { value:defCfg.MY_LABEL_YEAR_POSITION.value, handler:this.delegateConfig, validator:this.cfg.checkNumber } );
-               
-               /**
-               * The suffix used after the month when rendering the Calendar header
-               * @config MY_LABEL_MONTH_SUFFIX
-               * @type String
-               * @default " "
-               */
-               this.cfg.addProperty(defCfg.MY_LABEL_MONTH_SUFFIX.key,  { value:defCfg.MY_LABEL_MONTH_SUFFIX.value, handler:this.delegateConfig } );
-               
-               /**
-               * The suffix used after the year when rendering the Calendar header
-               * @config MY_LABEL_YEAR_SUFFIX
-               * @type String
-               * @default ""
-               */
-               this.cfg.addProperty(defCfg.MY_LABEL_YEAR_SUFFIX.key, { value:defCfg.MY_LABEL_YEAR_SUFFIX.value, handler:this.delegateConfig } );
-
-               /**
-               * Configuration for the Month Year Navigation UI. By default it is disabled
-               * @config NAV
-               * @type Object
-               * @default null
-               */
-               this.cfg.addProperty(defCfg.NAV.key, { value:defCfg.NAV.value, handler:this.configNavigator } );
-       },
-
-       /**
-       * Initializes CalendarGroup's built-in CustomEvents
-       * @method initEvents
-       */
-       initEvents : function() {
-               var me = this;
-               var strEvent = "Event";
-
-               /**
-               * Proxy subscriber to subscribe to the CalendarGroup's child Calendars' CustomEvents
-               * @method sub
-               * @private
-               * @param {Function} fn  The function to subscribe to this CustomEvent
-               * @param {Object}       obj     The CustomEvent's scope object
-               * @param {Boolean}      bOverride       Whether or not to apply scope correction
-               */
-               var sub = function(fn, obj, bOverride) {
-                       for (var p=0;p<me.pages.length;++p) {
-                               var cal = me.pages[p];
-                               cal[this.type + strEvent].subscribe(fn, obj, bOverride);
-                       }
-               };
-
-               /**
-               * Proxy unsubscriber to unsubscribe from the CalendarGroup's child Calendars' CustomEvents
-               * @method unsub
-               * @private
-               * @param {Function} fn  The function to subscribe to this CustomEvent
-               * @param {Object}       obj     The CustomEvent's scope object
-               */
-               var unsub = function(fn, obj) {
-                       for (var p=0;p<me.pages.length;++p) {
-                               var cal = me.pages[p];
-                               cal[this.type + strEvent].unsubscribe(fn, obj);
-                       }
-               };
-               
-               var defEvents = YAHOO.widget.Calendar._EVENT_TYPES;
-       
-               /**
-               * Fired before a selection is made
-               * @event beforeSelectEvent
-               */
-               this.beforeSelectEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_SELECT);
-               this.beforeSelectEvent.subscribe = sub; this.beforeSelectEvent.unsubscribe = unsub;
-       
-               /**
-               * Fired when a selection is made
-               * @event selectEvent
-               * @param {Array}        Array of Date field arrays in the format [YYYY, MM, DD].
-               */
-               this.selectEvent = new YAHOO.util.CustomEvent(defEvents.SELECT); 
-               this.selectEvent.subscribe = sub; this.selectEvent.unsubscribe = unsub;
-       
-               /**
-               * Fired before a selection is made
-               * @event beforeDeselectEvent
-               */
-               this.beforeDeselectEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_DESELECT); 
-               this.beforeDeselectEvent.subscribe = sub; this.beforeDeselectEvent.unsubscribe = unsub;
-       
-               /**
-               * Fired when a selection is made
-               * @event deselectEvent
-               * @param {Array}        Array of Date field arrays in the format [YYYY, MM, DD].
-               */
-               this.deselectEvent = new YAHOO.util.CustomEvent(defEvents.DESELECT); 
-               this.deselectEvent.subscribe = sub; this.deselectEvent.unsubscribe = unsub;
-               
-               /**
-               * Fired when the Calendar page is changed
-               * @event changePageEvent
-               */
-               this.changePageEvent = new YAHOO.util.CustomEvent(defEvents.CHANGE_PAGE); 
-               this.changePageEvent.subscribe = sub; this.changePageEvent.unsubscribe = unsub;
-       
-               /**
-               * Fired before the Calendar is rendered
-               * @event beforeRenderEvent
-               */
-               this.beforeRenderEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_RENDER);
-               this.beforeRenderEvent.subscribe = sub; this.beforeRenderEvent.unsubscribe = unsub;
-       
-               /**
-               * Fired when the Calendar is rendered
-               * @event renderEvent
-               */
-               this.renderEvent = new YAHOO.util.CustomEvent(defEvents.RENDER);
-               this.renderEvent.subscribe = sub; this.renderEvent.unsubscribe = unsub;
-       
-               /**
-               * Fired when the Calendar is reset
-               * @event resetEvent
-               */
-               this.resetEvent = new YAHOO.util.CustomEvent(defEvents.RESET); 
-               this.resetEvent.subscribe = sub; this.resetEvent.unsubscribe = unsub;
-       
-               /**
-               * Fired when the Calendar is cleared
-               * @event clearEvent
-               */
-               this.clearEvent = new YAHOO.util.CustomEvent(defEvents.CLEAR);
-               this.clearEvent.subscribe = sub; this.clearEvent.unsubscribe = unsub;
-       
-               /**
-               * Fired just before the CalendarGroup is to be shown
-               * @event beforeShowEvent
-               */
-               this.beforeShowEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_SHOW);
-       
-               /**
-               * Fired after the CalendarGroup is shown
-               * @event showEvent
-               */
-               this.showEvent = new YAHOO.util.CustomEvent(defEvents.SHOW);
-       
-               /**
-               * Fired just before the CalendarGroup is to be hidden
-               * @event beforeHideEvent
-               */
-               this.beforeHideEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_HIDE);
-       
-               /**
-               * Fired after the CalendarGroup is hidden
-               * @event hideEvent
-               */
-               this.hideEvent = new YAHOO.util.CustomEvent(defEvents.HIDE);
-
-               /**
-               * Fired just before the CalendarNavigator is to be shown
-               * @event beforeShowNavEvent
-               */
-               this.beforeShowNavEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_SHOW_NAV);
-       
-               /**
-               * Fired after the CalendarNavigator is shown
-               * @event showNavEvent
-               */
-               this.showNavEvent = new YAHOO.util.CustomEvent(defEvents.SHOW_NAV);
-       
-               /**
-               * Fired just before the CalendarNavigator is to be hidden
-               * @event beforeHideNavEvent
-               */
-               this.beforeHideNavEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_HIDE_NAV);
-       
-               /**
-               * Fired after the CalendarNavigator is hidden
-               * @event hideNavEvent
-               */
-               this.hideNavEvent = new YAHOO.util.CustomEvent(defEvents.HIDE_NAV);
-
-               /**
-               * Fired just before the CalendarNavigator is to be rendered
-               * @event beforeRenderNavEvent
-               */
-               this.beforeRenderNavEvent = new YAHOO.util.CustomEvent(defEvents.BEFORE_RENDER_NAV);
-
-               /**
-               * Fired after the CalendarNavigator is rendered
-               * @event renderNavEvent
-               */
-               this.renderNavEvent = new YAHOO.util.CustomEvent(defEvents.RENDER_NAV);
-       },
-       
-       /**
-       * The default Config handler for the "pages" property
-       * @method configPages
-       * @param {String} type  The CustomEvent type (usually the property name)
-       * @param {Object[]}     args    The CustomEvent arguments. For configuration handlers, args[0] will equal the newly applied value for the property.
-       * @param {Object} obj   The scope object. For configuration handlers, this will usually equal the owner.
-       */
-       configPages : function(type, args, obj) {
-               var pageCount = args[0];
-       
-               var cfgPageDate = YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGEDATE.key;
-       
-               // Define literals outside loop 
-               var sep = "_";
-               var groupCalClass = "groupcal";
-       
-               var firstClass = "first-of-type";
-               var lastClass = "last-of-type";
-       
-               for (var p=0;p<pageCount;++p) {
-                       var calId = this.id + sep + p;
-                       var calContainerId = this.containerId + sep + p;
-       
-                       var childConfig = this.cfg.getConfig();
-                       childConfig.close = false;
-                       childConfig.title = false;
-                       childConfig.navigator = null;
-
-                       var cal = this.constructChild(calId, calContainerId, childConfig);
-                       var caldate = cal.cfg.getProperty(cfgPageDate);
-                       this._setMonthOnDate(caldate, caldate.getMonth() + p);
-                       cal.cfg.setProperty(cfgPageDate, caldate);
-       
-                       YAHOO.util.Dom.removeClass(cal.oDomContainer, this.Style.CSS_SINGLE);
-                       YAHOO.util.Dom.addClass(cal.oDomContainer, groupCalClass);
-
-                       if (p===0) {
-                               YAHOO.util.Dom.addClass(cal.oDomContainer, firstClass);
-                       }
-       
-                       if (p==(pageCount-1)) {
-                               YAHOO.util.Dom.addClass(cal.oDomContainer, lastClass);
-                       }
-       
-                       cal.parent = this;
-                       cal.index = p; 
-       
-                       this.pages[this.pages.length] = cal;
-               }
-       },
-       
-       /**
-       * The default Config handler for the "pagedate" property
-       * @method configPageDate
-       * @param {String} type  The CustomEvent type (usually the property name)
-       * @param {Object[]}     args    The CustomEvent arguments. For configuration handlers, args[0] will equal the newly applied value for the property.
-       * @param {Object} obj   The scope object. For configuration handlers, this will usually equal the owner.
-       */
-       configPageDate : function(type, args, obj) {
-               var val = args[0];
-               var firstPageDate;
-               
-               var cfgPageDate = YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGEDATE.key;
-               
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       if (p === 0) {
-                               firstPageDate = cal._parsePageDate(val);
-                               cal.cfg.setProperty(cfgPageDate, firstPageDate);
-                       } else {
-                               var pageDate = new Date(firstPageDate);
-                               this._setMonthOnDate(pageDate, pageDate.getMonth() + p);
-                               cal.cfg.setProperty(cfgPageDate, pageDate);
-                       }
-               }
-       },
-       
-       /**
-       * The default Config handler for the CalendarGroup "selected" property
-       * @method configSelected
-       * @param {String} type  The CustomEvent type (usually the property name)
-       * @param {Object[]}     args    The CustomEvent arguments. For configuration handlers, args[0] will equal the newly applied value for the property.
-       * @param {Object} obj   The scope object. For configuration handlers, this will usually equal the owner.
-       */
-       configSelected : function(type, args, obj) {
-               var cfgSelected = YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.SELECTED.key;
-               this.delegateConfig(type, args, obj);
-               var selected = (this.pages.length > 0) ? this.pages[0].cfg.getProperty(cfgSelected) : []; 
-               this.cfg.setProperty(cfgSelected, selected, true);
-       },
-
-       
-       /**
-       * Delegates a configuration property to the CustomEvents associated with the CalendarGroup's children
-       * @method delegateConfig
-       * @param {String} type  The CustomEvent type (usually the property name)
-       * @param {Object[]}     args    The CustomEvent arguments. For configuration handlers, args[0] will equal the newly applied value for the property.
-       * @param {Object} obj   The scope object. For configuration handlers, this will usually equal the owner.
-       */
-       delegateConfig : function(type, args, obj) {
-               var val = args[0];
-               var cal;
-       
-               for (var p=0;p<this.pages.length;p++) {
-                       cal = this.pages[p];
-                       cal.cfg.setProperty(type, val);
-               }
-       },
-
-       /**
-       * Adds a function to all child Calendars within this CalendarGroup.
-       * @method setChildFunction
-       * @param {String}               fnName          The name of the function
-       * @param {Function}             fn                      The function to apply to each Calendar page object
-       */
-       setChildFunction : function(fnName, fn) {
-               var pageCount = this.cfg.getProperty(YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGES.key);
-       
-               for (var p=0;p<pageCount;++p) {
-                       this.pages[p][fnName] = fn;
-               }
-       },
-
-       /**
-       * Calls a function within all child Calendars within this CalendarGroup.
-       * @method callChildFunction
-       * @param {String}               fnName          The name of the function
-       * @param {Array}                args            The arguments to pass to the function
-       */
-       callChildFunction : function(fnName, args) {
-               var pageCount = this.cfg.getProperty(YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGES.key);
-       
-               for (var p=0;p<pageCount;++p) {
-                       var page = this.pages[p];
-                       if (page[fnName]) {
-                               var fn = page[fnName];
-                               fn.call(page, args);
-                       }
-               }       
-       },
-
-       /**
-       * Constructs a child calendar. This method can be overridden if a subclassed version of the default
-       * calendar is to be used.
-       * @method constructChild
-       * @param {String}       id                      The id of the table element that will represent the calendar widget
-       * @param {String}       containerId     The id of the container div element that will wrap the calendar table
-       * @param {Object}       config          The configuration object containing the Calendar's arguments
-       * @return {YAHOO.widget.Calendar}       The YAHOO.widget.Calendar instance that is constructed
-       */
-       constructChild : function(id,containerId,config) {
-               var container = document.getElementById(containerId);
-               if (! container) {
-                       container = document.createElement("div");
-                       container.id = containerId;
-                       this.oDomContainer.appendChild(container);
-               }
-               return new YAHOO.widget.Calendar(id,containerId,config);
-       },
-       
-       /**
-       * Sets the calendar group's month explicitly. This month will be set into the first
-       * page of the multi-page calendar, and all other months will be iterated appropriately.
-       * @method setMonth
-       * @param {Number}       month           The numeric month, from 0 (January) to 11 (December)
-       */
-       setMonth : function(month) {
-               month = parseInt(month, 10);
-               var currYear;
-
-               var cfgPageDate = YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGEDATE.key;
-
-               for (var p=0; p<this.pages.length; ++p) {
-                       var cal = this.pages[p];
-                       var pageDate = cal.cfg.getProperty(cfgPageDate);
-                       if (p === 0) {
-                               currYear = pageDate.getFullYear();
-                       } else {
-                               pageDate.setFullYear(currYear);
-                       }
-                       this._setMonthOnDate(pageDate, month+p); 
-                       cal.cfg.setProperty(cfgPageDate, pageDate);
-               }
-       },
-
-       /**
-       * Sets the calendar group's year explicitly. This year will be set into the first
-       * page of the multi-page calendar, and all other months will be iterated appropriately.
-       * @method setYear
-       * @param {Number}       year            The numeric 4-digit year
-       */
-       setYear : function(year) {
-       
-               var cfgPageDate = YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGEDATE.key;
-       
-               year = parseInt(year, 10);
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       var pageDate = cal.cfg.getProperty(cfgPageDate);
-       
-                       if ((pageDate.getMonth()+1) == 1 && p>0) {
-                               year+=1;
-                       }
-                       cal.setYear(year);
-               }
-       },
-
-       /**
-       * Calls the render function of all child calendars within the group.
-       * @method render
-       */
-       render : function() {
-               this.renderHeader();
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.render();
-               }
-               this.renderFooter();
-       },
-
-       /**
-       * Selects a date or a collection of dates on the current calendar. This method, by default,
-       * does not call the render method explicitly. Once selection has completed, render must be 
-       * called for the changes to be reflected visually.
-       * @method select
-       * @param        {String/Date/Date[]}    date    The date string of dates to select in the current calendar. Valid formats are
-       *                                                               individual date(s) (12/24/2005,12/26/2005) or date range(s) (12/24/2005-1/1/2006).
-       *                                                               Multiple comma-delimited dates can also be passed to this method (12/24/2005,12/11/2005-12/13/2005).
-       *                                                               This method can also take a JavaScript Date object or an array of Date objects.
-       * @return       {Date[]}                        Array of JavaScript Date objects representing all individual dates that are currently selected.
-       */
-       select : function(date) {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.select(date);
-               }
-               return this.getSelectedDates();
-       },
-
-       /**
-       * Selects dates in the CalendarGroup based on the cell index provided. This method is used to select cells without having to do a full render. The selected style is applied to the cells directly.
-       * The value of the MULTI_SELECT Configuration attribute will determine the set of dates which get selected. 
-       * <ul>
-       *    <li>If MULTI_SELECT is false, selectCell will select the cell at the specified index for only the last displayed Calendar page.</li>
-       *    <li>If MULTI_SELECT is true, selectCell will select the cell at the specified index, on each displayed Calendar page.</li>
-       * </ul>
-       * @method selectCell
-       * @param        {Number}        cellIndex       The index of the cell to be selected. 
-       * @return       {Date[]}        Array of JavaScript Date objects representing all individual dates that are currently selected.
-       */
-       selectCell : function(cellIndex) {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.selectCell(cellIndex);
-               }
-               return this.getSelectedDates();
-       },
-       
-       /**
-       * Deselects a date or a collection of dates on the current calendar. This method, by default,
-       * does not call the render method explicitly. Once deselection has completed, render must be 
-       * called for the changes to be reflected visually.
-       * @method deselect
-       * @param        {String/Date/Date[]}    date    The date string of dates to deselect in the current calendar. Valid formats are
-       *                                                               individual date(s) (12/24/2005,12/26/2005) or date range(s) (12/24/2005-1/1/2006).
-       *                                                               Multiple comma-delimited dates can also be passed to this method (12/24/2005,12/11/2005-12/13/2005).
-       *                                                               This method can also take a JavaScript Date object or an array of Date objects. 
-       * @return       {Date[]}                        Array of JavaScript Date objects representing all individual dates that are currently selected.
-       */
-       deselect : function(date) {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.deselect(date);
-               }
-               return this.getSelectedDates();
-       },
-       
-       /**
-       * Deselects all dates on the current calendar.
-       * @method deselectAll
-       * @return {Date[]}              Array of JavaScript Date objects representing all individual dates that are currently selected.
-       *                                               Assuming that this function executes properly, the return value should be an empty array.
-       *                                               However, the empty array is returned for the sake of being able to check the selection status
-       *                                               of the calendar.
-       */
-       deselectAll : function() {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.deselectAll();
-               }
-               return this.getSelectedDates();
-       },
-       
-       /**
-       * Deselects dates in the CalendarGroup based on the cell index provided. This method is used to select cells without having to do a full render. The selected style is applied to the cells directly.
-       * deselectCell will deselect the cell at the specified index on each displayed Calendar page.
-       *
-       * @method deselectCell
-       * @param        {Number}        cellIndex       The index of the cell to deselect. 
-       * @return       {Date[]}        Array of JavaScript Date objects representing all individual dates that are currently selected.
-       */
-       deselectCell : function(cellIndex) {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.deselectCell(cellIndex);
-               }
-               return this.getSelectedDates();
-       },
-       
-       /**
-       * Resets the calendar widget to the originally selected month and year, and 
-       * sets the calendar to the initial selection(s).
-       * @method reset
-       */
-       reset : function() {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.reset();
-               }
-       },
-       
-       /**
-       * Clears the selected dates in the current calendar widget and sets the calendar
-       * to the current month and year.
-       * @method clear
-       */
-       clear : function() {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.clear();
-               }
-       },
-       
-       /**
-       * Navigates to the next month page in the calendar widget.
-       * @method nextMonth
-       */
-       nextMonth : function() {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.nextMonth();
-               }
-       },
-       
-       /**
-       * Navigates to the previous month page in the calendar widget.
-       * @method previousMonth
-       */
-       previousMonth : function() {
-               for (var p=this.pages.length-1;p>=0;--p) {
-                       var cal = this.pages[p];
-                       cal.previousMonth();
-               }
-       },
-       
-       /**
-       * Navigates to the next year in the currently selected month in the calendar widget.
-       * @method nextYear
-       */
-       nextYear : function() {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.nextYear();
-               }
-       },
-
-       /**
-       * Navigates to the previous year in the currently selected month in the calendar widget.
-       * @method previousYear
-       */
-       previousYear : function() {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.previousYear();
-               }
-       },
-
-       /**
-       * Gets the list of currently selected dates from the calendar.
-       * @return                       An array of currently selected JavaScript Date objects.
-       * @type Date[]
-       */
-       getSelectedDates : function() { 
-               var returnDates = [];
-               var selected = this.cfg.getProperty(YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.SELECTED.key);
-               for (var d=0;d<selected.length;++d) {
-                       var dateArray = selected[d];
-
-                       var date = YAHOO.widget.DateMath.getDate(dateArray[0],dateArray[1]-1,dateArray[2]);
-                       returnDates.push(date);
-               }
-
-               returnDates.sort( function(a,b) { return a-b; } );
-               return returnDates;
-       },
-
-       /**
-       * Adds a renderer to the render stack. The function reference passed to this method will be executed
-       * when a date cell matches the conditions specified in the date string for this renderer.
-       * @method addRenderer
-       * @param        {String}        sDates          A date string to associate with the specified renderer. Valid formats
-       *                                                                       include date (12/24/2005), month/day (12/24), and range (12/1/2004-1/1/2005)
-       * @param        {Function}      fnRender        The function executed to render cells that match the render rules for this renderer.
-       */
-       addRenderer : function(sDates, fnRender) {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.addRenderer(sDates, fnRender);
-               }
-       },
-
-       /**
-       * Adds a month to the render stack. The function reference passed to this method will be executed
-       * when a date cell matches the month passed to this method.
-       * @method addMonthRenderer
-       * @param        {Number}        month           The month (1-12) to associate with this renderer
-       * @param        {Function}      fnRender        The function executed to render cells that match the render rules for this renderer.
-       */
-       addMonthRenderer : function(month, fnRender) {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.addMonthRenderer(month, fnRender);
-               }
-       },
-
-       /**
-       * Adds a weekday to the render stack. The function reference passed to this method will be executed
-       * when a date cell matches the weekday passed to this method.
-       * @method addWeekdayRenderer
-       * @param        {Number}        weekday         The weekday (1-7) to associate with this renderer. 1=Sunday, 2=Monday etc.
-       * @param        {Function}      fnRender        The function executed to render cells that match the render rules for this renderer.
-       */
-       addWeekdayRenderer : function(weekday, fnRender) {
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       cal.addWeekdayRenderer(weekday, fnRender);
-               }
-       },
-
-       /**
-        * Removes all custom renderers added to the CalendarGroup through the addRenderer, addMonthRenderer and 
-        * addWeekRenderer methods. CalendarGroup's render method needs to be called to after removing renderers 
-        * to see the changes applied.
-        * 
-        * @method removeRenderers
-        */
-       removeRenderers : function() {
-               this.callChildFunction("removeRenderers");
-       },
-
-       /**
-       * Renders the header for the CalendarGroup.
-       * @method renderHeader
-       */
-       renderHeader : function() {
-               // EMPTY DEFAULT IMPL
-       },
-
-       /**
-       * Renders a footer for the 2-up calendar container. By default, this method is
-       * unimplemented.
-       * @method renderFooter
-       */
-       renderFooter : function() {
-               // EMPTY DEFAULT IMPL
-       },
-
-       /**
-       * Adds the designated number of months to the current calendar month, and sets the current
-       * calendar page date to the new month.
-       * @method addMonths
-       * @param {Number}       count   The number of months to add to the current calendar
-       */
-       addMonths : function(count) {
-               this.callChildFunction("addMonths", count);
-       },
-       
-       /**
-       * Subtracts the designated number of months from the current calendar month, and sets the current
-       * calendar page date to the new month.
-       * @method subtractMonths
-       * @param {Number}       count   The number of months to subtract from the current calendar
-       */
-       subtractMonths : function(count) {
-               this.callChildFunction("subtractMonths", count);
-       },
-
-       /**
-       * Adds the designated number of years to the current calendar, and sets the current
-       * calendar page date to the new month.
-       * @method addYears
-       * @param {Number}       count   The number of years to add to the current calendar
-       */
-       addYears : function(count) {
-               this.callChildFunction("addYears", count);
-       },
-
-       /**
-       * Subtcats the designated number of years from the current calendar, and sets the current
-       * calendar page date to the new month.
-       * @method subtractYears
-       * @param {Number}       count   The number of years to subtract from the current calendar
-       */
-       subtractYears : function(count) {
-               this.callChildFunction("subtractYears", count);
-       },
-
-       /**
-        * Returns the Calendar page instance which has a pagedate (month/year) matching the given date. 
-        * Returns null if no match is found.
-        * 
-        * @method getCalendarPage
-        * @param {Date} date The JavaScript Date object for which a Calendar page is to be found.
-        * @return {Calendar} The Calendar page instance representing the month to which the date 
-        * belongs.
-        */
-       getCalendarPage : function(date) {
-               var cal = null;
-               if (date) {
-                       var y = date.getFullYear(),
-                               m = date.getMonth();
-
-                       var pages = this.pages;
-                       for (var i = 0; i < pages.length; ++i) {
-                               var pageDate = pages[i].cfg.getProperty("pagedate");
-                               if (pageDate.getFullYear() === y && pageDate.getMonth() === m) {
-                                       cal = pages[i];
-                                       break;
-                               }
-                       }
-               }
-               return cal;
-       },
-
-       /**
-       * Sets the month on a Date object, taking into account year rollover if the month is less than 0 or greater than 11.
-       * The Date object passed in is modified. It should be cloned before passing it into this method if the original value needs to be maintained
-       * @method       _setMonthOnDate
-       * @private
-       * @param        {Date}  date    The Date object on which to set the month index
-       * @param        {Number}        iMonth  The month index to set
-       */
-       _setMonthOnDate : function(date, iMonth) {
-               // Bug in Safari 1.3, 2.0 (WebKit build < 420), Date.setMonth does not work consistently if iMonth is not 0-11
-               if (YAHOO.env.ua.webkit && YAHOO.env.ua.webkit < 420 && (iMonth < 0 || iMonth > 11)) {
-                       var DM = YAHOO.widget.DateMath;
-                       var newDate = DM.add(date, DM.MONTH, iMonth-date.getMonth());
-                       date.setTime(newDate.getTime());
-               } else {
-                       date.setMonth(iMonth);
-               }
-       },
-       
-       /**
-        * Fixes the width of the CalendarGroup container element, to account for miswrapped floats
-        * @method _fixWidth
-        * @private
-        */
-       _fixWidth : function() {
-               var w = 0;
-               for (var p=0;p<this.pages.length;++p) {
-                       var cal = this.pages[p];
-                       w += cal.oDomContainer.offsetWidth;
-               }
-               if (w > 0) {
-                       this.oDomContainer.style.width = w + "px";
-               }
-       },
-       
-       /**
-       * Returns a string representation of the object.
-       * @method toString
-       * @return {String}      A string representation of the CalendarGroup object.
-       */
-       toString : function() {
-               return "CalendarGroup " + this.id;
-       }
-};
-
-/**
-* CSS class representing the container for the calendar
-* @property YAHOO.widget.CalendarGroup.CSS_CONTAINER
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.CalendarGroup.CSS_CONTAINER = "yui-calcontainer";
-
-/**
-* CSS class representing the container for the calendar
-* @property YAHOO.widget.CalendarGroup.CSS_MULTI_UP
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.CalendarGroup.CSS_MULTI_UP = "multi";
-
-/**
-* CSS class representing the title for the 2-up calendar
-* @property YAHOO.widget.CalendarGroup.CSS_2UPTITLE
-* @static
-* @final
-* @type String
-*/
-YAHOO.widget.CalendarGroup.CSS_2UPTITLE = "title";
-
-/**
-* CSS class representing the close icon for the 2-up calendar
-* @property YAHOO.widget.CalendarGroup.CSS_2UPCLOSE
-* @static
-* @final
-* @deprecated  Along with Calendar.IMG_ROOT and NAV_ARROW_LEFT, NAV_ARROW_RIGHT configuration properties.
-*                                      Calendar's <a href="YAHOO.widget.Calendar.html#Style.CSS_CLOSE">Style.CSS_CLOSE</a> property now represents the CSS class used to render the close icon
-* @type String
-*/
-YAHOO.widget.CalendarGroup.CSS_2UPCLOSE = "close-icon";
-
-YAHOO.lang.augmentProto(YAHOO.widget.CalendarGroup, YAHOO.widget.Calendar, "buildDayLabel",
-                                                                                                                                "buildMonthLabel",
-                                                                                                                                "renderOutOfBoundsDate",
-                                                                                                                                "renderRowHeader",
-                                                                                                                                "renderRowFooter",
-                                                                                                                                "renderCellDefault",
-                                                                                                                                "styleCellDefault",
-                                                                                                                                "renderCellStyleHighlight1",
-                                                                                                                                "renderCellStyleHighlight2",
-                                                                                                                                "renderCellStyleHighlight3",
-                                                                                                                                "renderCellStyleHighlight4",
-                                                                                                                                "renderCellStyleToday",
-                                                                                                                                "renderCellStyleSelected",
-                                                                                                                                "renderCellNotThisMonth",
-                                                                                                                                "renderBodyCellRestricted",
-                                                                                                                                "initStyles",
-                                                                                                                                "configTitle",
-                                                                                                                                "configClose",
-                                                                                                                                "configIframe",
-                                                                                                                                "configNavigator",
-                                                                                                                                "createTitleBar",
-                                                                                                                                "createCloseButton",
-                                                                                                                                "removeTitleBar",
-                                                                                                                                "removeCloseButton",
-                                                                                                                                "hide",
-                                                                                                                                "show",
-                                                                                                                                "toDate",
-                                                                                                                                "_parseArgs",
-                                                                                                                                "browser");
-
-/**
-* The set of default Config property keys and values for the CalendarGroup
-* @property YAHOO.widget.CalendarGroup._DEFAULT_CONFIG
-* @final
-* @static
-* @private
-* @type Object
-*/
-YAHOO.widget.CalendarGroup._DEFAULT_CONFIG = YAHOO.widget.Calendar._DEFAULT_CONFIG;
-YAHOO.widget.CalendarGroup._DEFAULT_CONFIG.PAGES = {key:"pages", value:2};
-
-YAHOO.widget.CalGrp = YAHOO.widget.CalendarGroup;
-
-/**
-* @class YAHOO.widget.Calendar2up
-* @extends YAHOO.widget.CalendarGroup
-* @deprecated The old Calendar2up class is no longer necessary, since CalendarGroup renders in a 2up view by default.
-*/
-YAHOO.widget.Calendar2up = function(id, containerId, config) {
-       this.init(id, containerId, config);
-};
-
-YAHOO.extend(YAHOO.widget.Calendar2up, YAHOO.widget.CalendarGroup);
-
-/**
-* @deprecated The old Calendar2up class is no longer necessary, since CalendarGroup renders in a 2up view by default.
-*/
-YAHOO.widget.Cal2up = YAHOO.widget.Calendar2up;
-
-/**
- * The CalendarNavigator is used along with a Calendar/CalendarGroup to 
- * provide a Month/Year popup navigation control, allowing the user to navigate 
- * to a specific month/year in the Calendar/CalendarGroup without having to 
- * scroll through months sequentially
- *
- * @namespace YAHOO.widget
- * @class CalendarNavigator
- * @constructor
- * @param {Calendar|CalendarGroup} cal The instance of the Calendar or CalendarGroup to which this CalendarNavigator should be attached.
- */
-YAHOO.widget.CalendarNavigator = function(cal) {
-       this.init(cal);
-};
-
-(function() {
-       // Setup static properties (inside anon fn, so that we can use shortcuts)
-       var CN = YAHOO.widget.CalendarNavigator;
-
-       /**
-        * YAHOO.widget.CalendarNavigator.CLASSES contains constants
-        * for the class values applied to the CalendarNaviatgator's 
-        * DOM elements
-        * @property YAHOO.widget.CalendarNavigator.CLASSES
-        * @type Object
-        * @static
-        */
-       CN.CLASSES = {
-               /**
-                * Class applied to the Calendar Navigator's bounding box
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.NAV
-                * @type String
-                * @static
-                */
-               NAV :"yui-cal-nav",
-               /**
-                * Class applied to the Calendar/CalendarGroup's bounding box to indicate
-                * the Navigator is currently visible
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.NAV_VISIBLE
-                * @type String
-                * @static
-                */
-               NAV_VISIBLE: "yui-cal-nav-visible",
-               /**
-                * Class applied to the Navigator mask's bounding box
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.MASK
-                * @type String
-                * @static
-                */
-               MASK : "yui-cal-nav-mask",
-               /**
-                * Class applied to the year label/control bounding box
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.YEAR
-                * @type String
-                * @static
-                */
-               YEAR : "yui-cal-nav-y",
-               /**
-                * Class applied to the month label/control bounding box
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.MONTH
-                * @type String
-                * @static
-                */
-               MONTH : "yui-cal-nav-m",
-               /**
-                * Class applied to the submit/cancel button's bounding box
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.BUTTONS
-                * @type String
-                * @static
-                */
-               BUTTONS : "yui-cal-nav-b",
-               /**
-                * Class applied to buttons wrapping element
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.BUTTON
-                * @type String
-                * @static
-                */
-               BUTTON : "yui-cal-nav-btn",
-               /**
-                * Class applied to the validation error area's bounding box
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.ERROR
-                * @type String
-                * @static
-                */
-               ERROR : "yui-cal-nav-e",
-               /**
-                * Class applied to the year input control
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.YEAR_CTRL
-                * @type String
-                * @static
-                */
-               YEAR_CTRL : "yui-cal-nav-yc",
-               /**
-                * Class applied to the month input control
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.MONTH_CTRL
-                * @type String
-                * @static
-                */
-               MONTH_CTRL : "yui-cal-nav-mc",
-               /**
-                * Class applied to controls with invalid data (e.g. a year input field with invalid an year)
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.INVALID
-                * @type String
-                * @static
-                */
-               INVALID : "yui-invalid",
-               /**
-                * Class applied to default controls
-                * @property YAHOO.widget.CalendarNavigator.CLASSES.DEFAULT
-                * @type String
-                * @static
-                */
-               DEFAULT : "yui-default"
-       };
-
-       /**
-        * Object literal containing the default configuration values for the CalendarNavigator
-        * The configuration object is expected to follow the format below, with the properties being
-        * case sensitive.
-        * <dl>
-        * <dt>strings</dt>
-        * <dd><em>Object</em> :  An object with the properties shown below, defining the string labels to use in the Navigator's UI
-        *     <dl>
-        *         <dt>month</dt><dd><em>String</em> : The string to use for the month label. Defaults to "Month".</dd>
-        *         <dt>year</dt><dd><em>String</em> : The string to use for the year label. Defaults to "Year".</dd>
-        *         <dt>submit</dt><dd><em>String</em> : The string to use for the submit button label. Defaults to "Okay".</dd>
-        *         <dt>cancel</dt><dd><em>String</em> : The string to use for the cancel button label. Defaults to "Cancel".</dd>
-        *         <dt>invalidYear</dt><dd><em>String</em> : The string to use for invalid year values. Defaults to "Year needs to be a number".</dd>
-        *     </dl>
-        * </dd>
-        * <dt>monthFormat</dt><dd><em>String</em> : The month format to use. Either YAHOO.widget.Calendar.LONG, or YAHOO.widget.Calendar.SHORT. Defaults to YAHOO.widget.Calendar.LONG</dd>
-        * <dt>initialFocus</dt><dd><em>String</em> : Either "year" or "month" specifying which input control should get initial focus. Defaults to "year"</dd>
-        * </dl>
-        * @property _DEFAULT_CFG
-        * @protected
-        * @type Object
-        * @static
-        */
-       CN._DEFAULT_CFG = {
-               strings : {
-                       month: "Month",
-                       year: "Year",
-                       submit: "Okay",
-                       cancel: "Cancel",
-                       invalidYear : "Year needs to be a number"
-               },
-               monthFormat: YAHOO.widget.Calendar.LONG,
-               initialFocus: "year"
-       };
-
-       /**
-        * The suffix added to the Calendar/CalendarGroup's ID, to generate
-        * a unique ID for the Navigator and it's bounding box.
-        * @property YAHOO.widget.CalendarNavigator.ID_SUFFIX
-        * @static
-        * @type String
-        * @final
-        */
-       CN.ID_SUFFIX = "_nav";
-       /**
-        * The suffix added to the Navigator's ID, to generate
-        * a unique ID for the month control.
-        * @property YAHOO.widget.CalendarNavigator.MONTH_SUFFIX
-        * @static
-        * @type String 
-        * @final
-        */
-       CN.MONTH_SUFFIX = "_month";
-       /**
-        * The suffix added to the Navigator's ID, to generate
-        * a unique ID for the year control.
-        * @property YAHOO.widget.CalendarNavigator.YEAR_SUFFIX
-        * @static
-        * @type String
-        * @final
-        */
-       CN.YEAR_SUFFIX = "_year";
-       /**
-        * The suffix added to the Navigator's ID, to generate
-        * a unique ID for the error bounding box.
-        * @property YAHOO.widget.CalendarNavigator.ERROR_SUFFIX
-        * @static
-        * @type String
-        * @final
-        */
-       CN.ERROR_SUFFIX = "_error";
-       /**
-        * The suffix added to the Navigator's ID, to generate
-        * a unique ID for the "Cancel" button.
-        * @property YAHOO.widget.CalendarNavigator.CANCEL_SUFFIX
-        * @static
-        * @type String
-        * @final
-        */
-       CN.CANCEL_SUFFIX = "_cancel";
-       /**
-        * The suffix added to the Navigator's ID, to generate
-        * a unique ID for the "Submit" button.
-        * @property YAHOO.widget.CalendarNavigator.SUBMIT_SUFFIX
-        * @static
-        * @type String
-        * @final
-        */
-       CN.SUBMIT_SUFFIX = "_submit";
-
-       /**
-        * The number of digits to which the year input control is to be limited.
-        * @property YAHOO.widget.CalendarNavigator.YR_MAX_DIGITS
-        * @static
-        * @type Number
-        */
-       CN.YR_MAX_DIGITS = 4;
-
-       /**
-        * The amount by which to increment the current year value,
-        * when the arrow up/down key is pressed on the year control
-        * @property YAHOO.widget.CalendarNavigator.YR_MINOR_INC
-        * @static
-        * @type Number
-        */
-       CN.YR_MINOR_INC = 1;
-
-       /**
-        * The amount by which to increment the current year value,
-        * when the page up/down key is pressed on the year control
-        * @property YAHOO.widget.CalendarNavigator.YR_MAJOR_INC
-        * @static
-        * @type Number
-        */
-       CN.YR_MAJOR_INC = 10;
-
-       /**
-        * Artificial delay (in ms) between the time the Navigator is hidden
-        * and the Calendar/CalendarGroup state is updated. Allows the user
-        * the see the Calendar/CalendarGroup page changing. If set to 0
-        * the Calendar/CalendarGroup page will be updated instantly
-        * @property YAHOO.widget.CalendarNavigator.UPDATE_DELAY
-        * @static
-        * @type Number
-        */
-       CN.UPDATE_DELAY = 50;
-
-       /**
-        * Regular expression used to validate the year input
-        * @property YAHOO.widget.CalendarNavigator.YR_PATTERN
-        * @static
-        * @type RegExp
-        */
-       CN.YR_PATTERN = /^\d+$/;
-       /**
-        * Regular expression used to trim strings
-        * @property YAHOO.widget.CalendarNavigator.TRIM
-        * @static
-        * @type RegExp
-        */
-       CN.TRIM = /^\s*(.*?)\s*$/;
-})();
-
-YAHOO.widget.CalendarNavigator.prototype = {
-
-       /**
-        * The unique ID for this CalendarNavigator instance
-        * @property id
-        * @type String
-        */
-       id : null,
-
-       /**
-        * The Calendar/CalendarGroup instance to which the navigator belongs
-        * @property cal
-        * @type {Calendar|CalendarGroup}
-        */
-       cal : null,
-
-       /**
-        * Reference to the HTMLElement used to render the navigator's bounding box
-        * @property navEl
-        * @type HTMLElement
-        */
-       navEl : null,
-
-       /**
-        * Reference to the HTMLElement used to render the navigator's mask
-        * @property maskEl
-        * @type HTMLElement
-        */
-       maskEl : null,
-
-       /**
-        * Reference to the HTMLElement used to input the year
-        * @property yearEl
-        * @type HTMLElement
-        */
-       yearEl : null,
-
-       /**
-        * Reference to the HTMLElement used to input the month
-        * @property monthEl
-        * @type HTMLElement
-        */
-       monthEl : null,
-
-       /**
-        * Reference to the HTMLElement used to display validation errors
-        * @property errorEl
-        * @type HTMLElement
-        */
-       errorEl : null,
-
-       /**
-        * Reference to the HTMLElement used to update the Calendar/Calendar group
-        * with the month/year values
-        * @property submitEl
-        * @type HTMLElement
-        */
-       submitEl : null,
-       
-       /**
-        * Reference to the HTMLElement used to hide the navigator without updating the 
-        * Calendar/Calendar group
-        * @property cancelEl
-        * @type HTMLElement
-        */
-       cancelEl : null,
-
-       /** 
-        * Reference to the first focusable control in the navigator (by default monthEl)
-        * @property firstCtrl
-        * @type HTMLElement
-        */
-       firstCtrl : null,
-       
-       /** 
-        * Reference to the last focusable control in the navigator (by default cancelEl)
-        * @property lastCtrl
-        * @type HTMLElement
-        */
-       lastCtrl : null,
-
-       /**
-        * The document containing the Calendar/Calendar group instance
-        * @protected
-        * @property _doc
-        * @type HTMLDocument
-        */
-       _doc : null,
-
-       /**
-        * Internal state property for the current year displayed in the navigator
-        * @protected
-        * @property _year
-        * @type Number
-        */
-       _year: null,
-       
-       /**
-        * Internal state property for the current month index displayed in the navigator
-        * @protected
-        * @property _month
-        * @type Number
-        */
-       _month: 0,
-
-       /**
-        * Private internal state property which indicates whether or not the 
-        * Navigator has been rendered.
-        * @private
-        * @property __rendered
-        * @type Boolean
-        */
-       __rendered: false,
-
-       /**
-        * Init lifecycle method called as part of construction
-        * 
-        * @method init
-        * @param {Calendar} cal The instance of the Calendar or CalendarGroup to which this CalendarNavigator should be attached
-        */
-       init : function(cal) {
-               var calBox = cal.oDomContainer;
-
-               this.cal = cal;
-               this.id = calBox.id + YAHOO.widget.CalendarNavigator.ID_SUFFIX;
-               this._doc = calBox.ownerDocument;
-
-               /**
-                * Private flag, to identify IE6/IE7 Quirks
-                * @private
-                * @property __isIEQuirks
-                */
-               var ie = YAHOO.env.ua.ie;
-               this.__isIEQuirks = (ie && ((ie <= 6) || (ie === 7 && this._doc.compatMode == "BackCompat")));
-       },
-
-       /**
-        * Displays the navigator and mask, updating the input controls to reflect the 
-        * currently set month and year. The show method will invoke the render method
-        * if the navigator has not been renderered already, allowing for lazy rendering
-        * of the control.
-        * 
-        * The show method will fire the Calendar/CalendarGroup's beforeShowNav and showNav events
-        * 
-        * @method show
-        */
-       show : function() {
-               var CLASSES = YAHOO.widget.CalendarNavigator.CLASSES;
-
-               if (this.cal.beforeShowNavEvent.fire()) {
-                       if (!this.__rendered) {
-                               this.render();
-                       }
-                       this.clearErrors();
-
-                       this._updateMonthUI();
-                       this._updateYearUI();
-                       this._show(this.navEl, true);
-
-                       this.setInitialFocus();
-                       this.showMask();
-
-                       YAHOO.util.Dom.addClass(this.cal.oDomContainer, CLASSES.NAV_VISIBLE);
-                       this.cal.showNavEvent.fire();
-               }
-       },
-
-       /**
-        * Hides the navigator and mask
-        * 
-        * The show method will fire the Calendar/CalendarGroup's beforeHideNav event and hideNav events
-        * @method hide
-        */
-       hide : function() {
-               var CLASSES = YAHOO.widget.CalendarNavigator.CLASSES;
-
-               if (this.cal.beforeHideNavEvent.fire()) {
-                       this._show(this.navEl, false);
-                       this.hideMask();
-                       YAHOO.util.Dom.removeClass(this.cal.oDomContainer, CLASSES.NAV_VISIBLE);
-                       this.cal.hideNavEvent.fire();
-               }
-       },
-       
-
-       /**
-        * Displays the navigator's mask element
-        * 
-        * @method showMask
-        */
-       showMask : function() {
-               this._show(this.maskEl, true);
-               if (this.__isIEQuirks) {
-                       this._syncMask();
-               }
-       },
-
-       /**
-        * Hides the navigator's mask element
-        * 
-        * @method hideMask
-        */
-       hideMask : function() {
-               this._show(this.maskEl, false);
-       },
-
-       /**
-        * Returns the current month set on the navigator
-        * 
-        * Note: This may not be the month set in the UI, if 
-        * the UI contains an invalid value.
-        * 
-        * @method getMonth
-        * @return {Number} The Navigator's current month index
-        */
-       getMonth: function() {
-               return this._month;
-       },
-
-       /**
-        * Returns the current year set on the navigator
-        * 
-        * Note: This may not be the year set in the UI, if 
-        * the UI contains an invalid value.
-        * 
-        * @method getYear
-        * @return {Number} The Navigator's current year value
-        */
-       getYear: function() {
-               return this._year;
-       },
-
-       /**
-        * Sets the current month on the Navigator, and updates the UI
-        * 
-        * @method setMonth
-        * @param {Number} nMonth The month index, from 0 (Jan) through 11 (Dec).
-        */
-       setMonth : function(nMonth) {
-               if (nMonth >= 0 && nMonth < 12) {
-                       this._month = nMonth;
-               }
-               this._updateMonthUI();
-       },
-
-       /**
-        * Sets the current year on the Navigator, and updates the UI. If the 
-        * provided year is invalid, it will not be set.
-        * 
-        * @method setYear
-        * @param {Number} nYear The full year value to set the Navigator to.
-        */
-       setYear : function(nYear) {
-               var yrPattern = YAHOO.widget.CalendarNavigator.YR_PATTERN;
-               if (YAHOO.lang.isNumber(nYear) && yrPattern.test(nYear+"")) {
-                       this._year = nYear;
-               }
-               this._updateYearUI();
-       },
-
-       /**
-        * Renders the HTML for the navigator, adding it to the 
-        * document and attaches event listeners if it has not 
-        * already been rendered.
-        * 
-        * @method render
-        */
-       render: function() {
-               this.cal.beforeRenderNavEvent.fire();
-               if (!this.__rendered) {
-                       this.createNav();
-                       this.createMask();
-                       this.applyListeners();
-                       this.__rendered = true;
-               }
-               this.cal.renderNavEvent.fire();
-       },
-
-       /**
-        * Creates the navigator's containing HTMLElement, it's contents, and appends 
-        * the containg element to the Calendar/CalendarGroup's container.
-        * 
-        * @method createNav
-        */
-       createNav : function() {
-               var NAV = YAHOO.widget.CalendarNavigator;
-               var doc = this._doc;
-
-               var d = doc.createElement("div");
-               d.className = NAV.CLASSES.NAV;
-
-               var htmlBuf = this.renderNavContents([]);
-
-               d.innerHTML = htmlBuf.join('');
-               this.cal.oDomContainer.appendChild(d);
-
-               this.navEl = d;
-
-               this.yearEl = doc.getElementById(this.id + NAV.YEAR_SUFFIX);
-               this.monthEl = doc.getElementById(this.id + NAV.MONTH_SUFFIX);
-               this.errorEl = doc.getElementById(this.id + NAV.ERROR_SUFFIX);
-               this.submitEl = doc.getElementById(this.id + NAV.SUBMIT_SUFFIX);
-               this.cancelEl = doc.getElementById(this.id + NAV.CANCEL_SUFFIX);
-
-               if (YAHOO.env.ua.gecko && this.yearEl && this.yearEl.type == "text") {
-                       // Avoid XUL error on focus, select [ https://bugzilla.mozilla.org/show_bug.cgi?id=236791, 
-                       // supposedly fixed in 1.8.1, but there are reports of it still being around for methods other than blur ]
-                       this.yearEl.setAttribute("autocomplete", "off");
-               }
-
-               this._setFirstLastElements();
-       },
-
-       /**
-        * Creates the Mask HTMLElement and appends it to the Calendar/CalendarGroups
-        * container.
-        * 
-        * @method createMask
-        */
-       createMask : function() {
-               var C = YAHOO.widget.CalendarNavigator.CLASSES;
-
-               var d = this._doc.createElement("div");
-               d.className = C.MASK;
-
-               this.cal.oDomContainer.appendChild(d);
-               this.maskEl = d;
-       },
-
-       /**
-        * Used to set the width/height of the mask in pixels to match the Calendar Container.
-        * Currently only used for IE6 and IE7 quirks mode. The other A-Grade browser are handled using CSS (width/height 100%).
-        * <p>
-        * The method is also registered as an HTMLElement resize listener on the Calendars container element.
-        * </p>
-        * @protected
-        * @method _syncMask
-        */
-       _syncMask : function() {
-               var c = this.cal.oDomContainer;
-               if (c && this.maskEl) {
-                       var r = YAHOO.util.Dom.getRegion(c);
-                       YAHOO.util.Dom.setStyle(this.maskEl, "width", r.right - r.left + "px");
-                       YAHOO.util.Dom.setStyle(this.maskEl, "height", r.bottom - r.top + "px");
-               }
-       },
-
-       /**
-        * Renders the contents of the navigator
-        * 
-        * @method renderNavContents
-        * 
-        * @param {Array} html The HTML buffer to append the HTML to.
-        * @return {Array} A reference to the buffer passed in.
-        */
-       renderNavContents : function(html) {
-               var NAV = YAHOO.widget.CalendarNavigator,
-                       C = NAV.CLASSES,
-                       h = html; // just to use a shorter name
-
-               h[h.length] = '<div class="' + C.MONTH + '">';
-               this.renderMonth(h);
-               h[h.length] = '</div>';
-               h[h.length] = '<div class="' + C.YEAR + '">';
-               this.renderYear(h);
-               h[h.length] = '</div>';
-               h[h.length] = '<div class="' + C.BUTTONS + '">';
-               this.renderButtons(h);
-               h[h.length] = '</div>';
-               h[h.length] = '<div class="' + C.ERROR + '" id="' + this.id + NAV.ERROR_SUFFIX + '"></div>';
-
-               return h;
-       },
-
-       /**
-        * Renders the month label and control for the navigator
-        * 
-        * @method renderNavContents
-        * @param {Array} html The HTML buffer to append the HTML to.
-        * @return {Array} A reference to the buffer passed in.
-        */
-       renderMonth : function(html) {
-               var NAV = YAHOO.widget.CalendarNavigator,
-                       C = NAV.CLASSES;
-
-               var id = this.id + NAV.MONTH_SUFFIX,
-                       mf = this.__getCfg("monthFormat"),
-                       months = this.cal.cfg.getProperty((mf == YAHOO.widget.Calendar.SHORT) ? "MONTHS_SHORT" : "MONTHS_LONG"),
-                       h = html;
-
-               if (months && months.length > 0) {
-                       h[h.length] = '<label for="' + id + '">';
-                       h[h.length] = this.__getCfg("month", true);
-                       h[h.length] = '</label>';
-                       h[h.length] = '<select name="' + id + '" id="' + id + '" class="' + C.MONTH_CTRL + '">';
-                       for (var i = 0; i < months.length; i++) {
-                               h[h.length] = '<option value="' + i + '">';
-                               h[h.length] = months[i];
-                               h[h.length] = '</option>';
-                       }
-                       h[h.length] = '</select>';
-               }
-               return h;
-       },
-
-       /**
-        * Renders the year label and control for the navigator
-        * 
-        * @method renderYear
-        * @param {Array} html The HTML buffer to append the HTML to.
-        * @return {Array} A reference to the buffer passed in.
-        */
-       renderYear : function(html) {
-               var NAV = YAHOO.widget.CalendarNavigator,
-                       C = NAV.CLASSES;
-
-               var id = this.id + NAV.YEAR_SUFFIX,
-                       size = NAV.YR_MAX_DIGITS,
-                       h = html;
-
-               h[h.length] = '<label for="' + id + '">';
-               h[h.length] = this.__getCfg("year", true);
-               h[h.length] = '</label>';
-               h[h.length] = '<input type="text" name="' + id + '" id="' + id + '" class="' + C.YEAR_CTRL + '" maxlength="' + size + '"/>';
-               return h;
-       },
-
-       /**
-        * Renders the submit/cancel buttons for the navigator
-        * 
-        * @method renderButton
-        * @return {String} The HTML created for the Button UI
-        */
-       renderButtons : function(html) {
-               var C = YAHOO.widget.CalendarNavigator.CLASSES;
-               var h = html;
-
-               h[h.length] = '<span class="' + C.BUTTON + ' ' + C.DEFAULT + '">';
-               h[h.length] = '<button type="button" id="' + this.id + '_submit' + '">';
-               h[h.length] = this.__getCfg("submit", true);
-               h[h.length] = '</button>';
-               h[h.length] = '</span>';
-               h[h.length] = '<span class="' + C.BUTTON +'">';
-               h[h.length] = '<button type="button" id="' + this.id + '_cancel' + '">';
-               h[h.length] = this.__getCfg("cancel", true);
-               h[h.length] = '</button>';
-               h[h.length] = '</span>';
-
-               return h;
-       },
-
-       /**
-        * Attaches DOM event listeners to the rendered elements
-        * <p>
-        * The method will call applyKeyListeners, to setup keyboard specific 
-        * listeners
-        * </p>
-        * @method applyListeners
-        */
-       applyListeners : function() {
-               var E = YAHOO.util.Event;
-
-               function yearUpdateHandler() {
-                       if (this.validate()) {
-                               this.setYear(this._getYearFromUI());
-                       }
-               }
-
-               function monthUpdateHandler() {
-                       this.setMonth(this._getMonthFromUI());
-               }
-
-               E.on(this.submitEl, "click", this.submit, this, true);
-               E.on(this.cancelEl, "click", this.cancel, this, true);
-               E.on(this.yearEl, "blur", yearUpdateHandler, this, true);
-               E.on(this.monthEl, "change", monthUpdateHandler, this, true);
-
-               if (this.__isIEQuirks) {
-                       YAHOO.util.Event.on(this.cal.oDomContainer, "resize", this._syncMask, this, true);
-               }
-
-               this.applyKeyListeners();
-       },
-
-       /**
-        * Removes/purges DOM event listeners from the rendered elements
-        * 
-        * @method purgeListeners
-        */
-       purgeListeners : function() {
-               var E = YAHOO.util.Event;
-               E.removeListener(this.submitEl, "click", this.submit);
-               E.removeListener(this.cancelEl, "click", this.cancel);
-               E.removeListener(this.yearEl, "blur");
-               E.removeListener(this.monthEl, "change");
-               if (this.__isIEQuirks) {
-                       E.removeListener(this.cal.oDomContainer, "resize", this._syncMask);
-               }
-
-               this.purgeKeyListeners();
-       },
-
-       /**
-        * Attaches DOM listeners for keyboard support. 
-        * Tab/Shift-Tab looping, Enter Key Submit on Year element,
-        * Up/Down/PgUp/PgDown year increment on Year element
-        * <p>
-        * NOTE: MacOSX Safari 2.x doesn't let you tab to buttons and 
-        * MacOSX Gecko does not let you tab to buttons or select controls,
-        * so for these browsers, Tab/Shift-Tab looping is limited to the 
-        * elements which can be reached using the tab key.
-        * </p>
-        * @method applyKeyListeners
-        */
-       applyKeyListeners : function() {
-               var E = YAHOO.util.Event;
-
-               // IE doesn't fire keypress for arrow/pg keys (non-char keys)
-               var ua = YAHOO.env.ua;
-               var arrowEvt = (ua.ie) ? "keydown" : "keypress";
-
-               // - IE doesn't fire keypress for non-char keys
-               // - Opera doesn't allow us to cancel keydown or keypress for tab, but 
-               //   changes focus successfully on keydown (keypress is too late to change focus - opera's already moved on).
-               var tabEvt = (ua.ie || ua.opera) ? "keydown" : "keypress";
-
-               // Everyone likes keypress for Enter (char keys) - whoo hoo!
-               E.on(this.yearEl, "keypress", this._handleEnterKey, this, true);
-
-               E.on(this.yearEl, arrowEvt, this._handleDirectionKeys, this, true);
-               E.on(this.lastCtrl, tabEvt, this._handleTabKey, this, true);
-               E.on(this.firstCtrl, tabEvt, this._handleShiftTabKey, this, true);
-       },
-
-       /**
-        * Removes/purges DOM listeners for keyboard support
-        *
-        * @method purgeKeyListeners
-        */
-       purgeKeyListeners : function() {
-               var E = YAHOO.util.Event;
-
-               var arrowEvt = (YAHOO.env.ua.ie) ? "keydown" : "keypress";
-               var tabEvt = (YAHOO.env.ua.ie || YAHOO.env.ua.opera) ? "keydown" : "keypress";
-
-               E.removeListener(this.yearEl, "keypress", this._handleEnterKey);
-               E.removeListener(this.yearEl, arrowEvt, this._handleDirectionKeys);
-               E.removeListener(this.lastCtrl, tabEvt, this._handleTabKey);
-               E.removeListener(this.firstCtrl, tabEvt, this._handleShiftTabKey);
-       },
-
-       /**
-        * Updates the Calendar/CalendarGroup's pagedate with the currently set month and year if valid.
-        * <p>
-        * If the currently set month/year is invalid, a validation error will be displayed and the 
-        * Calendar/CalendarGroup's pagedate will not be updated.
-        * </p>
-        * @method submit
-        */
-       submit : function() {
-               if (this.validate()) {
-                       this.hide();
-
-                       this.setMonth(this._getMonthFromUI());
-                       this.setYear(this._getYearFromUI());
-
-                       var cal = this.cal;
-                       var nav = this;
-                       
-                       function update() {
-                               cal.setYear(nav.getYear());
-                               cal.setMonth(nav.getMonth());
-                               cal.render();
-                       }
-                       // Artificial delay, just to help the user see something changed
-                       var delay = YAHOO.widget.CalendarNavigator.UPDATE_DELAY;
-                       if (delay > 0) {
-                               window.setTimeout(update, delay);
-                       } else {
-                               update();
-                       }
-               }
-       },
-
-       /**
-        * Hides the navigator and mask, without updating the Calendar/CalendarGroup's state
-        * 
-        * @method cancel
-        */
-       cancel : function() {
-               this.hide();
-       },
-
-       /**
-        * Validates the current state of the UI controls
-        * 
-        * @method validate
-        * @return {Boolean} true, if the current UI state contains valid values, false if not
-        */
-       validate : function() {
-               if (this._getYearFromUI() !== null) {
-                       this.clearErrors();
-                       return true;
-               } else {
-                       this.setYearError();
-                       this.setError(this.__getCfg("invalidYear", true));
-                       return false;
-               }
-       },
-
-       /**
-        * Displays an error message in the Navigator's error panel
-        * @method setError
-        * @param {String} msg The error message to display
-        */
-       setError : function(msg) {
-               if (this.errorEl) {
-                       this.errorEl.innerHTML = msg;
-                       this._show(this.errorEl, true);
-               }
-       },
-
-       /**
-        * Clears the navigator's error message and hides the error panel
-        * @method clearError 
-        */
-       clearError : function() {
-               if (this.errorEl) {
-                       this.errorEl.innerHTML = "";
-                       this._show(this.errorEl, false);
-               }
-       },
-
-       /**
-        * Displays the validation error UI for the year control
-        * @method setYearError
-        */
-       setYearError : function() {
-               YAHOO.util.Dom.addClass(this.yearEl, YAHOO.widget.CalendarNavigator.CLASSES.INVALID);
-       },
-
-       /**
-        * Removes the validation error UI for the year control
-        * @method clearYearError
-        */
-       clearYearError : function() {
-               YAHOO.util.Dom.removeClass(this.yearEl, YAHOO.widget.CalendarNavigator.CLASSES.INVALID);
-       },
-
-       /**
-        * Clears all validation and error messages in the UI
-        * @method clearErrors
-        */
-       clearErrors : function() {
-               this.clearError();
-               this.clearYearError();
-       },
-
-       /**
-        * Sets the initial focus, based on the configured value
-        * @method setInitialFocus
-        */
-       setInitialFocus : function() {
-               var el = this.submitEl;
-               var f = this.__getCfg("initialFocus");
-
-               if (f && f.toLowerCase) {
-                       f = f.toLowerCase();
-                       if (f == "year") {
-                               el = this.yearEl;
-                               try {
-                                       this.yearEl.select();
-                               } catch (e) {
-                                       // Ignore;
-                               }
-                       } else if (f == "month") {
-                               el = this.monthEl;
-                       }
-               }
-
-               if (el && YAHOO.lang.isFunction(el.focus)) {
-                       try {
-                               el.focus();
-                       } catch (e) {
-                               // TODO: Fall back if focus fails?
-                       }
-               }
-       },
-
-       /**
-        * Removes all renderered HTML elements for the Navigator from
-        * the DOM, purges event listeners and clears (nulls) any property
-        * references to HTML references
-        * @method erase
-        */
-       erase : function() {
-               if (this.__rendered) {
-                       this.purgeListeners();
-
-                       // Clear out innerHTML references
-                       this.yearEl = null;
-                       this.monthEl = null;
-                       this.errorEl = null;
-                       this.submitEl = null;
-                       this.cancelEl = null;
-                       this.firstCtrl = null;
-                       this.lastCtrl = null;
-                       if (this.navEl) {
-                               this.navEl.innerHTML = "";
-                       }
-
-                       var p = this.navEl.parentNode;
-                       if (p) {
-                               p.removeChild(this.navEl);
-                       }
-                       this.navEl = null;
-
-                       var pm = this.maskEl.parentNode;
-                       if (pm) {
-                               pm.removeChild(this.maskEl);
-                       }
-                       this.maskEl = null;
-                       this.__rendered = false;
-               }
-       },
-
-       /**
-        * Destroys the Navigator object and any HTML references
-        * @method destroy
-        */
-       destroy : function() {
-               this.erase();
-               this._doc = null;
-               this.cal = null;
-               this.id = null;
-       },
-
-       /**
-        * Protected implementation to handle how UI elements are 
-        * hidden/shown.
-        *
-        * @method _show
-        * @protected
-        */
-       _show : function(el, bShow) {
-               if (el) {
-                       YAHOO.util.Dom.setStyle(el, "display", (bShow) ? "block" : "none");
-               }
-       },
-
-       /**
-        * Returns the month value (index), from the month UI element
-        * @protected
-        * @method _getMonthFromUI
-        * @return {Number} The month index, or 0 if a UI element for the month
-        * is not found
-        */
-       _getMonthFromUI : function() {
-               if (this.monthEl) {
-                       return this.monthEl.selectedIndex;
-               } else {
-                       return 0; // Default to Jan
-               }
-       },
-
-       /**
-        * Returns the year value, from the Navitator's year UI element
-        * @protected
-        * @method _getYearFromUI
-        * @return {Number} The year value set in the UI, if valid. null is returned if 
-        * the UI does not contain a valid year value.
-        */
-       _getYearFromUI : function() {
-               var NAV = YAHOO.widget.CalendarNavigator;
-
-               var yr = null;
-               if (this.yearEl) {
-                       var value = this.yearEl.value;
-                       value = value.replace(NAV.TRIM, "$1");
-
-                       if (NAV.YR_PATTERN.test(value)) {
-                               yr = parseInt(value, 10);
-                       }
-               }
-               return yr;
-       },
-
-       /**
-        * Updates the Navigator's year UI, based on the year value set on the Navigator object
-        * @protected
-        * @method _updateYearUI
-        */
-       _updateYearUI : function() {
-               if (this.yearEl && this._year !== null) {
-                       this.yearEl.value = this._year;
-               }
-       },
-
-       /**
-        * Updates the Navigator's month UI, based on the month value set on the Navigator object
-        * @protected
-        * @method _updateMonthUI
-        */
-       _updateMonthUI : function() {
-               if (this.monthEl) {
-                       this.monthEl.selectedIndex = this._month;
-               }
-       },
-
-       /**
-        * Sets up references to the first and last focusable element in the Navigator's UI
-        * in terms of tab order (Naviagator's firstEl and lastEl properties). The references
-        * are used to control modality by looping around from the first to the last control
-        * and visa versa for tab/shift-tab navigation.
-        * <p>
-        * See <a href="#applyKeyListeners">applyKeyListeners</a>
-        * </p>
-        * @protected
-        * @method _setFirstLastElements
-        */
-       _setFirstLastElements : function() {
-               this.firstCtrl = this.monthEl;
-               this.lastCtrl = this.cancelEl;
-
-               // Special handling for MacOSX.
-               // - Safari 2.x can't focus on buttons
-               // - Gecko can't focus on select boxes or buttons
-               if (this.__isMac) {
-                       if (YAHOO.env.ua.webkit && YAHOO.env.ua.webkit < 420){
-                               this.firstCtrl = this.monthEl;
-                               this.lastCtrl = this.yearEl;
-                       }
-                       if (YAHOO.env.ua.gecko) {
-                               this.firstCtrl = this.yearEl;
-                               this.lastCtrl = this.yearEl;
-                       }
-               }
-       },
-
-       /**
-        * Default Keyboard event handler to capture Enter 
-        * on the Navigator's year control (yearEl)
-        * 
-        * @method _handleEnterKey
-        * @protected
-        * @param {Event} e The DOM event being handled
-        */
-       _handleEnterKey : function(e) {
-               var KEYS = YAHOO.util.KeyListener.KEY;
-
-               if (YAHOO.util.Event.getCharCode(e) == KEYS.ENTER) {
-                       this.submit();
-               }
-       },
-
-       /**
-        * Default Keyboard event handler to capture up/down/pgup/pgdown
-        * on the Navigator's year control (yearEl).
-        * 
-        * @method _handleDirectionKeys
-        * @protected
-        * @param {Event} e The DOM event being handled
-        */
-       _handleDirectionKeys : function(e) {
-               var E = YAHOO.util.Event;
-               var KEYS = YAHOO.util.KeyListener.KEY;
-               var NAV = YAHOO.widget.CalendarNavigator;
-
-               var value = (this.yearEl.value) ? parseInt(this.yearEl.value, 10) : null;
-               if (isFinite(value)) {
-                       var dir = false;
-                       switch(E.getCharCode(e)) {
-                               case KEYS.UP:
-                                       this.yearEl.value = value + NAV.YR_MINOR_INC;
-                                       dir = true;
-                                       break;
-                               case KEYS.DOWN:
-                                       this.yearEl.value = Math.max(value - NAV.YR_MINOR_INC, 0);
-                                       dir = true;
-                                       break;
-                               case KEYS.PAGE_UP:
-                                       this.yearEl.value = value + NAV.YR_MAJOR_INC;
-                                       dir = true;
-                                       break;
-                               case KEYS.PAGE_DOWN:
-                                       this.yearEl.value = Math.max(value - NAV.YR_MAJOR_INC, 0);
-                                       dir = true;
-                                       break;
-                               default:
-                                       break;
-                       }
-                       if (dir) {
-                               E.preventDefault(e);
-                               try {
-                                       this.yearEl.select();
-                               } catch(e) {
-                                       // Ignore
-                               }
-                       }
-               }
-       },
-
-       /**
-        * Default Keyboard event handler to capture Tab 
-        * on the last control (lastCtrl) in the Navigator.
-        * 
-        * @method _handleTabKey
-        * @protected
-        * @param {Event} e The DOM event being handled
-        */
-       _handleTabKey : function(e) {
-               var E = YAHOO.util.Event;
-               var KEYS = YAHOO.util.KeyListener.KEY;
-
-               if (E.getCharCode(e) == KEYS.TAB && !e.shiftKey) {
-                       try {
-                               E.preventDefault(e);
-                               this.firstCtrl.focus();
-                       } catch (e) {
-                               // Ignore - mainly for focus edge cases
-                       }
-               }
-       },
-
-       /**
-        * Default Keyboard event handler to capture Shift-Tab 
-        * on the first control (firstCtrl) in the Navigator.
-        * 
-        * @method _handleShiftTabKey
-        * @protected
-        * @param {Event} e The DOM event being handled
-        */
-       _handleShiftTabKey : function(e) {
-               var E = YAHOO.util.Event;
-               var KEYS = YAHOO.util.KeyListener.KEY;
-
-               if (e.shiftKey && E.getCharCode(e) == KEYS.TAB) {
-                       try {
-                               E.preventDefault(e);
-                               this.lastCtrl.focus();
-                       } catch (e) {
-                               // Ignore - mainly for focus edge cases
-                       }
-               }
-       },
-
-       /**
-        * Retrieve Navigator configuration values from 
-        * the parent Calendar/CalendarGroup's config value.
-        * <p>
-        * If it has not been set in the user provided configuration, the method will 
-        * return the default value of the configuration property, as set in _DEFAULT_CFG
-        * </p>
-        * @private
-        * @method __getCfg
-        * @param {String} Case sensitive property name.
-        * @param {Boolean} true, if the property is a string property, false if not.
-        * @return The value of the configuration property
-        */
-       __getCfg : function(prop, bIsStr) {
-               var DEF_CFG = YAHOO.widget.CalendarNavigator._DEFAULT_CFG;
-               var cfg = this.cal.cfg.getProperty("navigator");
-
-               if (bIsStr) {
-                       return (cfg !== true && cfg.strings && cfg.strings[prop]) ? cfg.strings[prop] : DEF_CFG.strings[prop];
-               } else {
-                       return (cfg !== true && cfg[prop]) ? cfg[prop] : DEF_CFG[prop];
-               }
-       },
-
-       /**
-        * Private flag, to identify MacOS
-        * @private
-        * @property __isMac
-        */
-       __isMac : (navigator.userAgent.toLowerCase().indexOf("macintosh") != -1)
-
-};
-
-YAHOO.register("calendar", YAHOO.widget.Calendar, {version: "2.4.1", build: "742"});
diff --git a/sonar-server/src/main/webapp/javascripts/calendar/yahoo-dom-event-min.js b/sonar-server/src/main/webapp/javascripts/calendar/yahoo-dom-event-min.js
new file mode 100644 (file)
index 0000000..106a144
--- /dev/null
@@ -0,0 +1,10 @@
+/*
+Copyright (c) 2007, Yahoo! Inc. All rights reserved.
+Code licensed under the BSD License:
+http://developer.yahoo.net/yui/license.txt
+version: 2.4.1
+*/
+if(typeof YAHOO=="undefined"||!YAHOO){var YAHOO={};}YAHOO.namespace=function(){var A=arguments,E=null,C,B,D;for(C=0;C<A.length;C=C+1){D=A[C].split(".");E=YAHOO;for(B=(D[0]=="YAHOO")?1:0;B<D.length;B=B+1){E[D[B]]=E[D[B]]||{};E=E[D[B]];}}return E;};YAHOO.log=function(D,A,C){var B=YAHOO.widget.Logger;if(B&&B.log){return B.log(D,A,C);}else{return false;}};YAHOO.register=function(A,E,D){var I=YAHOO.env.modules;if(!I[A]){I[A]={versions:[],builds:[]};}var B=I[A],H=D.version,G=D.build,F=YAHOO.env.listeners;B.name=A;B.version=H;B.build=G;B.versions.push(H);B.builds.push(G);B.mainClass=E;for(var C=0;C<F.length;C=C+1){F[C](B);}if(E){E.VERSION=H;E.BUILD=G;}else{YAHOO.log("mainClass is undefined for module "+A,"warn");}};YAHOO.env=YAHOO.env||{modules:[],listeners:[]};YAHOO.env.getVersion=function(A){return YAHOO.env.modules[A]||null;};YAHOO.env.ua=function(){var C={ie:0,opera:0,gecko:0,webkit:0,mobile:null};var B=navigator.userAgent,A;if((/KHTML/).test(B)){C.webkit=1;}A=B.match(/AppleWebKit\/([^\s]*)/);if(A&&A[1]){C.webkit=parseFloat(A[1]);if(/ Mobile\//.test(B)){C.mobile="Apple";}else{A=B.match(/NokiaN[^\/]*/);if(A){C.mobile=A[0];}}}if(!C.webkit){A=B.match(/Opera[\s\/]([^\s]*)/);if(A&&A[1]){C.opera=parseFloat(A[1]);A=B.match(/Opera Mini[^;]*/);if(A){C.mobile=A[0];}}else{A=B.match(/MSIE\s([^;]*)/);if(A&&A[1]){C.ie=parseFloat(A[1]);}else{A=B.match(/Gecko\/([^\s]*)/);if(A){C.gecko=1;A=B.match(/rv:([^\s\)]*)/);if(A&&A[1]){C.gecko=parseFloat(A[1]);}}}}}return C;}();(function(){YAHOO.namespace("util","widget","example");if("undefined"!==typeof YAHOO_config){var B=YAHOO_config.listener,A=YAHOO.env.listeners,D=true,C;if(B){for(C=0;C<A.length;C=C+1){if(A[C]==B){D=false;break;}}if(D){A.push(B);}}}})();YAHOO.lang=YAHOO.lang||{isArray:function(B){if(B){var A=YAHOO.lang;return A.isNumber(B.length)&&A.isFunction(B.splice);}return false;},isBoolean:function(A){return typeof A==="boolean";},isFunction:function(A){return typeof A==="function";},isNull:function(A){return A===null;},isNumber:function(A){return typeof A==="number"&&isFinite(A);},isObject:function(A){return(A&&(typeof A==="object"||YAHOO.lang.isFunction(A)))||false;},isString:function(A){return typeof A==="string";},isUndefined:function(A){return typeof A==="undefined";},hasOwnProperty:function(A,B){if(Object.prototype.hasOwnProperty){return A.hasOwnProperty(B);}return !YAHOO.lang.isUndefined(A[B])&&A.constructor.prototype[B]!==A[B];},_IEEnumFix:function(C,B){if(YAHOO.env.ua.ie){var E=["toString","valueOf"],A;for(A=0;A<E.length;A=A+1){var F=E[A],D=B[F];if(YAHOO.lang.isFunction(D)&&D!=Object.prototype[F]){C[F]=D;}}}},extend:function(D,E,C){if(!E||!D){throw new Error("YAHOO.lang.extend failed, please check that all dependencies are included.");}var B=function(){};B.prototype=E.prototype;D.prototype=new B();D.prototype.constructor=D;D.superclass=E.prototype;if(E.prototype.constructor==Object.prototype.constructor){E.prototype.constructor=E;}if(C){for(var A in C){D.prototype[A]=C[A];}YAHOO.lang._IEEnumFix(D.prototype,C);}},augmentObject:function(E,D){if(!D||!E){throw new Error("Absorb failed, verify dependencies.");}var A=arguments,C,F,B=A[2];if(B&&B!==true){for(C=2;C<A.length;C=C+1){E[A[C]]=D[A[C]];}}else{for(F in D){if(B||!E[F]){E[F]=D[F];}}YAHOO.lang._IEEnumFix(E,D);}},augmentProto:function(D,C){if(!C||!D){throw new Error("Augment failed, verify dependencies.");}var A=[D.prototype,C.prototype];for(var B=2;B<arguments.length;B=B+1){A.push(arguments[B]);}YAHOO.lang.augmentObject.apply(this,A);},dump:function(A,G){var C=YAHOO.lang,D,F,I=[],J="{...}",B="f(){...}",H=", ",E=" => ";if(!C.isObject(A)){return A+"";}else{if(A instanceof Date||("nodeType" in A&&"tagName" in A)){return A;}else{if(C.isFunction(A)){return B;}}}G=(C.isNumber(G))?G:3;if(C.isArray(A)){I.push("[");for(D=0,F=A.length;D<F;D=D+1){if(C.isObject(A[D])){I.push((G>0)?C.dump(A[D],G-1):J);}else{I.push(A[D]);}I.push(H);}if(I.length>1){I.pop();}I.push("]");}else{I.push("{");for(D in A){if(C.hasOwnProperty(A,D)){I.push(D+E);if(C.isObject(A[D])){I.push((G>0)?C.dump(A[D],G-1):J);}else{I.push(A[D]);}I.push(H);}}if(I.length>1){I.pop();}I.push("}");}return I.join("");},substitute:function(Q,B,J){var G,F,E,M,N,P,D=YAHOO.lang,L=[],C,H="dump",K=" ",A="{",O="}";for(;;){G=Q.lastIndexOf(A);if(G<0){break;}F=Q.indexOf(O,G);if(G+1>=F){break;}C=Q.substring(G+1,F);M=C;P=null;E=M.indexOf(K);if(E>-1){P=M.substring(E+1);M=M.substring(0,E);}N=B[M];if(J){N=J(M,N,P);}if(D.isObject(N)){if(D.isArray(N)){N=D.dump(N,parseInt(P,10));}else{P=P||"";var I=P.indexOf(H);if(I>-1){P=P.substring(4);}if(N.toString===Object.prototype.toString||I>-1){N=D.dump(N,parseInt(P,10));}else{N=N.toString();}}}else{if(!D.isString(N)&&!D.isNumber(N)){N="~-"+L.length+"-~";L[L.length]=C;}}Q=Q.substring(0,G)+N+Q.substring(F+1);}for(G=L.length-1;G>=0;G=G-1){Q=Q.replace(new RegExp("~-"+G+"-~"),"{"+L[G]+"}","g");}return Q;},trim:function(A){try{return A.replace(/^\s+|\s+$/g,"");}catch(B){return A;}},merge:function(){var D={},B=arguments;for(var C=0,A=B.length;C<A;C=C+1){YAHOO.lang.augmentObject(D,B[C],true);}return D;},later:function(H,B,I,D,E){H=H||0;B=B||{};var C=I,G=D,F,A;if(YAHOO.lang.isString(I)){C=B[I];}if(!C){throw new TypeError("method undefined");}if(!YAHOO.lang.isArray(G)){G=[D];}F=function(){C.apply(B,G);};A=(E)?setInterval(F,H):setTimeout(F,H);return{interval:E,cancel:function(){if(this.interval){clearInterval(A);}else{clearTimeout(A);}}};},isValue:function(B){var A=YAHOO.lang;return(A.isObject(B)||A.isString(B)||A.isNumber(B)||A.isBoolean(B));}};YAHOO.util.Lang=YAHOO.lang;YAHOO.lang.augment=YAHOO.lang.augmentProto;YAHOO.augment=YAHOO.lang.augmentProto;YAHOO.extend=YAHOO.lang.extend;YAHOO.register("yahoo",YAHOO,{version:"2.4.1",build:"742"});(function(){var B=YAHOO.util,L,J,H=0,K={},F={},N=window.document;var C=YAHOO.env.ua.opera,M=YAHOO.env.ua.webkit,A=YAHOO.env.ua.gecko,G=YAHOO.env.ua.ie;var E={HYPHEN:/(-[a-z])/i,ROOT_TAG:/^body|html$/i};var O=function(Q){if(!E.HYPHEN.test(Q)){return Q;}if(K[Q]){return K[Q];}var R=Q;while(E.HYPHEN.exec(R)){R=R.replace(RegExp.$1,RegExp.$1.substr(1).toUpperCase());}K[Q]=R;return R;};var P=function(R){var Q=F[R];if(!Q){Q=new RegExp("(?:^|\\s+)"+R+"(?:\\s+|$)");F[R]=Q;}return Q;};if(N.defaultView&&N.defaultView.getComputedStyle){L=function(Q,T){var S=null;if(T=="float"){T="cssFloat";}var R=N.defaultView.getComputedStyle(Q,"");if(R){S=R[O(T)];}return Q.style[T]||S;};}else{if(N.documentElement.currentStyle&&G){L=function(Q,S){switch(O(S)){case"opacity":var U=100;try{U=Q.filters["DXImageTransform.Microsoft.Alpha"].opacity;}catch(T){try{U=Q.filters("alpha").opacity;}catch(T){}}return U/100;case"float":S="styleFloat";default:var R=Q.currentStyle?Q.currentStyle[S]:null;return(Q.style[S]||R);}};}else{L=function(Q,R){return Q.style[R];};}}if(G){J=function(Q,R,S){switch(R){case"opacity":if(YAHOO.lang.isString(Q.style.filter)){Q.style.filter="alpha(opacity="+S*100+")";if(!Q.currentStyle||!Q.currentStyle.hasLayout){Q.style.zoom=1;}}break;case"float":R="styleFloat";default:Q.style[R]=S;}};}else{J=function(Q,R,S){if(R=="float"){R="cssFloat";}Q.style[R]=S;};}var D=function(Q,R){return Q&&Q.nodeType==1&&(!R||R(Q));};YAHOO.util.Dom={get:function(S){if(S&&(S.tagName||S.item)){return S;}if(YAHOO.lang.isString(S)||!S){return N.getElementById(S);}if(S.length!==undefined){var T=[];for(var R=0,Q=S.length;R<Q;++R){T[T.length]=B.Dom.get(S[R]);}return T;}return S;},getStyle:function(Q,S){S=O(S);var R=function(T){return L(T,S);};return B.Dom.batch(Q,R,B.Dom,true);},setStyle:function(Q,S,T){S=O(S);var R=function(U){J(U,S,T);};B.Dom.batch(Q,R,B.Dom,true);},getXY:function(Q){var R=function(S){if((S.parentNode===null||S.offsetParent===null||this.getStyle(S,"display")=="none")&&S!=S.ownerDocument.body){return false;}return I(S);};return B.Dom.batch(Q,R,B.Dom,true);},getX:function(Q){var R=function(S){return B.Dom.getXY(S)[0];};return B.Dom.batch(Q,R,B.Dom,true);},getY:function(Q){var R=function(S){return B.Dom.getXY(S)[1];};return B.Dom.batch(Q,R,B.Dom,true);},setXY:function(Q,T,S){var R=function(W){var V=this.getStyle(W,"position");if(V=="static"){this.setStyle(W,"position","relative");V="relative";}var Y=this.getXY(W);if(Y===false){return false;}var X=[parseInt(this.getStyle(W,"left"),10),parseInt(this.getStyle(W,"top"),10)];if(isNaN(X[0])){X[0]=(V=="relative")?0:W.offsetLeft;}if(isNaN(X[1])){X[1]=(V=="relative")?0:W.offsetTop;}if(T[0]!==null){W.style.left=T[0]-Y[0]+X[0]+"px";}if(T[1]!==null){W.style.top=T[1]-Y[1]+X[1]+"px";}if(!S){var U=this.getXY(W);if((T[0]!==null&&U[0]!=T[0])||(T[1]!==null&&U[1]!=T[1])){this.setXY(W,T,true);}}};B.Dom.batch(Q,R,B.Dom,true);},setX:function(R,Q){B.Dom.setXY(R,[Q,null]);},setY:function(Q,R){B.Dom.setXY(Q,[null,R]);},getRegion:function(Q){var R=function(S){if((S.parentNode===null||S.offsetParent===null||this.getStyle(S,"display")=="none")&&S!=N.body){return false;}var T=B.Region.getRegion(S);return T;};return B.Dom.batch(Q,R,B.Dom,true);},getClientWidth:function(){return B.Dom.getViewportWidth();},getClientHeight:function(){return B.Dom.getViewportHeight();},getElementsByClassName:function(U,Y,V,W){Y=Y||"*";V=(V)?B.Dom.get(V):null||N;if(!V){return[];}var R=[],Q=V.getElementsByTagName(Y),X=P(U);for(var S=0,T=Q.length;S<T;++S){if(X.test(Q[S].className)){R[R.length]=Q[S];if(W){W.call(Q[S],Q[S]);}}}return R;},hasClass:function(S,R){var Q=P(R);var T=function(U){return Q.test(U.className);};return B.Dom.batch(S,T,B.Dom,true);},addClass:function(R,Q){var S=function(T){if(this.hasClass(T,Q)){return false;}T.className=YAHOO.lang.trim([T.className,Q].join(" "));return true;};return B.Dom.batch(R,S,B.Dom,true);},removeClass:function(S,R){var Q=P(R);var T=function(U){if(!this.hasClass(U,R)){return false;}var V=U.className;U.className=V.replace(Q," ");if(this.hasClass(U,R)){this.removeClass(U,R);}U.className=YAHOO.lang.trim(U.className);return true;};return B.Dom.batch(S,T,B.Dom,true);},replaceClass:function(T,R,Q){if(!Q||R===Q){return false;}var S=P(R);var U=function(V){if(!this.hasClass(V,R)){this.addClass(V,Q);return true;}V.className=V.className.replace(S," "+Q+" ");if(this.hasClass(V,R)){this.replaceClass(V,R,Q);}V.className=YAHOO.lang.trim(V.className);return true;};return B.Dom.batch(T,U,B.Dom,true);},generateId:function(Q,S){S=S||"yui-gen";var R=function(T){if(T&&T.id){return T.id;}var U=S+H++;if(T){T.id=U;}return U;};return B.Dom.batch(Q,R,B.Dom,true)||R.apply(B.Dom,arguments);},isAncestor:function(Q,R){Q=B.Dom.get(Q);R=B.Dom.get(R);if(!Q||!R){return false;}if(Q.contains&&R.nodeType&&!M){return Q.contains(R);}else{if(Q.compareDocumentPosition&&R.nodeType){return !!(Q.compareDocumentPosition(R)&16);}else{if(R.nodeType){return !!this.getAncestorBy(R,function(S){return S==Q;});}}}return false;},inDocument:function(Q){return this.isAncestor(N.documentElement,Q);},getElementsBy:function(X,R,S,U){R=R||"*";S=(S)?B.Dom.get(S):null||N;if(!S){return[];}var T=[],W=S.getElementsByTagName(R);for(var V=0,Q=W.length;V<Q;++V){if(X(W[V])){T[T.length]=W[V];if(U){U(W[V]);}}}return T;},batch:function(U,X,W,S){U=(U&&(U.tagName||U.item))?U:B.Dom.get(U);if(!U||!X){return false;}var T=(S)?W:window;if(U.tagName||U.length===undefined){return X.call(T,U,W);}var V=[];for(var R=0,Q=U.length;R<Q;++R){V[V.length]=X.call(T,U[R],W);}return V;},getDocumentHeight:function(){var R=(N.compatMode!="CSS1Compat")?N.body.scrollHeight:N.documentElement.scrollHeight;var Q=Math.max(R,B.Dom.getViewportHeight());return Q;},getDocumentWidth:function(){var R=(N.compatMode!="CSS1Compat")?N.body.scrollWidth:N.documentElement.scrollWidth;var Q=Math.max(R,B.Dom.getViewportWidth());return Q;},getViewportHeight:function(){var Q=self.innerHeight;var R=N.compatMode;if((R||G)&&!C){Q=(R=="CSS1Compat")?N.documentElement.clientHeight:N.body.clientHeight;
+}return Q;},getViewportWidth:function(){var Q=self.innerWidth;var R=N.compatMode;if(R||G){Q=(R=="CSS1Compat")?N.documentElement.clientWidth:N.body.clientWidth;}return Q;},getAncestorBy:function(Q,R){while(Q=Q.parentNode){if(D(Q,R)){return Q;}}return null;},getAncestorByClassName:function(R,Q){R=B.Dom.get(R);if(!R){return null;}var S=function(T){return B.Dom.hasClass(T,Q);};return B.Dom.getAncestorBy(R,S);},getAncestorByTagName:function(R,Q){R=B.Dom.get(R);if(!R){return null;}var S=function(T){return T.tagName&&T.tagName.toUpperCase()==Q.toUpperCase();};return B.Dom.getAncestorBy(R,S);},getPreviousSiblingBy:function(Q,R){while(Q){Q=Q.previousSibling;if(D(Q,R)){return Q;}}return null;},getPreviousSibling:function(Q){Q=B.Dom.get(Q);if(!Q){return null;}return B.Dom.getPreviousSiblingBy(Q);},getNextSiblingBy:function(Q,R){while(Q){Q=Q.nextSibling;if(D(Q,R)){return Q;}}return null;},getNextSibling:function(Q){Q=B.Dom.get(Q);if(!Q){return null;}return B.Dom.getNextSiblingBy(Q);},getFirstChildBy:function(Q,S){var R=(D(Q.firstChild,S))?Q.firstChild:null;return R||B.Dom.getNextSiblingBy(Q.firstChild,S);},getFirstChild:function(Q,R){Q=B.Dom.get(Q);if(!Q){return null;}return B.Dom.getFirstChildBy(Q);},getLastChildBy:function(Q,S){if(!Q){return null;}var R=(D(Q.lastChild,S))?Q.lastChild:null;return R||B.Dom.getPreviousSiblingBy(Q.lastChild,S);},getLastChild:function(Q){Q=B.Dom.get(Q);return B.Dom.getLastChildBy(Q);},getChildrenBy:function(R,T){var S=B.Dom.getFirstChildBy(R,T);var Q=S?[S]:[];B.Dom.getNextSiblingBy(S,function(U){if(!T||T(U)){Q[Q.length]=U;}return false;});return Q;},getChildren:function(Q){Q=B.Dom.get(Q);if(!Q){}return B.Dom.getChildrenBy(Q);},getDocumentScrollLeft:function(Q){Q=Q||N;return Math.max(Q.documentElement.scrollLeft,Q.body.scrollLeft);},getDocumentScrollTop:function(Q){Q=Q||N;return Math.max(Q.documentElement.scrollTop,Q.body.scrollTop);},insertBefore:function(R,Q){R=B.Dom.get(R);Q=B.Dom.get(Q);if(!R||!Q||!Q.parentNode){return null;}return Q.parentNode.insertBefore(R,Q);},insertAfter:function(R,Q){R=B.Dom.get(R);Q=B.Dom.get(Q);if(!R||!Q||!Q.parentNode){return null;}if(Q.nextSibling){return Q.parentNode.insertBefore(R,Q.nextSibling);}else{return Q.parentNode.appendChild(R);}},getClientRegion:function(){var S=B.Dom.getDocumentScrollTop(),R=B.Dom.getDocumentScrollLeft(),T=B.Dom.getViewportWidth()+R,Q=B.Dom.getViewportHeight()+S;return new B.Region(S,T,Q,R);}};var I=function(){if(N.documentElement.getBoundingClientRect){return function(R){var S=R.getBoundingClientRect();var Q=R.ownerDocument;return[S.left+B.Dom.getDocumentScrollLeft(Q),S.top+B.Dom.getDocumentScrollTop(Q)];};}else{return function(S){var T=[S.offsetLeft,S.offsetTop];var R=S.offsetParent;var Q=(M&&B.Dom.getStyle(S,"position")=="absolute"&&S.offsetParent==S.ownerDocument.body);if(R!=S){while(R){T[0]+=R.offsetLeft;T[1]+=R.offsetTop;if(!Q&&M&&B.Dom.getStyle(R,"position")=="absolute"){Q=true;}R=R.offsetParent;}}if(Q){T[0]-=S.ownerDocument.body.offsetLeft;T[1]-=S.ownerDocument.body.offsetTop;}R=S.parentNode;while(R.tagName&&!E.ROOT_TAG.test(R.tagName)){if(B.Dom.getStyle(R,"display").search(/^inline|table-row.*$/i)){T[0]-=R.scrollLeft;T[1]-=R.scrollTop;}R=R.parentNode;}return T;};}}();})();YAHOO.util.Region=function(C,D,A,B){this.top=C;this[1]=C;this.right=D;this.bottom=A;this.left=B;this[0]=B;};YAHOO.util.Region.prototype.contains=function(A){return(A.left>=this.left&&A.right<=this.right&&A.top>=this.top&&A.bottom<=this.bottom);};YAHOO.util.Region.prototype.getArea=function(){return((this.bottom-this.top)*(this.right-this.left));};YAHOO.util.Region.prototype.intersect=function(E){var C=Math.max(this.top,E.top);var D=Math.min(this.right,E.right);var A=Math.min(this.bottom,E.bottom);var B=Math.max(this.left,E.left);if(A>=C&&D>=B){return new YAHOO.util.Region(C,D,A,B);}else{return null;}};YAHOO.util.Region.prototype.union=function(E){var C=Math.min(this.top,E.top);var D=Math.max(this.right,E.right);var A=Math.max(this.bottom,E.bottom);var B=Math.min(this.left,E.left);return new YAHOO.util.Region(C,D,A,B);};YAHOO.util.Region.prototype.toString=function(){return("Region {top: "+this.top+", right: "+this.right+", bottom: "+this.bottom+", left: "+this.left+"}");};YAHOO.util.Region.getRegion=function(D){var F=YAHOO.util.Dom.getXY(D);var C=F[1];var E=F[0]+D.offsetWidth;var A=F[1]+D.offsetHeight;var B=F[0];return new YAHOO.util.Region(C,E,A,B);};YAHOO.util.Point=function(A,B){if(YAHOO.lang.isArray(A)){B=A[1];A=A[0];}this.x=this.right=this.left=this[0]=A;this.y=this.top=this.bottom=this[1]=B;};YAHOO.util.Point.prototype=new YAHOO.util.Region();YAHOO.register("dom",YAHOO.util.Dom,{version:"2.4.1",build:"742"});YAHOO.util.CustomEvent=function(D,B,C,A){this.type=D;this.scope=B||window;this.silent=C;this.signature=A||YAHOO.util.CustomEvent.LIST;this.subscribers=[];if(!this.silent){}var E="_YUICEOnSubscribe";if(D!==E){this.subscribeEvent=new YAHOO.util.CustomEvent(E,this,true);}this.lastError=null;};YAHOO.util.CustomEvent.LIST=0;YAHOO.util.CustomEvent.FLAT=1;YAHOO.util.CustomEvent.prototype={subscribe:function(B,C,A){if(!B){throw new Error("Invalid callback for subscriber to '"+this.type+"'");}if(this.subscribeEvent){this.subscribeEvent.fire(B,C,A);}this.subscribers.push(new YAHOO.util.Subscriber(B,C,A));},unsubscribe:function(D,F){if(!D){return this.unsubscribeAll();}var E=false;for(var B=0,A=this.subscribers.length;B<A;++B){var C=this.subscribers[B];if(C&&C.contains(D,F)){this._delete(B);E=true;}}return E;},fire:function(){var D=this.subscribers.length;if(!D&&this.silent){return true;}var H=[],F=true,C,I=false;for(C=0;C<arguments.length;++C){H.push(arguments[C]);}if(!this.silent){}for(C=0;C<D;++C){var L=this.subscribers[C];if(!L){I=true;}else{if(!this.silent){}var K=L.getScope(this.scope);if(this.signature==YAHOO.util.CustomEvent.FLAT){var A=null;if(H.length>0){A=H[0];}try{F=L.fn.call(K,A,L.obj);}catch(E){this.lastError=E;}}else{try{F=L.fn.call(K,this.type,H,L.obj);}catch(G){this.lastError=G;}}if(false===F){if(!this.silent){}return false;}}}if(I){var J=[],B=this.subscribers;for(C=0,D=B.length;C<D;C=C+1){J.push(B[C]);}this.subscribers=J;}return true;},unsubscribeAll:function(){for(var B=0,A=this.subscribers.length;B<A;++B){this._delete(A-1-B);}this.subscribers=[];return B;},_delete:function(A){var B=this.subscribers[A];if(B){delete B.fn;delete B.obj;}this.subscribers[A]=null;},toString:function(){return"CustomEvent: '"+this.type+"', scope: "+this.scope;}};YAHOO.util.Subscriber=function(B,C,A){this.fn=B;this.obj=YAHOO.lang.isUndefined(C)?null:C;this.override=A;};YAHOO.util.Subscriber.prototype.getScope=function(A){if(this.override){if(this.override===true){return this.obj;}else{return this.override;}}return A;};YAHOO.util.Subscriber.prototype.contains=function(A,B){if(B){return(this.fn==A&&this.obj==B);}else{return(this.fn==A);}};YAHOO.util.Subscriber.prototype.toString=function(){return"Subscriber { obj: "+this.obj+", override: "+(this.override||"no")+" }";};if(!YAHOO.util.Event){YAHOO.util.Event=function(){var H=false;var I=[];var J=[];var G=[];var E=[];var C=0;var F=[];var B=[];var A=0;var D={63232:38,63233:40,63234:37,63235:39,63276:33,63277:34,25:9};return{POLL_RETRYS:4000,POLL_INTERVAL:10,EL:0,TYPE:1,FN:2,WFN:3,UNLOAD_OBJ:3,ADJ_SCOPE:4,OBJ:5,OVERRIDE:6,lastError:null,isSafari:YAHOO.env.ua.webkit,webkit:YAHOO.env.ua.webkit,isIE:YAHOO.env.ua.ie,_interval:null,_dri:null,DOMReady:false,startInterval:function(){if(!this._interval){var K=this;var L=function(){K._tryPreloadAttach();};this._interval=setInterval(L,this.POLL_INTERVAL);}},onAvailable:function(P,M,Q,O,N){var K=(YAHOO.lang.isString(P))?[P]:P;for(var L=0;L<K.length;L=L+1){F.push({id:K[L],fn:M,obj:Q,override:O,checkReady:N});}C=this.POLL_RETRYS;this.startInterval();},onContentReady:function(M,K,N,L){this.onAvailable(M,K,N,L,true);},onDOMReady:function(K,M,L){if(this.DOMReady){setTimeout(function(){var N=window;if(L){if(L===true){N=M;}else{N=L;}}K.call(N,"DOMReady",[],M);},0);}else{this.DOMReadyEvent.subscribe(K,M,L);}},addListener:function(M,K,V,Q,L){if(!V||!V.call){return false;}if(this._isValidCollection(M)){var W=true;for(var R=0,T=M.length;R<T;++R){W=this.on(M[R],K,V,Q,L)&&W;}return W;}else{if(YAHOO.lang.isString(M)){var P=this.getEl(M);if(P){M=P;}else{this.onAvailable(M,function(){YAHOO.util.Event.on(M,K,V,Q,L);});return true;}}}if(!M){return false;}if("unload"==K&&Q!==this){J[J.length]=[M,K,V,Q,L];return true;}var Y=M;if(L){if(L===true){Y=Q;}else{Y=L;}}var N=function(Z){return V.call(Y,YAHOO.util.Event.getEvent(Z,M),Q);};var X=[M,K,V,N,Y,Q,L];var S=I.length;I[S]=X;if(this.useLegacyEvent(M,K)){var O=this.getLegacyIndex(M,K);if(O==-1||M!=G[O][0]){O=G.length;B[M.id+K]=O;G[O]=[M,K,M["on"+K]];E[O]=[];M["on"+K]=function(Z){YAHOO.util.Event.fireLegacyEvent(YAHOO.util.Event.getEvent(Z),O);};}E[O].push(X);}else{try{this._simpleAdd(M,K,N,false);}catch(U){this.lastError=U;this.removeListener(M,K,V);return false;}}return true;},fireLegacyEvent:function(O,M){var Q=true,K,S,R,T,P;S=E[M];for(var L=0,N=S.length;L<N;++L){R=S[L];if(R&&R[this.WFN]){T=R[this.ADJ_SCOPE];P=R[this.WFN].call(T,O);Q=(Q&&P);}}K=G[M];if(K&&K[2]){K[2](O);}return Q;},getLegacyIndex:function(L,M){var K=this.generateId(L)+M;if(typeof B[K]=="undefined"){return -1;}else{return B[K];}},useLegacyEvent:function(L,M){if(this.webkit&&("click"==M||"dblclick"==M)){var K=parseInt(this.webkit,10);if(!isNaN(K)&&K<418){return true;}}return false;},removeListener:function(L,K,T){var O,R,V;if(typeof L=="string"){L=this.getEl(L);}else{if(this._isValidCollection(L)){var U=true;for(O=0,R=L.length;O<R;++O){U=(this.removeListener(L[O],K,T)&&U);}return U;}}if(!T||!T.call){return this.purgeElement(L,false,K);}if("unload"==K){for(O=0,R=J.length;O<R;O++){V=J[O];if(V&&V[0]==L&&V[1]==K&&V[2]==T){J[O]=null;return true;}}return false;}var P=null;var Q=arguments[3];if("undefined"===typeof Q){Q=this._getCacheIndex(L,K,T);}if(Q>=0){P=I[Q];}if(!L||!P){return false;}if(this.useLegacyEvent(L,K)){var N=this.getLegacyIndex(L,K);var M=E[N];if(M){for(O=0,R=M.length;O<R;++O){V=M[O];if(V&&V[this.EL]==L&&V[this.TYPE]==K&&V[this.FN]==T){M[O]=null;break;}}}}else{try{this._simpleRemove(L,K,P[this.WFN],false);}catch(S){this.lastError=S;return false;}}delete I[Q][this.WFN];delete I[Q][this.FN];I[Q]=null;return true;},getTarget:function(M,L){var K=M.target||M.srcElement;return this.resolveTextNode(K);},resolveTextNode:function(K){if(K&&3==K.nodeType){return K.parentNode;}else{return K;}},getPageX:function(L){var K=L.pageX;if(!K&&0!==K){K=L.clientX||0;if(this.isIE){K+=this._getScrollLeft();}}return K;},getPageY:function(K){var L=K.pageY;if(!L&&0!==L){L=K.clientY||0;if(this.isIE){L+=this._getScrollTop();}}return L;},getXY:function(K){return[this.getPageX(K),this.getPageY(K)];
+},getRelatedTarget:function(L){var K=L.relatedTarget;if(!K){if(L.type=="mouseout"){K=L.toElement;}else{if(L.type=="mouseover"){K=L.fromElement;}}}return this.resolveTextNode(K);},getTime:function(M){if(!M.time){var L=new Date().getTime();try{M.time=L;}catch(K){this.lastError=K;return L;}}return M.time;},stopEvent:function(K){this.stopPropagation(K);this.preventDefault(K);},stopPropagation:function(K){if(K.stopPropagation){K.stopPropagation();}else{K.cancelBubble=true;}},preventDefault:function(K){if(K.preventDefault){K.preventDefault();}else{K.returnValue=false;}},getEvent:function(M,K){var L=M||window.event;if(!L){var N=this.getEvent.caller;while(N){L=N.arguments[0];if(L&&Event==L.constructor){break;}N=N.caller;}}return L;},getCharCode:function(L){var K=L.keyCode||L.charCode||0;if(YAHOO.env.ua.webkit&&(K in D)){K=D[K];}return K;},_getCacheIndex:function(O,P,N){for(var M=0,L=I.length;M<L;++M){var K=I[M];if(K&&K[this.FN]==N&&K[this.EL]==O&&K[this.TYPE]==P){return M;}}return -1;},generateId:function(K){var L=K.id;if(!L){L="yuievtautoid-"+A;++A;K.id=L;}return L;},_isValidCollection:function(L){try{return(L&&typeof L!=="string"&&L.length&&!L.tagName&&!L.alert&&typeof L[0]!=="undefined");}catch(K){return false;}},elCache:{},getEl:function(K){return(typeof K==="string")?document.getElementById(K):K;},clearCache:function(){},DOMReadyEvent:new YAHOO.util.CustomEvent("DOMReady",this),_load:function(L){if(!H){H=true;var K=YAHOO.util.Event;K._ready();K._tryPreloadAttach();}},_ready:function(L){var K=YAHOO.util.Event;if(!K.DOMReady){K.DOMReady=true;K.DOMReadyEvent.fire();K._simpleRemove(document,"DOMContentLoaded",K._ready);}},_tryPreloadAttach:function(){if(this.locked){return false;}if(this.isIE){if(!this.DOMReady){this.startInterval();return false;}}this.locked=true;var P=!H;if(!P){P=(C>0);}var O=[];var Q=function(S,T){var R=S;if(T.override){if(T.override===true){R=T.obj;}else{R=T.override;}}T.fn.call(R,T.obj);};var L,K,N,M;for(L=0,K=F.length;L<K;++L){N=F[L];if(N&&!N.checkReady){M=this.getEl(N.id);if(M){Q(M,N);F[L]=null;}else{O.push(N);}}}for(L=0,K=F.length;L<K;++L){N=F[L];if(N&&N.checkReady){M=this.getEl(N.id);if(M){if(H||M.nextSibling){Q(M,N);F[L]=null;}}else{O.push(N);}}}C=(O.length===0)?0:C-1;if(P){this.startInterval();}else{clearInterval(this._interval);this._interval=null;}this.locked=false;return true;},purgeElement:function(O,P,R){var M=(YAHOO.lang.isString(O))?this.getEl(O):O;var Q=this.getListeners(M,R),N,K;if(Q){for(N=0,K=Q.length;N<K;++N){var L=Q[N];this.removeListener(M,L.type,L.fn,L.index);}}if(P&&M&&M.childNodes){for(N=0,K=M.childNodes.length;N<K;++N){this.purgeElement(M.childNodes[N],P,R);}}},getListeners:function(M,K){var P=[],L;if(!K){L=[I,J];}else{if(K==="unload"){L=[J];}else{L=[I];}}var R=(YAHOO.lang.isString(M))?this.getEl(M):M;for(var O=0;O<L.length;O=O+1){var T=L[O];if(T&&T.length>0){for(var Q=0,S=T.length;Q<S;++Q){var N=T[Q];if(N&&N[this.EL]===R&&(!K||K===N[this.TYPE])){P.push({type:N[this.TYPE],fn:N[this.FN],obj:N[this.OBJ],adjust:N[this.OVERRIDE],scope:N[this.ADJ_SCOPE],index:Q});}}}}return(P.length)?P:null;},_unload:function(R){var Q=YAHOO.util.Event,O,N,L,K,M;for(O=0,K=J.length;O<K;++O){L=J[O];if(L){var P=window;if(L[Q.ADJ_SCOPE]){if(L[Q.ADJ_SCOPE]===true){P=L[Q.UNLOAD_OBJ];}else{P=L[Q.ADJ_SCOPE];}}L[Q.FN].call(P,Q.getEvent(R,L[Q.EL]),L[Q.UNLOAD_OBJ]);J[O]=null;L=null;P=null;}}J=null;if(YAHOO.env.ua.ie&&I&&I.length>0){N=I.length;while(N){M=N-1;L=I[M];if(L){Q.removeListener(L[Q.EL],L[Q.TYPE],L[Q.FN],M);}N--;}L=null;}G=null;Q._simpleRemove(window,"unload",Q._unload);},_getScrollLeft:function(){return this._getScroll()[1];},_getScrollTop:function(){return this._getScroll()[0];},_getScroll:function(){var K=document.documentElement,L=document.body;if(K&&(K.scrollTop||K.scrollLeft)){return[K.scrollTop,K.scrollLeft];}else{if(L){return[L.scrollTop,L.scrollLeft];}else{return[0,0];}}},regCE:function(){},_simpleAdd:function(){if(window.addEventListener){return function(M,N,L,K){M.addEventListener(N,L,(K));};}else{if(window.attachEvent){return function(M,N,L,K){M.attachEvent("on"+N,L);};}else{return function(){};}}}(),_simpleRemove:function(){if(window.removeEventListener){return function(M,N,L,K){M.removeEventListener(N,L,(K));};}else{if(window.detachEvent){return function(L,M,K){L.detachEvent("on"+M,K);};}else{return function(){};}}}()};}();(function(){var A=YAHOO.util.Event;A.on=A.addListener;if(A.isIE){YAHOO.util.Event.onDOMReady(YAHOO.util.Event._tryPreloadAttach,YAHOO.util.Event,true);A._dri=setInterval(function(){var C=document.createElement("p");try{C.doScroll("left");clearInterval(A._dri);A._dri=null;A._ready();C=null;}catch(B){C=null;}},A.POLL_INTERVAL);}else{if(A.webkit){A._dri=setInterval(function(){var B=document.readyState;if("loaded"==B||"complete"==B){clearInterval(A._dri);A._dri=null;A._ready();}},A.POLL_INTERVAL);}else{A._simpleAdd(document,"DOMContentLoaded",A._ready);}}A._simpleAdd(window,"load",A._load);A._simpleAdd(window,"unload",A._unload);A._tryPreloadAttach();})();}YAHOO.util.EventProvider=function(){};YAHOO.util.EventProvider.prototype={__yui_events:null,__yui_subscribers:null,subscribe:function(A,C,F,E){this.__yui_events=this.__yui_events||{};var D=this.__yui_events[A];if(D){D.subscribe(C,F,E);}else{this.__yui_subscribers=this.__yui_subscribers||{};var B=this.__yui_subscribers;if(!B[A]){B[A]=[];}B[A].push({fn:C,obj:F,override:E});}},unsubscribe:function(C,E,G){this.__yui_events=this.__yui_events||{};var A=this.__yui_events;if(C){var F=A[C];if(F){return F.unsubscribe(E,G);}}else{var B=true;for(var D in A){if(YAHOO.lang.hasOwnProperty(A,D)){B=B&&A[D].unsubscribe(E,G);}}return B;}return false;},unsubscribeAll:function(A){return this.unsubscribe(A);},createEvent:function(G,D){this.__yui_events=this.__yui_events||{};var A=D||{};var I=this.__yui_events;if(I[G]){}else{var H=A.scope||this;var E=(A.silent);var B=new YAHOO.util.CustomEvent(G,H,E,YAHOO.util.CustomEvent.FLAT);I[G]=B;if(A.onSubscribeCallback){B.subscribeEvent.subscribe(A.onSubscribeCallback);}this.__yui_subscribers=this.__yui_subscribers||{};
+var F=this.__yui_subscribers[G];if(F){for(var C=0;C<F.length;++C){B.subscribe(F[C].fn,F[C].obj,F[C].override);}}}return I[G];},fireEvent:function(E,D,A,C){this.__yui_events=this.__yui_events||{};var G=this.__yui_events[E];if(!G){return null;}var B=[];for(var F=1;F<arguments.length;++F){B.push(arguments[F]);}return G.fire.apply(G,B);},hasEvent:function(A){if(this.__yui_events){if(this.__yui_events[A]){return true;}}return false;}};YAHOO.util.KeyListener=function(A,F,B,C){if(!A){}else{if(!F){}else{if(!B){}}}if(!C){C=YAHOO.util.KeyListener.KEYDOWN;}var D=new YAHOO.util.CustomEvent("keyPressed");this.enabledEvent=new YAHOO.util.CustomEvent("enabled");this.disabledEvent=new YAHOO.util.CustomEvent("disabled");if(typeof A=="string"){A=document.getElementById(A);}if(typeof B=="function"){D.subscribe(B);}else{D.subscribe(B.fn,B.scope,B.correctScope);}function E(J,I){if(!F.shift){F.shift=false;}if(!F.alt){F.alt=false;}if(!F.ctrl){F.ctrl=false;}if(J.shiftKey==F.shift&&J.altKey==F.alt&&J.ctrlKey==F.ctrl){var G;if(F.keys instanceof Array){for(var H=0;H<F.keys.length;H++){G=F.keys[H];if(G==J.charCode){D.fire(J.charCode,J);break;}else{if(G==J.keyCode){D.fire(J.keyCode,J);break;}}}}else{G=F.keys;if(G==J.charCode){D.fire(J.charCode,J);}else{if(G==J.keyCode){D.fire(J.keyCode,J);}}}}}this.enable=function(){if(!this.enabled){YAHOO.util.Event.addListener(A,C,E);this.enabledEvent.fire(F);}this.enabled=true;};this.disable=function(){if(this.enabled){YAHOO.util.Event.removeListener(A,C,E);this.disabledEvent.fire(F);}this.enabled=false;};this.toString=function(){return"KeyListener ["+F.keys+"] "+A.tagName+(A.id?"["+A.id+"]":"");};};YAHOO.util.KeyListener.KEYDOWN="keydown";YAHOO.util.KeyListener.KEYUP="keyup";YAHOO.util.KeyListener.KEY={ALT:18,BACK_SPACE:8,CAPS_LOCK:20,CONTROL:17,DELETE:46,DOWN:40,END:35,ENTER:13,ESCAPE:27,HOME:36,LEFT:37,META:224,NUM_LOCK:144,PAGE_DOWN:34,PAGE_UP:33,PAUSE:19,PRINTSCREEN:44,RIGHT:39,SCROLL_LOCK:145,SHIFT:16,SPACE:32,TAB:9,UP:38};YAHOO.register("event",YAHOO.util.Event,{version:"2.4.1",build:"742"});YAHOO.register("yahoo-dom-event", YAHOO, {version: "2.4.1", build: "742"});
diff --git a/sonar-server/src/main/webapp/javascripts/calendar/yahoo-dom-event.js b/sonar-server/src/main/webapp/javascripts/calendar/yahoo-dom-event.js
deleted file mode 100644 (file)
index 106a144..0000000
+++ /dev/null
@@ -1,10 +0,0 @@
-/*
-Copyright (c) 2007, Yahoo! Inc. All rights reserved.
-Code licensed under the BSD License:
-http://developer.yahoo.net/yui/license.txt
-version: 2.4.1
-*/
-if(typeof YAHOO=="undefined"||!YAHOO){var YAHOO={};}YAHOO.namespace=function(){var A=arguments,E=null,C,B,D;for(C=0;C<A.length;C=C+1){D=A[C].split(".");E=YAHOO;for(B=(D[0]=="YAHOO")?1:0;B<D.length;B=B+1){E[D[B]]=E[D[B]]||{};E=E[D[B]];}}return E;};YAHOO.log=function(D,A,C){var B=YAHOO.widget.Logger;if(B&&B.log){return B.log(D,A,C);}else{return false;}};YAHOO.register=function(A,E,D){var I=YAHOO.env.modules;if(!I[A]){I[A]={versions:[],builds:[]};}var B=I[A],H=D.version,G=D.build,F=YAHOO.env.listeners;B.name=A;B.version=H;B.build=G;B.versions.push(H);B.builds.push(G);B.mainClass=E;for(var C=0;C<F.length;C=C+1){F[C](B);}if(E){E.VERSION=H;E.BUILD=G;}else{YAHOO.log("mainClass is undefined for module "+A,"warn");}};YAHOO.env=YAHOO.env||{modules:[],listeners:[]};YAHOO.env.getVersion=function(A){return YAHOO.env.modules[A]||null;};YAHOO.env.ua=function(){var C={ie:0,opera:0,gecko:0,webkit:0,mobile:null};var B=navigator.userAgent,A;if((/KHTML/).test(B)){C.webkit=1;}A=B.match(/AppleWebKit\/([^\s]*)/);if(A&&A[1]){C.webkit=parseFloat(A[1]);if(/ Mobile\//.test(B)){C.mobile="Apple";}else{A=B.match(/NokiaN[^\/]*/);if(A){C.mobile=A[0];}}}if(!C.webkit){A=B.match(/Opera[\s\/]([^\s]*)/);if(A&&A[1]){C.opera=parseFloat(A[1]);A=B.match(/Opera Mini[^;]*/);if(A){C.mobile=A[0];}}else{A=B.match(/MSIE\s([^;]*)/);if(A&&A[1]){C.ie=parseFloat(A[1]);}else{A=B.match(/Gecko\/([^\s]*)/);if(A){C.gecko=1;A=B.match(/rv:([^\s\)]*)/);if(A&&A[1]){C.gecko=parseFloat(A[1]);}}}}}return C;}();(function(){YAHOO.namespace("util","widget","example");if("undefined"!==typeof YAHOO_config){var B=YAHOO_config.listener,A=YAHOO.env.listeners,D=true,C;if(B){for(C=0;C<A.length;C=C+1){if(A[C]==B){D=false;break;}}if(D){A.push(B);}}}})();YAHOO.lang=YAHOO.lang||{isArray:function(B){if(B){var A=YAHOO.lang;return A.isNumber(B.length)&&A.isFunction(B.splice);}return false;},isBoolean:function(A){return typeof A==="boolean";},isFunction:function(A){return typeof A==="function";},isNull:function(A){return A===null;},isNumber:function(A){return typeof A==="number"&&isFinite(A);},isObject:function(A){return(A&&(typeof A==="object"||YAHOO.lang.isFunction(A)))||false;},isString:function(A){return typeof A==="string";},isUndefined:function(A){return typeof A==="undefined";},hasOwnProperty:function(A,B){if(Object.prototype.hasOwnProperty){return A.hasOwnProperty(B);}return !YAHOO.lang.isUndefined(A[B])&&A.constructor.prototype[B]!==A[B];},_IEEnumFix:function(C,B){if(YAHOO.env.ua.ie){var E=["toString","valueOf"],A;for(A=0;A<E.length;A=A+1){var F=E[A],D=B[F];if(YAHOO.lang.isFunction(D)&&D!=Object.prototype[F]){C[F]=D;}}}},extend:function(D,E,C){if(!E||!D){throw new Error("YAHOO.lang.extend failed, please check that all dependencies are included.");}var B=function(){};B.prototype=E.prototype;D.prototype=new B();D.prototype.constructor=D;D.superclass=E.prototype;if(E.prototype.constructor==Object.prototype.constructor){E.prototype.constructor=E;}if(C){for(var A in C){D.prototype[A]=C[A];}YAHOO.lang._IEEnumFix(D.prototype,C);}},augmentObject:function(E,D){if(!D||!E){throw new Error("Absorb failed, verify dependencies.");}var A=arguments,C,F,B=A[2];if(B&&B!==true){for(C=2;C<A.length;C=C+1){E[A[C]]=D[A[C]];}}else{for(F in D){if(B||!E[F]){E[F]=D[F];}}YAHOO.lang._IEEnumFix(E,D);}},augmentProto:function(D,C){if(!C||!D){throw new Error("Augment failed, verify dependencies.");}var A=[D.prototype,C.prototype];for(var B=2;B<arguments.length;B=B+1){A.push(arguments[B]);}YAHOO.lang.augmentObject.apply(this,A);},dump:function(A,G){var C=YAHOO.lang,D,F,I=[],J="{...}",B="f(){...}",H=", ",E=" => ";if(!C.isObject(A)){return A+"";}else{if(A instanceof Date||("nodeType" in A&&"tagName" in A)){return A;}else{if(C.isFunction(A)){return B;}}}G=(C.isNumber(G))?G:3;if(C.isArray(A)){I.push("[");for(D=0,F=A.length;D<F;D=D+1){if(C.isObject(A[D])){I.push((G>0)?C.dump(A[D],G-1):J);}else{I.push(A[D]);}I.push(H);}if(I.length>1){I.pop();}I.push("]");}else{I.push("{");for(D in A){if(C.hasOwnProperty(A,D)){I.push(D+E);if(C.isObject(A[D])){I.push((G>0)?C.dump(A[D],G-1):J);}else{I.push(A[D]);}I.push(H);}}if(I.length>1){I.pop();}I.push("}");}return I.join("");},substitute:function(Q,B,J){var G,F,E,M,N,P,D=YAHOO.lang,L=[],C,H="dump",K=" ",A="{",O="}";for(;;){G=Q.lastIndexOf(A);if(G<0){break;}F=Q.indexOf(O,G);if(G+1>=F){break;}C=Q.substring(G+1,F);M=C;P=null;E=M.indexOf(K);if(E>-1){P=M.substring(E+1);M=M.substring(0,E);}N=B[M];if(J){N=J(M,N,P);}if(D.isObject(N)){if(D.isArray(N)){N=D.dump(N,parseInt(P,10));}else{P=P||"";var I=P.indexOf(H);if(I>-1){P=P.substring(4);}if(N.toString===Object.prototype.toString||I>-1){N=D.dump(N,parseInt(P,10));}else{N=N.toString();}}}else{if(!D.isString(N)&&!D.isNumber(N)){N="~-"+L.length+"-~";L[L.length]=C;}}Q=Q.substring(0,G)+N+Q.substring(F+1);}for(G=L.length-1;G>=0;G=G-1){Q=Q.replace(new RegExp("~-"+G+"-~"),"{"+L[G]+"}","g");}return Q;},trim:function(A){try{return A.replace(/^\s+|\s+$/g,"");}catch(B){return A;}},merge:function(){var D={},B=arguments;for(var C=0,A=B.length;C<A;C=C+1){YAHOO.lang.augmentObject(D,B[C],true);}return D;},later:function(H,B,I,D,E){H=H||0;B=B||{};var C=I,G=D,F,A;if(YAHOO.lang.isString(I)){C=B[I];}if(!C){throw new TypeError("method undefined");}if(!YAHOO.lang.isArray(G)){G=[D];}F=function(){C.apply(B,G);};A=(E)?setInterval(F,H):setTimeout(F,H);return{interval:E,cancel:function(){if(this.interval){clearInterval(A);}else{clearTimeout(A);}}};},isValue:function(B){var A=YAHOO.lang;return(A.isObject(B)||A.isString(B)||A.isNumber(B)||A.isBoolean(B));}};YAHOO.util.Lang=YAHOO.lang;YAHOO.lang.augment=YAHOO.lang.augmentProto;YAHOO.augment=YAHOO.lang.augmentProto;YAHOO.extend=YAHOO.lang.extend;YAHOO.register("yahoo",YAHOO,{version:"2.4.1",build:"742"});(function(){var B=YAHOO.util,L,J,H=0,K={},F={},N=window.document;var C=YAHOO.env.ua.opera,M=YAHOO.env.ua.webkit,A=YAHOO.env.ua.gecko,G=YAHOO.env.ua.ie;var E={HYPHEN:/(-[a-z])/i,ROOT_TAG:/^body|html$/i};var O=function(Q){if(!E.HYPHEN.test(Q)){return Q;}if(K[Q]){return K[Q];}var R=Q;while(E.HYPHEN.exec(R)){R=R.replace(RegExp.$1,RegExp.$1.substr(1).toUpperCase());}K[Q]=R;return R;};var P=function(R){var Q=F[R];if(!Q){Q=new RegExp("(?:^|\\s+)"+R+"(?:\\s+|$)");F[R]=Q;}return Q;};if(N.defaultView&&N.defaultView.getComputedStyle){L=function(Q,T){var S=null;if(T=="float"){T="cssFloat";}var R=N.defaultView.getComputedStyle(Q,"");if(R){S=R[O(T)];}return Q.style[T]||S;};}else{if(N.documentElement.currentStyle&&G){L=function(Q,S){switch(O(S)){case"opacity":var U=100;try{U=Q.filters["DXImageTransform.Microsoft.Alpha"].opacity;}catch(T){try{U=Q.filters("alpha").opacity;}catch(T){}}return U/100;case"float":S="styleFloat";default:var R=Q.currentStyle?Q.currentStyle[S]:null;return(Q.style[S]||R);}};}else{L=function(Q,R){return Q.style[R];};}}if(G){J=function(Q,R,S){switch(R){case"opacity":if(YAHOO.lang.isString(Q.style.filter)){Q.style.filter="alpha(opacity="+S*100+")";if(!Q.currentStyle||!Q.currentStyle.hasLayout){Q.style.zoom=1;}}break;case"float":R="styleFloat";default:Q.style[R]=S;}};}else{J=function(Q,R,S){if(R=="float"){R="cssFloat";}Q.style[R]=S;};}var D=function(Q,R){return Q&&Q.nodeType==1&&(!R||R(Q));};YAHOO.util.Dom={get:function(S){if(S&&(S.tagName||S.item)){return S;}if(YAHOO.lang.isString(S)||!S){return N.getElementById(S);}if(S.length!==undefined){var T=[];for(var R=0,Q=S.length;R<Q;++R){T[T.length]=B.Dom.get(S[R]);}return T;}return S;},getStyle:function(Q,S){S=O(S);var R=function(T){return L(T,S);};return B.Dom.batch(Q,R,B.Dom,true);},setStyle:function(Q,S,T){S=O(S);var R=function(U){J(U,S,T);};B.Dom.batch(Q,R,B.Dom,true);},getXY:function(Q){var R=function(S){if((S.parentNode===null||S.offsetParent===null||this.getStyle(S,"display")=="none")&&S!=S.ownerDocument.body){return false;}return I(S);};return B.Dom.batch(Q,R,B.Dom,true);},getX:function(Q){var R=function(S){return B.Dom.getXY(S)[0];};return B.Dom.batch(Q,R,B.Dom,true);},getY:function(Q){var R=function(S){return B.Dom.getXY(S)[1];};return B.Dom.batch(Q,R,B.Dom,true);},setXY:function(Q,T,S){var R=function(W){var V=this.getStyle(W,"position");if(V=="static"){this.setStyle(W,"position","relative");V="relative";}var Y=this.getXY(W);if(Y===false){return false;}var X=[parseInt(this.getStyle(W,"left"),10),parseInt(this.getStyle(W,"top"),10)];if(isNaN(X[0])){X[0]=(V=="relative")?0:W.offsetLeft;}if(isNaN(X[1])){X[1]=(V=="relative")?0:W.offsetTop;}if(T[0]!==null){W.style.left=T[0]-Y[0]+X[0]+"px";}if(T[1]!==null){W.style.top=T[1]-Y[1]+X[1]+"px";}if(!S){var U=this.getXY(W);if((T[0]!==null&&U[0]!=T[0])||(T[1]!==null&&U[1]!=T[1])){this.setXY(W,T,true);}}};B.Dom.batch(Q,R,B.Dom,true);},setX:function(R,Q){B.Dom.setXY(R,[Q,null]);},setY:function(Q,R){B.Dom.setXY(Q,[null,R]);},getRegion:function(Q){var R=function(S){if((S.parentNode===null||S.offsetParent===null||this.getStyle(S,"display")=="none")&&S!=N.body){return false;}var T=B.Region.getRegion(S);return T;};return B.Dom.batch(Q,R,B.Dom,true);},getClientWidth:function(){return B.Dom.getViewportWidth();},getClientHeight:function(){return B.Dom.getViewportHeight();},getElementsByClassName:function(U,Y,V,W){Y=Y||"*";V=(V)?B.Dom.get(V):null||N;if(!V){return[];}var R=[],Q=V.getElementsByTagName(Y),X=P(U);for(var S=0,T=Q.length;S<T;++S){if(X.test(Q[S].className)){R[R.length]=Q[S];if(W){W.call(Q[S],Q[S]);}}}return R;},hasClass:function(S,R){var Q=P(R);var T=function(U){return Q.test(U.className);};return B.Dom.batch(S,T,B.Dom,true);},addClass:function(R,Q){var S=function(T){if(this.hasClass(T,Q)){return false;}T.className=YAHOO.lang.trim([T.className,Q].join(" "));return true;};return B.Dom.batch(R,S,B.Dom,true);},removeClass:function(S,R){var Q=P(R);var T=function(U){if(!this.hasClass(U,R)){return false;}var V=U.className;U.className=V.replace(Q," ");if(this.hasClass(U,R)){this.removeClass(U,R);}U.className=YAHOO.lang.trim(U.className);return true;};return B.Dom.batch(S,T,B.Dom,true);},replaceClass:function(T,R,Q){if(!Q||R===Q){return false;}var S=P(R);var U=function(V){if(!this.hasClass(V,R)){this.addClass(V,Q);return true;}V.className=V.className.replace(S," "+Q+" ");if(this.hasClass(V,R)){this.replaceClass(V,R,Q);}V.className=YAHOO.lang.trim(V.className);return true;};return B.Dom.batch(T,U,B.Dom,true);},generateId:function(Q,S){S=S||"yui-gen";var R=function(T){if(T&&T.id){return T.id;}var U=S+H++;if(T){T.id=U;}return U;};return B.Dom.batch(Q,R,B.Dom,true)||R.apply(B.Dom,arguments);},isAncestor:function(Q,R){Q=B.Dom.get(Q);R=B.Dom.get(R);if(!Q||!R){return false;}if(Q.contains&&R.nodeType&&!M){return Q.contains(R);}else{if(Q.compareDocumentPosition&&R.nodeType){return !!(Q.compareDocumentPosition(R)&16);}else{if(R.nodeType){return !!this.getAncestorBy(R,function(S){return S==Q;});}}}return false;},inDocument:function(Q){return this.isAncestor(N.documentElement,Q);},getElementsBy:function(X,R,S,U){R=R||"*";S=(S)?B.Dom.get(S):null||N;if(!S){return[];}var T=[],W=S.getElementsByTagName(R);for(var V=0,Q=W.length;V<Q;++V){if(X(W[V])){T[T.length]=W[V];if(U){U(W[V]);}}}return T;},batch:function(U,X,W,S){U=(U&&(U.tagName||U.item))?U:B.Dom.get(U);if(!U||!X){return false;}var T=(S)?W:window;if(U.tagName||U.length===undefined){return X.call(T,U,W);}var V=[];for(var R=0,Q=U.length;R<Q;++R){V[V.length]=X.call(T,U[R],W);}return V;},getDocumentHeight:function(){var R=(N.compatMode!="CSS1Compat")?N.body.scrollHeight:N.documentElement.scrollHeight;var Q=Math.max(R,B.Dom.getViewportHeight());return Q;},getDocumentWidth:function(){var R=(N.compatMode!="CSS1Compat")?N.body.scrollWidth:N.documentElement.scrollWidth;var Q=Math.max(R,B.Dom.getViewportWidth());return Q;},getViewportHeight:function(){var Q=self.innerHeight;var R=N.compatMode;if((R||G)&&!C){Q=(R=="CSS1Compat")?N.documentElement.clientHeight:N.body.clientHeight;
-}return Q;},getViewportWidth:function(){var Q=self.innerWidth;var R=N.compatMode;if(R||G){Q=(R=="CSS1Compat")?N.documentElement.clientWidth:N.body.clientWidth;}return Q;},getAncestorBy:function(Q,R){while(Q=Q.parentNode){if(D(Q,R)){return Q;}}return null;},getAncestorByClassName:function(R,Q){R=B.Dom.get(R);if(!R){return null;}var S=function(T){return B.Dom.hasClass(T,Q);};return B.Dom.getAncestorBy(R,S);},getAncestorByTagName:function(R,Q){R=B.Dom.get(R);if(!R){return null;}var S=function(T){return T.tagName&&T.tagName.toUpperCase()==Q.toUpperCase();};return B.Dom.getAncestorBy(R,S);},getPreviousSiblingBy:function(Q,R){while(Q){Q=Q.previousSibling;if(D(Q,R)){return Q;}}return null;},getPreviousSibling:function(Q){Q=B.Dom.get(Q);if(!Q){return null;}return B.Dom.getPreviousSiblingBy(Q);},getNextSiblingBy:function(Q,R){while(Q){Q=Q.nextSibling;if(D(Q,R)){return Q;}}return null;},getNextSibling:function(Q){Q=B.Dom.get(Q);if(!Q){return null;}return B.Dom.getNextSiblingBy(Q);},getFirstChildBy:function(Q,S){var R=(D(Q.firstChild,S))?Q.firstChild:null;return R||B.Dom.getNextSiblingBy(Q.firstChild,S);},getFirstChild:function(Q,R){Q=B.Dom.get(Q);if(!Q){return null;}return B.Dom.getFirstChildBy(Q);},getLastChildBy:function(Q,S){if(!Q){return null;}var R=(D(Q.lastChild,S))?Q.lastChild:null;return R||B.Dom.getPreviousSiblingBy(Q.lastChild,S);},getLastChild:function(Q){Q=B.Dom.get(Q);return B.Dom.getLastChildBy(Q);},getChildrenBy:function(R,T){var S=B.Dom.getFirstChildBy(R,T);var Q=S?[S]:[];B.Dom.getNextSiblingBy(S,function(U){if(!T||T(U)){Q[Q.length]=U;}return false;});return Q;},getChildren:function(Q){Q=B.Dom.get(Q);if(!Q){}return B.Dom.getChildrenBy(Q);},getDocumentScrollLeft:function(Q){Q=Q||N;return Math.max(Q.documentElement.scrollLeft,Q.body.scrollLeft);},getDocumentScrollTop:function(Q){Q=Q||N;return Math.max(Q.documentElement.scrollTop,Q.body.scrollTop);},insertBefore:function(R,Q){R=B.Dom.get(R);Q=B.Dom.get(Q);if(!R||!Q||!Q.parentNode){return null;}return Q.parentNode.insertBefore(R,Q);},insertAfter:function(R,Q){R=B.Dom.get(R);Q=B.Dom.get(Q);if(!R||!Q||!Q.parentNode){return null;}if(Q.nextSibling){return Q.parentNode.insertBefore(R,Q.nextSibling);}else{return Q.parentNode.appendChild(R);}},getClientRegion:function(){var S=B.Dom.getDocumentScrollTop(),R=B.Dom.getDocumentScrollLeft(),T=B.Dom.getViewportWidth()+R,Q=B.Dom.getViewportHeight()+S;return new B.Region(S,T,Q,R);}};var I=function(){if(N.documentElement.getBoundingClientRect){return function(R){var S=R.getBoundingClientRect();var Q=R.ownerDocument;return[S.left+B.Dom.getDocumentScrollLeft(Q),S.top+B.Dom.getDocumentScrollTop(Q)];};}else{return function(S){var T=[S.offsetLeft,S.offsetTop];var R=S.offsetParent;var Q=(M&&B.Dom.getStyle(S,"position")=="absolute"&&S.offsetParent==S.ownerDocument.body);if(R!=S){while(R){T[0]+=R.offsetLeft;T[1]+=R.offsetTop;if(!Q&&M&&B.Dom.getStyle(R,"position")=="absolute"){Q=true;}R=R.offsetParent;}}if(Q){T[0]-=S.ownerDocument.body.offsetLeft;T[1]-=S.ownerDocument.body.offsetTop;}R=S.parentNode;while(R.tagName&&!E.ROOT_TAG.test(R.tagName)){if(B.Dom.getStyle(R,"display").search(/^inline|table-row.*$/i)){T[0]-=R.scrollLeft;T[1]-=R.scrollTop;}R=R.parentNode;}return T;};}}();})();YAHOO.util.Region=function(C,D,A,B){this.top=C;this[1]=C;this.right=D;this.bottom=A;this.left=B;this[0]=B;};YAHOO.util.Region.prototype.contains=function(A){return(A.left>=this.left&&A.right<=this.right&&A.top>=this.top&&A.bottom<=this.bottom);};YAHOO.util.Region.prototype.getArea=function(){return((this.bottom-this.top)*(this.right-this.left));};YAHOO.util.Region.prototype.intersect=function(E){var C=Math.max(this.top,E.top);var D=Math.min(this.right,E.right);var A=Math.min(this.bottom,E.bottom);var B=Math.max(this.left,E.left);if(A>=C&&D>=B){return new YAHOO.util.Region(C,D,A,B);}else{return null;}};YAHOO.util.Region.prototype.union=function(E){var C=Math.min(this.top,E.top);var D=Math.max(this.right,E.right);var A=Math.max(this.bottom,E.bottom);var B=Math.min(this.left,E.left);return new YAHOO.util.Region(C,D,A,B);};YAHOO.util.Region.prototype.toString=function(){return("Region {top: "+this.top+", right: "+this.right+", bottom: "+this.bottom+", left: "+this.left+"}");};YAHOO.util.Region.getRegion=function(D){var F=YAHOO.util.Dom.getXY(D);var C=F[1];var E=F[0]+D.offsetWidth;var A=F[1]+D.offsetHeight;var B=F[0];return new YAHOO.util.Region(C,E,A,B);};YAHOO.util.Point=function(A,B){if(YAHOO.lang.isArray(A)){B=A[1];A=A[0];}this.x=this.right=this.left=this[0]=A;this.y=this.top=this.bottom=this[1]=B;};YAHOO.util.Point.prototype=new YAHOO.util.Region();YAHOO.register("dom",YAHOO.util.Dom,{version:"2.4.1",build:"742"});YAHOO.util.CustomEvent=function(D,B,C,A){this.type=D;this.scope=B||window;this.silent=C;this.signature=A||YAHOO.util.CustomEvent.LIST;this.subscribers=[];if(!this.silent){}var E="_YUICEOnSubscribe";if(D!==E){this.subscribeEvent=new YAHOO.util.CustomEvent(E,this,true);}this.lastError=null;};YAHOO.util.CustomEvent.LIST=0;YAHOO.util.CustomEvent.FLAT=1;YAHOO.util.CustomEvent.prototype={subscribe:function(B,C,A){if(!B){throw new Error("Invalid callback for subscriber to '"+this.type+"'");}if(this.subscribeEvent){this.subscribeEvent.fire(B,C,A);}this.subscribers.push(new YAHOO.util.Subscriber(B,C,A));},unsubscribe:function(D,F){if(!D){return this.unsubscribeAll();}var E=false;for(var B=0,A=this.subscribers.length;B<A;++B){var C=this.subscribers[B];if(C&&C.contains(D,F)){this._delete(B);E=true;}}return E;},fire:function(){var D=this.subscribers.length;if(!D&&this.silent){return true;}var H=[],F=true,C,I=false;for(C=0;C<arguments.length;++C){H.push(arguments[C]);}if(!this.silent){}for(C=0;C<D;++C){var L=this.subscribers[C];if(!L){I=true;}else{if(!this.silent){}var K=L.getScope(this.scope);if(this.signature==YAHOO.util.CustomEvent.FLAT){var A=null;if(H.length>0){A=H[0];}try{F=L.fn.call(K,A,L.obj);}catch(E){this.lastError=E;}}else{try{F=L.fn.call(K,this.type,H,L.obj);}catch(G){this.lastError=G;}}if(false===F){if(!this.silent){}return false;}}}if(I){var J=[],B=this.subscribers;for(C=0,D=B.length;C<D;C=C+1){J.push(B[C]);}this.subscribers=J;}return true;},unsubscribeAll:function(){for(var B=0,A=this.subscribers.length;B<A;++B){this._delete(A-1-B);}this.subscribers=[];return B;},_delete:function(A){var B=this.subscribers[A];if(B){delete B.fn;delete B.obj;}this.subscribers[A]=null;},toString:function(){return"CustomEvent: '"+this.type+"', scope: "+this.scope;}};YAHOO.util.Subscriber=function(B,C,A){this.fn=B;this.obj=YAHOO.lang.isUndefined(C)?null:C;this.override=A;};YAHOO.util.Subscriber.prototype.getScope=function(A){if(this.override){if(this.override===true){return this.obj;}else{return this.override;}}return A;};YAHOO.util.Subscriber.prototype.contains=function(A,B){if(B){return(this.fn==A&&this.obj==B);}else{return(this.fn==A);}};YAHOO.util.Subscriber.prototype.toString=function(){return"Subscriber { obj: "+this.obj+", override: "+(this.override||"no")+" }";};if(!YAHOO.util.Event){YAHOO.util.Event=function(){var H=false;var I=[];var J=[];var G=[];var E=[];var C=0;var F=[];var B=[];var A=0;var D={63232:38,63233:40,63234:37,63235:39,63276:33,63277:34,25:9};return{POLL_RETRYS:4000,POLL_INTERVAL:10,EL:0,TYPE:1,FN:2,WFN:3,UNLOAD_OBJ:3,ADJ_SCOPE:4,OBJ:5,OVERRIDE:6,lastError:null,isSafari:YAHOO.env.ua.webkit,webkit:YAHOO.env.ua.webkit,isIE:YAHOO.env.ua.ie,_interval:null,_dri:null,DOMReady:false,startInterval:function(){if(!this._interval){var K=this;var L=function(){K._tryPreloadAttach();};this._interval=setInterval(L,this.POLL_INTERVAL);}},onAvailable:function(P,M,Q,O,N){var K=(YAHOO.lang.isString(P))?[P]:P;for(var L=0;L<K.length;L=L+1){F.push({id:K[L],fn:M,obj:Q,override:O,checkReady:N});}C=this.POLL_RETRYS;this.startInterval();},onContentReady:function(M,K,N,L){this.onAvailable(M,K,N,L,true);},onDOMReady:function(K,M,L){if(this.DOMReady){setTimeout(function(){var N=window;if(L){if(L===true){N=M;}else{N=L;}}K.call(N,"DOMReady",[],M);},0);}else{this.DOMReadyEvent.subscribe(K,M,L);}},addListener:function(M,K,V,Q,L){if(!V||!V.call){return false;}if(this._isValidCollection(M)){var W=true;for(var R=0,T=M.length;R<T;++R){W=this.on(M[R],K,V,Q,L)&&W;}return W;}else{if(YAHOO.lang.isString(M)){var P=this.getEl(M);if(P){M=P;}else{this.onAvailable(M,function(){YAHOO.util.Event.on(M,K,V,Q,L);});return true;}}}if(!M){return false;}if("unload"==K&&Q!==this){J[J.length]=[M,K,V,Q,L];return true;}var Y=M;if(L){if(L===true){Y=Q;}else{Y=L;}}var N=function(Z){return V.call(Y,YAHOO.util.Event.getEvent(Z,M),Q);};var X=[M,K,V,N,Y,Q,L];var S=I.length;I[S]=X;if(this.useLegacyEvent(M,K)){var O=this.getLegacyIndex(M,K);if(O==-1||M!=G[O][0]){O=G.length;B[M.id+K]=O;G[O]=[M,K,M["on"+K]];E[O]=[];M["on"+K]=function(Z){YAHOO.util.Event.fireLegacyEvent(YAHOO.util.Event.getEvent(Z),O);};}E[O].push(X);}else{try{this._simpleAdd(M,K,N,false);}catch(U){this.lastError=U;this.removeListener(M,K,V);return false;}}return true;},fireLegacyEvent:function(O,M){var Q=true,K,S,R,T,P;S=E[M];for(var L=0,N=S.length;L<N;++L){R=S[L];if(R&&R[this.WFN]){T=R[this.ADJ_SCOPE];P=R[this.WFN].call(T,O);Q=(Q&&P);}}K=G[M];if(K&&K[2]){K[2](O);}return Q;},getLegacyIndex:function(L,M){var K=this.generateId(L)+M;if(typeof B[K]=="undefined"){return -1;}else{return B[K];}},useLegacyEvent:function(L,M){if(this.webkit&&("click"==M||"dblclick"==M)){var K=parseInt(this.webkit,10);if(!isNaN(K)&&K<418){return true;}}return false;},removeListener:function(L,K,T){var O,R,V;if(typeof L=="string"){L=this.getEl(L);}else{if(this._isValidCollection(L)){var U=true;for(O=0,R=L.length;O<R;++O){U=(this.removeListener(L[O],K,T)&&U);}return U;}}if(!T||!T.call){return this.purgeElement(L,false,K);}if("unload"==K){for(O=0,R=J.length;O<R;O++){V=J[O];if(V&&V[0]==L&&V[1]==K&&V[2]==T){J[O]=null;return true;}}return false;}var P=null;var Q=arguments[3];if("undefined"===typeof Q){Q=this._getCacheIndex(L,K,T);}if(Q>=0){P=I[Q];}if(!L||!P){return false;}if(this.useLegacyEvent(L,K)){var N=this.getLegacyIndex(L,K);var M=E[N];if(M){for(O=0,R=M.length;O<R;++O){V=M[O];if(V&&V[this.EL]==L&&V[this.TYPE]==K&&V[this.FN]==T){M[O]=null;break;}}}}else{try{this._simpleRemove(L,K,P[this.WFN],false);}catch(S){this.lastError=S;return false;}}delete I[Q][this.WFN];delete I[Q][this.FN];I[Q]=null;return true;},getTarget:function(M,L){var K=M.target||M.srcElement;return this.resolveTextNode(K);},resolveTextNode:function(K){if(K&&3==K.nodeType){return K.parentNode;}else{return K;}},getPageX:function(L){var K=L.pageX;if(!K&&0!==K){K=L.clientX||0;if(this.isIE){K+=this._getScrollLeft();}}return K;},getPageY:function(K){var L=K.pageY;if(!L&&0!==L){L=K.clientY||0;if(this.isIE){L+=this._getScrollTop();}}return L;},getXY:function(K){return[this.getPageX(K),this.getPageY(K)];
-},getRelatedTarget:function(L){var K=L.relatedTarget;if(!K){if(L.type=="mouseout"){K=L.toElement;}else{if(L.type=="mouseover"){K=L.fromElement;}}}return this.resolveTextNode(K);},getTime:function(M){if(!M.time){var L=new Date().getTime();try{M.time=L;}catch(K){this.lastError=K;return L;}}return M.time;},stopEvent:function(K){this.stopPropagation(K);this.preventDefault(K);},stopPropagation:function(K){if(K.stopPropagation){K.stopPropagation();}else{K.cancelBubble=true;}},preventDefault:function(K){if(K.preventDefault){K.preventDefault();}else{K.returnValue=false;}},getEvent:function(M,K){var L=M||window.event;if(!L){var N=this.getEvent.caller;while(N){L=N.arguments[0];if(L&&Event==L.constructor){break;}N=N.caller;}}return L;},getCharCode:function(L){var K=L.keyCode||L.charCode||0;if(YAHOO.env.ua.webkit&&(K in D)){K=D[K];}return K;},_getCacheIndex:function(O,P,N){for(var M=0,L=I.length;M<L;++M){var K=I[M];if(K&&K[this.FN]==N&&K[this.EL]==O&&K[this.TYPE]==P){return M;}}return -1;},generateId:function(K){var L=K.id;if(!L){L="yuievtautoid-"+A;++A;K.id=L;}return L;},_isValidCollection:function(L){try{return(L&&typeof L!=="string"&&L.length&&!L.tagName&&!L.alert&&typeof L[0]!=="undefined");}catch(K){return false;}},elCache:{},getEl:function(K){return(typeof K==="string")?document.getElementById(K):K;},clearCache:function(){},DOMReadyEvent:new YAHOO.util.CustomEvent("DOMReady",this),_load:function(L){if(!H){H=true;var K=YAHOO.util.Event;K._ready();K._tryPreloadAttach();}},_ready:function(L){var K=YAHOO.util.Event;if(!K.DOMReady){K.DOMReady=true;K.DOMReadyEvent.fire();K._simpleRemove(document,"DOMContentLoaded",K._ready);}},_tryPreloadAttach:function(){if(this.locked){return false;}if(this.isIE){if(!this.DOMReady){this.startInterval();return false;}}this.locked=true;var P=!H;if(!P){P=(C>0);}var O=[];var Q=function(S,T){var R=S;if(T.override){if(T.override===true){R=T.obj;}else{R=T.override;}}T.fn.call(R,T.obj);};var L,K,N,M;for(L=0,K=F.length;L<K;++L){N=F[L];if(N&&!N.checkReady){M=this.getEl(N.id);if(M){Q(M,N);F[L]=null;}else{O.push(N);}}}for(L=0,K=F.length;L<K;++L){N=F[L];if(N&&N.checkReady){M=this.getEl(N.id);if(M){if(H||M.nextSibling){Q(M,N);F[L]=null;}}else{O.push(N);}}}C=(O.length===0)?0:C-1;if(P){this.startInterval();}else{clearInterval(this._interval);this._interval=null;}this.locked=false;return true;},purgeElement:function(O,P,R){var M=(YAHOO.lang.isString(O))?this.getEl(O):O;var Q=this.getListeners(M,R),N,K;if(Q){for(N=0,K=Q.length;N<K;++N){var L=Q[N];this.removeListener(M,L.type,L.fn,L.index);}}if(P&&M&&M.childNodes){for(N=0,K=M.childNodes.length;N<K;++N){this.purgeElement(M.childNodes[N],P,R);}}},getListeners:function(M,K){var P=[],L;if(!K){L=[I,J];}else{if(K==="unload"){L=[J];}else{L=[I];}}var R=(YAHOO.lang.isString(M))?this.getEl(M):M;for(var O=0;O<L.length;O=O+1){var T=L[O];if(T&&T.length>0){for(var Q=0,S=T.length;Q<S;++Q){var N=T[Q];if(N&&N[this.EL]===R&&(!K||K===N[this.TYPE])){P.push({type:N[this.TYPE],fn:N[this.FN],obj:N[this.OBJ],adjust:N[this.OVERRIDE],scope:N[this.ADJ_SCOPE],index:Q});}}}}return(P.length)?P:null;},_unload:function(R){var Q=YAHOO.util.Event,O,N,L,K,M;for(O=0,K=J.length;O<K;++O){L=J[O];if(L){var P=window;if(L[Q.ADJ_SCOPE]){if(L[Q.ADJ_SCOPE]===true){P=L[Q.UNLOAD_OBJ];}else{P=L[Q.ADJ_SCOPE];}}L[Q.FN].call(P,Q.getEvent(R,L[Q.EL]),L[Q.UNLOAD_OBJ]);J[O]=null;L=null;P=null;}}J=null;if(YAHOO.env.ua.ie&&I&&I.length>0){N=I.length;while(N){M=N-1;L=I[M];if(L){Q.removeListener(L[Q.EL],L[Q.TYPE],L[Q.FN],M);}N--;}L=null;}G=null;Q._simpleRemove(window,"unload",Q._unload);},_getScrollLeft:function(){return this._getScroll()[1];},_getScrollTop:function(){return this._getScroll()[0];},_getScroll:function(){var K=document.documentElement,L=document.body;if(K&&(K.scrollTop||K.scrollLeft)){return[K.scrollTop,K.scrollLeft];}else{if(L){return[L.scrollTop,L.scrollLeft];}else{return[0,0];}}},regCE:function(){},_simpleAdd:function(){if(window.addEventListener){return function(M,N,L,K){M.addEventListener(N,L,(K));};}else{if(window.attachEvent){return function(M,N,L,K){M.attachEvent("on"+N,L);};}else{return function(){};}}}(),_simpleRemove:function(){if(window.removeEventListener){return function(M,N,L,K){M.removeEventListener(N,L,(K));};}else{if(window.detachEvent){return function(L,M,K){L.detachEvent("on"+M,K);};}else{return function(){};}}}()};}();(function(){var A=YAHOO.util.Event;A.on=A.addListener;if(A.isIE){YAHOO.util.Event.onDOMReady(YAHOO.util.Event._tryPreloadAttach,YAHOO.util.Event,true);A._dri=setInterval(function(){var C=document.createElement("p");try{C.doScroll("left");clearInterval(A._dri);A._dri=null;A._ready();C=null;}catch(B){C=null;}},A.POLL_INTERVAL);}else{if(A.webkit){A._dri=setInterval(function(){var B=document.readyState;if("loaded"==B||"complete"==B){clearInterval(A._dri);A._dri=null;A._ready();}},A.POLL_INTERVAL);}else{A._simpleAdd(document,"DOMContentLoaded",A._ready);}}A._simpleAdd(window,"load",A._load);A._simpleAdd(window,"unload",A._unload);A._tryPreloadAttach();})();}YAHOO.util.EventProvider=function(){};YAHOO.util.EventProvider.prototype={__yui_events:null,__yui_subscribers:null,subscribe:function(A,C,F,E){this.__yui_events=this.__yui_events||{};var D=this.__yui_events[A];if(D){D.subscribe(C,F,E);}else{this.__yui_subscribers=this.__yui_subscribers||{};var B=this.__yui_subscribers;if(!B[A]){B[A]=[];}B[A].push({fn:C,obj:F,override:E});}},unsubscribe:function(C,E,G){this.__yui_events=this.__yui_events||{};var A=this.__yui_events;if(C){var F=A[C];if(F){return F.unsubscribe(E,G);}}else{var B=true;for(var D in A){if(YAHOO.lang.hasOwnProperty(A,D)){B=B&&A[D].unsubscribe(E,G);}}return B;}return false;},unsubscribeAll:function(A){return this.unsubscribe(A);},createEvent:function(G,D){this.__yui_events=this.__yui_events||{};var A=D||{};var I=this.__yui_events;if(I[G]){}else{var H=A.scope||this;var E=(A.silent);var B=new YAHOO.util.CustomEvent(G,H,E,YAHOO.util.CustomEvent.FLAT);I[G]=B;if(A.onSubscribeCallback){B.subscribeEvent.subscribe(A.onSubscribeCallback);}this.__yui_subscribers=this.__yui_subscribers||{};
-var F=this.__yui_subscribers[G];if(F){for(var C=0;C<F.length;++C){B.subscribe(F[C].fn,F[C].obj,F[C].override);}}}return I[G];},fireEvent:function(E,D,A,C){this.__yui_events=this.__yui_events||{};var G=this.__yui_events[E];if(!G){return null;}var B=[];for(var F=1;F<arguments.length;++F){B.push(arguments[F]);}return G.fire.apply(G,B);},hasEvent:function(A){if(this.__yui_events){if(this.__yui_events[A]){return true;}}return false;}};YAHOO.util.KeyListener=function(A,F,B,C){if(!A){}else{if(!F){}else{if(!B){}}}if(!C){C=YAHOO.util.KeyListener.KEYDOWN;}var D=new YAHOO.util.CustomEvent("keyPressed");this.enabledEvent=new YAHOO.util.CustomEvent("enabled");this.disabledEvent=new YAHOO.util.CustomEvent("disabled");if(typeof A=="string"){A=document.getElementById(A);}if(typeof B=="function"){D.subscribe(B);}else{D.subscribe(B.fn,B.scope,B.correctScope);}function E(J,I){if(!F.shift){F.shift=false;}if(!F.alt){F.alt=false;}if(!F.ctrl){F.ctrl=false;}if(J.shiftKey==F.shift&&J.altKey==F.alt&&J.ctrlKey==F.ctrl){var G;if(F.keys instanceof Array){for(var H=0;H<F.keys.length;H++){G=F.keys[H];if(G==J.charCode){D.fire(J.charCode,J);break;}else{if(G==J.keyCode){D.fire(J.keyCode,J);break;}}}}else{G=F.keys;if(G==J.charCode){D.fire(J.charCode,J);}else{if(G==J.keyCode){D.fire(J.keyCode,J);}}}}}this.enable=function(){if(!this.enabled){YAHOO.util.Event.addListener(A,C,E);this.enabledEvent.fire(F);}this.enabled=true;};this.disable=function(){if(this.enabled){YAHOO.util.Event.removeListener(A,C,E);this.disabledEvent.fire(F);}this.enabled=false;};this.toString=function(){return"KeyListener ["+F.keys+"] "+A.tagName+(A.id?"["+A.id+"]":"");};};YAHOO.util.KeyListener.KEYDOWN="keydown";YAHOO.util.KeyListener.KEYUP="keyup";YAHOO.util.KeyListener.KEY={ALT:18,BACK_SPACE:8,CAPS_LOCK:20,CONTROL:17,DELETE:46,DOWN:40,END:35,ENTER:13,ESCAPE:27,HOME:36,LEFT:37,META:224,NUM_LOCK:144,PAGE_DOWN:34,PAGE_UP:33,PAUSE:19,PRINTSCREEN:44,RIGHT:39,SCROLL_LOCK:145,SHIFT:16,SPACE:32,TAB:9,UP:38};YAHOO.register("event",YAHOO.util.Event,{version:"2.4.1",build:"742"});YAHOO.register("yahoo-dom-event", YAHOO, {version: "2.4.1", build: "742"});
index 845ab7fb4eb488bcb134b6cad2f21e2430ab91ac..04a4779398a968ef045b02caa11ab63c74cc69a8 100644 (file)
@@ -1,5 +1,5 @@
-/*  Prototype JavaScript framework, version 1.6.1
- *  (c) 2005-2009 Sam Stephenson
+/*  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/
@@ -7,7 +7,8 @@
  *--------------------------------------------------------------------------*/
 
 var Prototype = {
-  Version: '1.6.1',
+
+  Version: '1.7',
 
   Browser: (function(){
     var ua = navigator.userAgent;
@@ -17,13 +18,15 @@ var Prototype = {
       Opera:          isOpera,
       WebKit:         ua.indexOf('AppleWebKit/') > -1,
       Gecko:          ua.indexOf('Gecko') > -1 && ua.indexOf('KHTML') === -1,
-      MobileSafari:   /Apple.*Mobile.*Safari/.test(ua)
+      MobileSafari:   /Apple.*Mobile/.test(ua)
     }
   })(),
 
   BrowserFeatures: {
     XPath: !!document.evaluate,
+
     SelectorsAPI: !!document.querySelector,
+
     ElementExtensions: (function() {
       var constructor = window.Element || window.HTMLElement;
       return !!(constructor && constructor.prototype);
@@ -32,9 +35,9 @@ var Prototype = {
       if (typeof window.HTMLDivElement !== 'undefined')
         return true;
 
-      var div = document.createElement('div');
-      var form = document.createElement('form');
-      var isSupported = false;
+      var div = document.createElement('div'),
+          form = document.createElement('form'),
+          isSupported = false;
 
       if (div['__proto__'] && (div['__proto__'] !== form['__proto__'])) {
         isSupported = true;
@@ -50,35 +53,23 @@ var Prototype = {
   JSONFilter: /^\/\*-secure-([\s\S]*)\*\/\s*$/,
 
   emptyFunction: function() { },
+
   K: function(x) { return x }
 };
 
 if (Prototype.Browser.MobileSafari)
   Prototype.BrowserFeatures.SpecificElementExtensions = false;
+/* Based on Alex Arnell's inheritance implementation. */
 
+var Class = (function() {
 
-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) { }
+  var IS_DONTENUM_BUGGY = (function(){
+    for (var p in { toString: 1 }) {
+      if (p === 'toString') return false;
     }
+    return true;
+  })();
 
-    return returnValue;
-  }
-};
-
-/* Based on Alex Arnell's inheritance implementation. */
-
-var Class = (function() {
   function subclass() {};
   function create() {
     var parent = null, properties = $A(arguments);
@@ -99,7 +90,7 @@ var Class = (function() {
       parent.subclasses.push(klass);
     }
 
-    for (var i = 0; i < properties.length; i++)
+    for (var i = 0, length = properties.length; i < length; i++)
       klass.addMethods(properties[i]);
 
     if (!klass.prototype.initialize)
@@ -110,10 +101,10 @@ var Class = (function() {
   }
 
   function addMethods(source) {
-    var ancestor   = this.superclass && this.superclass.prototype;
-    var properties = Object.keys(source);
+    var ancestor   = this.superclass && this.superclass.prototype,
+        properties = Object.keys(source);
 
-    if (!Object.keys({ toString: true }).length) {
+    if (IS_DONTENUM_BUGGY) {
       if (source.toString != Object.prototype.toString)
         properties.push("toString");
       if (source.valueOf != Object.prototype.valueOf)
@@ -123,7 +114,7 @@ var Class = (function() {
     for (var i = 0, length = properties.length; i < length; i++) {
       var property = properties[i], value = source[property];
       if (ancestor && Object.isFunction(value) &&
-          value.argumentNames().first() == "$super") {
+          value.argumentNames()[0] == "$super") {
         var method = value;
         value = (function(m) {
           return function() { return ancestor[m].apply(this, arguments); };
@@ -147,7 +138,37 @@ var Class = (function() {
 })();
 (function() {
 
-  var _toString = Object.prototype.toString;
+  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)
@@ -166,27 +187,70 @@ var Class = (function() {
     }
   }
 
-  function toJSON(object) {
-    var type = typeof object;
-    switch (type) {
-      case 'undefined':
-      case 'function':
-      case 'unknown': return;
-      case 'boolean': return object.toString();
+  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);
     }
 
-    if (object === null) return 'null';
-    if (object.toJSON) return object.toJSON();
-    if (isElement(object)) return;
+    var _class = _toString.call(value);
 
-    var results = [];
-    for (var property in object) {
-      var value = toJSON(object[property]);
-      if (!isUndefined(value))
-        results.push(property.toJSON() + ': ' + 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;
     }
+  }
 
-    return '{' + results.join(', ') + '}';
+  function stringify(object) {
+    return JSON.stringify(object);
   }
 
   function toQueryString(object) {
@@ -198,9 +262,13 @@ var Class = (function() {
   }
 
   function keys(object) {
+    if (Type(object) !== OBJECT_TYPE) { throw new TypeError(); }
     var results = [];
-    for (var property in object)
-      results.push(property);
+    for (var property in object) {
+      if (object.hasOwnProperty(property)) {
+        results.push(property);
+      }
+    }
     return results;
   }
 
@@ -220,24 +288,34 @@ var Class = (function() {
   }
 
   function isArray(object) {
-    return _toString.call(object) == "[object Array]";
+    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 typeof object === "function";
+    return _toString.call(object) === FUNCTION_CLASS;
   }
 
   function isString(object) {
-    return _toString.call(object) == "[object String]";
+    return _toString.call(object) === STRING_CLASS;
   }
 
   function isNumber(object) {
-    return _toString.call(object) == "[object Number]";
+    return _toString.call(object) === NUMBER_CLASS;
+  }
+
+  function isDate(object) {
+    return _toString.call(object) === DATE_CLASS;
   }
 
   function isUndefined(object) {
@@ -247,10 +325,10 @@ var Class = (function() {
   extend(Object, {
     extend:        extend,
     inspect:       inspect,
-    toJSON:        toJSON,
+    toJSON:        NATIVE_JSON_STRINGIFY_SUPPORT ? stringify : toJSON,
     toQueryString: toQueryString,
     toHTML:        toHTML,
-    keys:          keys,
+    keys:          Object.keys || keys,
     values:        values,
     clone:         clone,
     isElement:     isElement,
@@ -259,6 +337,7 @@ var Class = (function() {
     isFunction:    isFunction,
     isString:      isString,
     isNumber:      isNumber,
+    isDate:        isDate,
     isUndefined:   isUndefined
   });
 })();
@@ -311,7 +390,7 @@ Object.extend(Function.prototype, (function() {
 
   function delay(timeout) {
     var __method = this, args = slice.call(arguments, 1);
-    timeout = timeout * 1000
+    timeout = timeout * 1000;
     return window.setTimeout(function() {
       return __method.apply(__method, args);
     }, timeout);
@@ -352,14 +431,28 @@ Object.extend(Function.prototype, (function() {
 })());
 
 
-Date.prototype.toJSON = function() {
-  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(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;
@@ -418,6 +511,9 @@ Object.extend(String, {
 });
 
 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;
@@ -484,8 +580,8 @@ Object.extend(String.prototype, (function() {
   }
 
   function extractScripts() {
-    var matchAll = new RegExp(Prototype.ScriptFragment, 'img');
-    var matchOne = new RegExp(Prototype.ScriptFragment, 'im');
+    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];
     });
@@ -510,8 +606,9 @@ Object.extend(String.prototype, (function() {
 
     return match[1].split(separator || '&').inject({ }, function(hash, pair) {
       if ((pair = pair.split('='))[0]) {
-        var key = decodeURIComponent(pair.shift());
-        var value = pair.length > 1 ? pair.join('=') : pair[0];
+        var key = decodeURIComponent(pair.shift()),
+            value = pair.length > 1 ? pair.join('=') : pair[0];
+
         if (value != undefined) value = decodeURIComponent(value);
 
         if (key in hash) {
@@ -538,17 +635,9 @@ Object.extend(String.prototype, (function() {
   }
 
   function camelize() {
-    var parts = this.split('-'), len = parts.length;
-    if (len == 1) return parts[0];
-
-    var camelized = this.charAt(0) == '-'
-      ? parts[0].charAt(0).toUpperCase() + parts[0].substring(1)
-      : parts[0];
-
-    for (var i = 1; i < len; i++)
-      camelized += parts[i].charAt(0).toUpperCase() + parts[i].substring(1);
-
-    return camelized;
+    return this.replace(/-+(.)?/g, function(match, chr) {
+      return chr ? chr.toUpperCase() : '';
+    });
   }
 
   function capitalize() {
@@ -578,10 +667,6 @@ Object.extend(String.prototype, (function() {
     return "'" + escapedString.replace(/'/g, '\\\'') + "'";
   }
 
-  function toJSON() {
-    return this.inspect(true);
-  }
-
   function unfilterJSON(filter) {
     return this.replace(filter || Prototype.JSONFilter, '$1');
   }
@@ -589,29 +674,42 @@ Object.extend(String.prototype, (function() {
   function isJSON() {
     var str = this;
     if (str.blank()) return false;
-    str = this.replace(/\\./g, '@').replace(/"[^"\\\n\r]*"/g, '');
-    return (/^[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t]*$/).test(str);
+    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();
+    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.indexOf(pattern) === 0;
+    return this.lastIndexOf(pattern, 0) === 0;
   }
 
   function endsWith(pattern) {
     var d = this.length - pattern.length;
-    return d >= 0 && this.lastIndexOf(pattern) === d;
+    return d >= 0 && this.indexOf(pattern, d) === d;
   }
 
   function empty() {
@@ -631,7 +729,7 @@ Object.extend(String.prototype, (function() {
     sub:            sub,
     scan:           scan,
     truncate:       truncate,
-    strip:          String.prototype.trim ? String.prototype.trim : strip,
+    strip:          String.prototype.trim || strip,
     stripTags:      stripTags,
     stripScripts:   stripScripts,
     extractScripts: extractScripts,
@@ -648,10 +746,9 @@ Object.extend(String.prototype, (function() {
     underscore:     underscore,
     dasherize:      dasherize,
     inspect:        inspect,
-    toJSON:         toJSON,
     unfilterJSON:   unfilterJSON,
     isJSON:         isJSON,
-    evalJSON:       evalJSON,
+    evalJSON:       NATIVE_JSON_PARSE_SUPPORT ? parseJSON : evalJSON,
     include:        include,
     startsWith:     startsWith,
     endsWith:       endsWith,
@@ -677,8 +774,9 @@ var Template = Class.create({
       var before = match[1] || '';
       if (before == '\\') return match[2];
 
-      var ctx = object, expr = match[3];
-      var pattern = /^([^.[]+|\[((?:.*?[^\\])?)\])(\.|\[|$)/;
+      var ctx = object, expr = match[3],
+          pattern = /^([^.[]+|\[((?:.*?[^\\])?)\])(\.|\[|$)/;
+
       match = pattern.exec(expr);
       if (match == null) return before;
 
@@ -943,6 +1041,7 @@ var Enumerable = (function() {
     find:       detect
   };
 })();
+
 function $A(iterable) {
   if (!iterable) return [];
   if ('toArray' in Object(iterable)) return iterable.toArray();
@@ -951,6 +1050,7 @@ function $A(iterable) {
   return results;
 }
 
+
 function $w(string) {
   if (!Object.isString(string)) return [];
   string = string.strip();
@@ -965,9 +1065,10 @@ Array.from = $A;
       slice = arrayProto.slice,
       _each = arrayProto.forEach; // use native browser JS 1.6 implementation if available
 
-  function each(iterator) {
-    for (var i = 0, length = this.length; i < length; i++)
-      iterator(this[i]);
+  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;
 
@@ -1007,7 +1108,7 @@ Array.from = $A;
   }
 
   function reverse(inline) {
-    return (inline !== false ? this : this.toArray())._reverse();
+    return (inline === false ? this.toArray() : this)._reverse();
   }
 
   function uniq(sorted) {
@@ -1037,15 +1138,6 @@ Array.from = $A;
     return '[' + this.map(Object.inspect).join(', ') + ']';
   }
 
-  function toJSON() {
-    var results = [];
-    this.each(function(object) {
-      var value = Object.toJSON(object);
-      if (!Object.isUndefined(value)) results.push(value);
-    });
-    return '[' + results.join(', ') + ']';
-  }
-
   function indexOf(item, i) {
     i || (i = 0);
     var length = this.length;
@@ -1094,8 +1186,7 @@ Array.from = $A;
     clone:     clone,
     toArray:   clone,
     size:      size,
-    inspect:   inspect,
-    toJSON:    toJSON
+    inspect:   inspect
   });
 
   var CONCAT_ARGUMENTS_BUGGY = (function() {
@@ -1116,6 +1207,7 @@ var Hash = Class.create(Enumerable, (function() {
     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];
@@ -1144,6 +1236,8 @@ var Hash = Class.create(Enumerable, (function() {
     return Object.clone(this._object);
   }
 
+
+
   function keys() {
     return this.pluck('key');
   }
@@ -1180,8 +1274,14 @@ var Hash = Class.create(Enumerable, (function() {
       var key = encodeURIComponent(pair.key), values = pair.value;
 
       if (values && typeof values == 'object') {
-        if (Object.isArray(values))
-          return results.concat(values.map(toQueryPair.curry(key)));
+        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('&');
@@ -1193,10 +1293,6 @@ var Hash = Class.create(Enumerable, (function() {
     }).join(', ') + '}>';
   }
 
-  function toJSON() {
-    return Object.toJSON(this.toObject());
-  }
-
   function clone() {
     return new Hash(this);
   }
@@ -1216,7 +1312,7 @@ var Hash = Class.create(Enumerable, (function() {
     update:                 update,
     toQueryString:          toQueryString,
     inspect:                inspect,
-    toJSON:                 toJSON,
+    toJSON:                 toObject,
     clone:                  clone
   };
 })());
@@ -1241,10 +1337,6 @@ Object.extend(Number.prototype, (function() {
     return '0'.times(length - string.length) + string;
   }
 
-  function toJSON() {
-    return isFinite(this) ? this.toString() : 'null';
-  }
-
   function abs() {
     return Math.abs(this);
   }
@@ -1266,7 +1358,6 @@ Object.extend(Number.prototype, (function() {
     succ:           succ,
     times:          times,
     toPaddedString: toPaddedString,
-    toJSON:         toJSON,
     abs:            abs,
     round:          round,
     ceil:           ceil,
@@ -1310,6 +1401,25 @@ var ObjectRange = Class.create(Enumerable, (function() {
 
 
 
+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;
+  }
+};
+
 var Ajax = {
   getTransport: function() {
     return Try.these(
@@ -1370,9 +1480,7 @@ Ajax.Base = Class.create({
 
     this.options.method = this.options.method.toLowerCase();
 
-    if (Object.isString(this.options.parameters))
-      this.options.parameters = this.options.parameters.toQueryParams();
-    else if (Object.isHash(this.options.parameters))
+    if (Object.isHash(this.options.parameters))
       this.options.parameters = this.options.parameters.toObject();
   }
 });
@@ -1388,22 +1496,21 @@ Ajax.Request = Class.create(Ajax.Base, {
   request: function(url) {
     this.url = url;
     this.method = this.options.method;
-    var params = Object.clone(this.options.parameters);
+    var params = Object.isString(this.options.parameters) ?
+          this.options.parameters :
+          Object.toQueryString(this.options.parameters);
 
     if (!['get', 'post'].include(this.method)) {
-      params['_method'] = this.method;
+      params += (params ? '&' : '') + "_method=" + this.method;
       this.method = 'post';
     }
 
-    this.parameters = params;
-
-    if (params = Object.toQueryString(params)) {
-      if (this.method == 'get')
-        this.url += (this.url.include('?') ? '&' : '?') + params;
-      else if (/Konqueror|Safari|KHTML/.test(navigator.userAgent))
-        params += '&_=';
+    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);
@@ -1472,11 +1579,12 @@ Ajax.Request = Class.create(Ajax.Base, {
 
   success: function() {
     var status = this.getStatus();
-    return !status || (status >= 200 && status < 300);
+    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 }
   },
@@ -1558,14 +1666,14 @@ Ajax.Response = Class.create({
     var transport  = this.transport  = request.transport,
         readyState = this.readyState = transport.readyState;
 
-    if((readyState > 2 && !Prototype.Browser.IE) || readyState == 4) {
+    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) {
+    if (readyState == 4) {
       var xml = transport.responseXML;
       this.responseXML  = Object.isUndefined(xml) ? null : xml;
       this.responseJSON = this._getResponseJSON();
@@ -1705,7 +1813,6 @@ Ajax.PeriodicalUpdater = Class.create(Ajax.Base, {
 });
 
 
-
 function $(element) {
   if (arguments.length > 1) {
     for (var i = 0, elements = [], length = arguments.length; i < length; i++)
@@ -1730,7 +1837,7 @@ if (Prototype.BrowserFeatures.XPath) {
 
 /*--------------------------------------------------------------------------*/
 
-if (!window.Node) var Node = { };
+if (!Node) var Node = { };
 
 if (!Node.ELEMENT_NODE) {
   Object.extend(Node, {
@@ -1750,42 +1857,61 @@ if (!Node.ELEMENT_NODE) {
 }
 
 
+
 (function(global) {
+  function shouldUseCache(tagName, attributes) {
+    if (tagName === 'select') return false;
+    if ('type' in attributes) return false;
+    return true;
+  }
 
-  var SETATTRIBUTE_IGNORES_NAME = (function(){
-    var elForm = document.createElement("form");
-    var elInput = document.createElement("input");
-    var root = document.documentElement;
-    elInput.setAttribute("name", "test");
-    elForm.appendChild(elInput);
-    root.appendChild(elForm);
-    var isBuggy = elForm.elements
-      ? (typeof elForm.elements.test == "undefined")
-      : null;
-    root.removeChild(elForm);
-    elForm = elInput = null;
-    return isBuggy;
+  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 (SETATTRIBUTE_IGNORES_NAME && attributes.name) {
+
+    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));
-    return Element.writeAttribute(cache[tagName].cloneNode(false), attributes);
+
+    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.cache = { };
 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) {
@@ -1798,7 +1924,6 @@ Element.Methods = {
     return element;
   },
 
-
   hide: function(element) {
     element = $(element);
     element.style.display = 'none';
@@ -1844,6 +1969,21 @@ Element.Methods = {
       }
     })();
 
+    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;
@@ -1858,8 +1998,14 @@ Element.Methods = {
       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();
@@ -1876,7 +2022,7 @@ Element.Methods = {
         return element;
       }
 
-      if (SELECT_ELEMENT_INNERHTML_BUGGY || TABLE_ELEMENT_INNERHTML_BUGGY) {
+      if (ANY_INNERHTML_BUGGY) {
         if (tagName in Element._insertionTranslations.tags) {
           while (element.firstChild) {
             element.removeChild(element.firstChild);
@@ -1885,6 +2031,12 @@ Element.Methods = {
             .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();
@@ -1967,19 +2119,26 @@ Element.Methods = {
     element = $(element);
     var result = '<' + element.tagName.toLowerCase();
     $H({'id': 'id', 'className': 'class'}).each(function(pair) {
-      var property = pair.first(), attribute = pair.last();
-      var value = (element[property] || '').toString();
+      var property = pair.first(),
+          attribute = pair.last(),
+          value = (element[property] || '').toString();
       if (value) result += ' ' + attribute + '=' + value.inspect(true);
     });
     return result + '>';
   },
 
-  recursivelyCollect: function(element, property) {
+  recursivelyCollect: function(element, property, maximumLength) {
     element = $(element);
+    maximumLength = maximumLength || -1;
     var elements = [];
-    while (element = element[property])
+
+    while (element = element[property]) {
       if (element.nodeType == 1)
         elements.push(Element.extend(element));
+      if (elements.length == maximumLength)
+        break;
+    }
+
     return elements;
   },
 
@@ -1998,13 +2157,17 @@ Element.Methods = {
   },
 
   immediateDescendants: function(element) {
-    if (!(element = $(element).firstChild)) return [];
-    while (element && element.nodeType != 1) element = element.nextSibling;
-    if (element) return [element].concat($(element).nextSiblings());
-    return [];
+    var results = [], child = $(element).firstChild;
+    while (child) {
+      if (child.nodeType === 1) {
+        results.push(Element.extend(child));
+      }
+      child = child.nextSibling;
+    }
+    return results;
   },
 
-  previousSiblings: function(element) {
+  previousSiblings: function(element, maximumLength) {
     return Element.recursivelyCollect(element, 'previousSibling');
   },
 
@@ -2019,9 +2182,10 @@ Element.Methods = {
   },
 
   match: function(element, selector) {
+    element = $(element);
     if (Object.isString(selector))
-      selector = new Selector(selector);
-    return selector.match($(element));
+      return Prototype.Selector.match(element, selector);
+    return selector.match(element);
   },
 
   up: function(element, expression, index) {
@@ -2029,7 +2193,7 @@ Element.Methods = {
     if (arguments.length == 1) return $(element.parentNode);
     var ancestors = Element.ancestors(element);
     return Object.isNumber(expression) ? ancestors[expression] :
-      Selector.findElement(ancestors, expression, index);
+      Prototype.Selector.find(ancestors, expression, index);
   },
 
   down: function(element, expression, index) {
@@ -2041,29 +2205,40 @@ Element.Methods = {
 
   previous: function(element, expression, index) {
     element = $(element);
-    if (arguments.length == 1) return $(Selector.handlers.previousElementSibling(element));
-    var previousSiblings = Element.previousSiblings(element);
-    return Object.isNumber(expression) ? previousSiblings[expression] :
-      Selector.findElement(previousSiblings, expression, index);
+    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 (arguments.length == 1) return $(Selector.handlers.nextElementSibling(element));
-    var nextSiblings = Element.nextSiblings(element);
-    return Object.isNumber(expression) ? nextSiblings[expression] :
-      Selector.findElement(nextSiblings, expression, index);
+    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) {
-    var args = Array.prototype.slice.call(arguments, 1);
-    return Selector.findChildElements(element, args);
+    element = $(element);
+    var expressions = Array.prototype.slice.call(arguments, 1).join(', ');
+    return Prototype.Selector.select(expressions, element);
   },
 
   adjacent: function(element) {
-    var args = Array.prototype.slice.call(arguments, 1);
-    return Selector.findChildElements(element.parentNode, args).without(element);
+    element = $(element);
+    var expressions = Array.prototype.slice.call(arguments, 1).join(', ');
+    return Prototype.Selector.select(expressions, element.parentNode).without(element);
   },
 
   identify: function(element) {
@@ -2227,28 +2402,6 @@ Element.Methods = {
     return element;
   },
 
-  getDimensions: function(element) {
-    element = $(element);
-    var display = Element.getStyle(element, 'display');
-    if (display != 'none' && display != null) // Safari bug
-      return {width: element.offsetWidth, height: element.offsetHeight};
-
-    var els = element.style;
-    var originalVisibility = els.visibility;
-    var originalPosition = els.position;
-    var originalDisplay = els.display;
-    els.visibility = 'hidden';
-    if (originalPosition != 'fixed') // Switching fixed to absolute causes issues in Safari
-      els.position = 'absolute';
-    els.display = 'block';
-    var originalWidth = element.clientWidth;
-    var originalHeight = element.clientHeight;
-    els.display = originalDisplay;
-    els.position = originalPosition;
-    els.visibility = originalVisibility;
-    return {width: originalWidth, height: originalHeight};
-  },
-
   makePositioned: function(element) {
     element = $(element);
     var pos = Element.getStyle(element, 'position');
@@ -2293,114 +2446,6 @@ Element.Methods = {
     return element;
   },
 
-  cumulativeOffset: function(element) {
-    var valueT = 0, valueL = 0;
-    do {
-      valueT += element.offsetTop  || 0;
-      valueL += element.offsetLeft || 0;
-      element = element.offsetParent;
-    } while (element);
-    return Element._returnOffset(valueL, valueT);
-  },
-
-  positionedOffset: function(element) {
-    var valueT = 0, valueL = 0;
-    do {
-      valueT += element.offsetTop  || 0;
-      valueL += element.offsetLeft || 0;
-      element = element.offsetParent;
-      if (element) {
-        if (element.tagName.toUpperCase() == 'BODY') break;
-        var p = Element.getStyle(element, 'position');
-        if (p !== 'static') break;
-      }
-    } while (element);
-    return Element._returnOffset(valueL, valueT);
-  },
-
-  absolutize: function(element) {
-    element = $(element);
-    if (Element.getStyle(element, 'position') == 'absolute') return element;
-
-    var offsets = Element.positionedOffset(element);
-    var top     = offsets[1];
-    var left    = offsets[0];
-    var width   = element.clientWidth;
-    var height  = element.clientHeight;
-
-    element._originalLeft   = left - parseFloat(element.style.left  || 0);
-    element._originalTop    = top  - parseFloat(element.style.top || 0);
-    element._originalWidth  = element.style.width;
-    element._originalHeight = element.style.height;
-
-    element.style.position = 'absolute';
-    element.style.top    = top + 'px';
-    element.style.left   = left + 'px';
-    element.style.width  = width + 'px';
-    element.style.height = height + 'px';
-    return element;
-  },
-
-  relativize: function(element) {
-    element = $(element);
-    if (Element.getStyle(element, 'position') == 'relative') return element;
-
-    element.style.position = 'relative';
-    var top  = parseFloat(element.style.top  || 0) - (element._originalTop || 0);
-    var left = parseFloat(element.style.left || 0) - (element._originalLeft || 0);
-
-    element.style.top    = top + 'px';
-    element.style.left   = left + 'px';
-    element.style.height = element._originalHeight;
-    element.style.width  = element._originalWidth;
-    return element;
-  },
-
-  cumulativeScrollOffset: function(element) {
-    var valueT = 0, valueL = 0;
-    do {
-      valueT += element.scrollTop  || 0;
-      valueL += element.scrollLeft || 0;
-      element = element.parentNode;
-    } while (element);
-    return Element._returnOffset(valueL, valueT);
-  },
-
-  getOffsetParent: function(element) {
-    if (element.offsetParent) return $(element.offsetParent);
-    if (element == document.body) return $(element);
-
-    while ((element = element.parentNode) && element != document.body)
-      if (Element.getStyle(element, 'position') != 'static')
-        return $(element);
-
-    return $(document.body);
-  },
-
-  viewportOffset: function(forElement) {
-    var valueT = 0, valueL = 0;
-
-    var element = forElement;
-    do {
-      valueT += element.offsetTop  || 0;
-      valueL += element.offsetLeft || 0;
-
-      if (element.offsetParent == document.body &&
-        Element.getStyle(element, 'position') == 'absolute') break;
-
-    } while (element = element.offsetParent);
-
-    element = forElement;
-    do {
-      if (!Prototype.Browser.Opera || (element.tagName && (element.tagName.toUpperCase() == 'BODY'))) {
-        valueT -= element.scrollTop  || 0;
-        valueL -= element.scrollLeft || 0;
-      }
-    } while (element = element.parentNode);
-
-    return Element._returnOffset(valueL, valueT);
-  },
-
   clonePosition: function(element, source) {
     var options = Object.extend({
       setLeft:    true,
@@ -2412,11 +2457,10 @@ Element.Methods = {
     }, arguments[2] || { });
 
     source = $(source);
-    var p = Element.viewportOffset(source);
+    var p = Element.viewportOffset(source), delta = [0, 0], parent = null;
 
     element = $(element);
-    var delta = [0, 0];
-    var parent = null;
+
     if (Element.getStyle(element, 'position') == 'absolute') {
       parent = Element.getOffsetParent(element);
       delta = Element.viewportOffset(parent);
@@ -2455,8 +2499,6 @@ if (Prototype.Browser.Opera) {
   Element.Methods.getStyle = Element.Methods.getStyle.wrap(
     function(proceed, element, style) {
       switch (style) {
-        case 'left': case 'top': case 'right': case 'bottom':
-          if (proceed(element, 'position') === 'static') return null;
         case 'height': case 'width':
           if (!Element.visible(element)) return null;
 
@@ -2492,47 +2534,6 @@ if (Prototype.Browser.Opera) {
 }
 
 else if (Prototype.Browser.IE) {
-  Element.Methods.getOffsetParent = Element.Methods.getOffsetParent.wrap(
-    function(proceed, element) {
-      element = $(element);
-      try { element.offsetParent }
-      catch(e) { 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;
-    }
-  );
-
-  $w('positionedOffset viewportOffset').each(function(method) {
-    Element.Methods[method] = Element.Methods[method].wrap(
-      function(proceed, element) {
-        element = $(element);
-        try { element.offsetParent }
-        catch(e) { return Element._returnOffset(0,0) }
-        var position = element.getStyle('position');
-        if (position !== 'static') return proceed(element);
-        var offsetParent = element.getOffsetParent();
-        if (offsetParent && offsetParent.getStyle('position') === 'fixed')
-          offsetParent.setStyle({ zoom: 1 });
-        element.setStyle({ position: 'relative' });
-        var value = proceed(element);
-        element.setStyle({ position: position });
-        return value;
-      }
-    );
-  });
-
-  Element.Methods.cumulativeOffset = Element.Methods.cumulativeOffset.wrap(
-    function(proceed, element) {
-      try { element.offsetParent }
-      catch(e) { return Element._returnOffset(0,0) }
-      return proceed(element);
-    }
-  );
-
   Element.Methods.getStyle = function(element, style) {
     element = $(element);
     style = (style == 'float' || style == 'cssFloat') ? 'styleFloat' : style.camelize();
@@ -2576,10 +2577,9 @@ else if (Prototype.Browser.IE) {
 
   Element._attributeTranslations = (function(){
 
-    var classProp = 'className';
-    var forProp = 'for';
-
-    var el = document.createElement('div');
+    var classProp = 'className',
+        forProp = 'for',
+        el = document.createElement('div');
 
     el.setAttribute(classProp, 'x');
 
@@ -2622,10 +2622,9 @@ else if (Prototype.Browser.IE) {
           },
           _getEv: (function(){
 
-            var el = document.createElement('div');
+            var el = document.createElement('div'), f;
             el.onclick = Prototype.emptyFunction;
             var value = el.getAttribute('onclick');
-            var f;
 
             if (String(value).indexOf('{') > -1) {
               f = function(element, attribute) {
@@ -2753,7 +2752,7 @@ else if (Prototype.Browser.WebKit) {
       (value < 0.00001) ? 0 : value;
 
     if (value == 1)
-      if(element.tagName.toUpperCase() == 'IMG' && element.width) {
+      if (element.tagName.toUpperCase() == 'IMG' && element.width) {
         element.width++; element.width--;
       } else try {
         var n = document.createTextNode(' ');
@@ -2763,20 +2762,6 @@ else if (Prototype.Browser.WebKit) {
 
     return element;
   };
-
-  Element.Methods.cumulativeOffset = function(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 Element._returnOffset(valueL, valueT);
-  };
 }
 
 if ('outerHTML' in document.documentElement) {
@@ -2793,8 +2778,8 @@ if ('outerHTML' in document.documentElement) {
     var parent = element.parentNode, tagName = parent.tagName.toUpperCase();
 
     if (Element._insertionTranslations.tags[tagName]) {
-      var nextSibling = element.next();
-      var fragments = Element._getContentFromAnonymousElement(tagName, content.stripScripts());
+      var nextSibling = element.next(),
+          fragments = Element._getContentFromAnonymousElement(tagName, content.stripScripts());
       parent.removeChild(element);
       if (nextSibling)
         fragments.each(function(node) { parent.insertBefore(node, nextSibling) });
@@ -2815,12 +2800,27 @@ Element._returnOffset = function(l, t) {
   return result;
 };
 
-Element._getContentFromAnonymousElement = function(tagName, html) {
-  var div = new Element('div'), t = Element._insertionTranslations.tags[tagName];
-  if (t) {
-    div.innerHTML = t[0] + html + t[1];
-    t[2].times(function() { div = div.firstChild });
-  } else div.innerHTML = html;
+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);
 };
 
@@ -2877,7 +2877,7 @@ Object.extend(Element, Element.Methods);
 
   div = null;
 
-})(document.createElement('div'))
+})(document.createElement('div'));
 
 Element.extend = (function() {
 
@@ -2885,8 +2885,8 @@ Element.extend = (function() {
     if (typeof window.Element != 'undefined') {
       var proto = window.Element.prototype;
       if (proto) {
-        var id = '_' + (Math.random()+'').slice(2);
-        var el = document.createElement(tagName);
+        var id = '_' + (Math.random()+'').slice(2),
+            el = document.createElement(tagName);
         proto[id] = 'x';
         var isBuggy = (el[id] !== 'x');
         delete proto[id];
@@ -2953,10 +2953,14 @@ Element.extend = (function() {
   return extend;
 })();
 
-Element.hasAttribute = function(element, attribute) {
-  if (element.hasAttribute) return element.hasAttribute(attribute);
-  return Element.Methods.Simulated.hasAttribute(element, attribute);
-};
+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;
@@ -2968,7 +2972,8 @@ Element.addMethods = function(methods) {
       "FORM":     Object.clone(Form.Methods),
       "INPUT":    Object.clone(Form.Element.Methods),
       "SELECT":   Object.clone(Form.Element.Methods),
-      "TEXTAREA": Object.clone(Form.Element.Methods)
+      "TEXTAREA": Object.clone(Form.Element.Methods),
+      "BUTTON":   Object.clone(Form.Element.Methods)
     });
   }
 
@@ -3020,8 +3025,9 @@ Element.addMethods = function(methods) {
     klass = 'HTML' + tagName.capitalize() + 'Element';
     if (window[klass]) return window[klass];
 
-    var element = document.createElement(tagName);
-    var proto = element['__proto__'] || element.constructor.prototype;
+    var element = document.createElement(tagName),
+        proto = element['__proto__'] || element.constructor.prototype;
+
     element = null;
     return proto;
   }
@@ -3104,8 +3110,8 @@ Element.addMethods({
       uid = 0;
     } else {
       if (typeof element._prototypeUID === "undefined")
-        element._prototypeUID = [Element.Storage.UID++];
-      uid = element._prototypeUID[0];
+        element._prototypeUID = Element.Storage.UID++;
+      uid = element._prototypeUID;
     }
 
     if (!Element.Storage[uid])
@@ -3150,769 +3156,1808 @@ Element.addMethods({
       }
     }
     return Element.extend(clone);
-  }
-});
-/* Portions of the Selector class are derived from Jack Slocum's DomQuery,
- * part of YUI-Ext version 0.40, distributed under the terms of an MIT-style
- * license.  Please see http://www.yui-ext.com/ for more information. */
-
-var Selector = Class.create({
-  initialize: function(expression) {
-    this.expression = expression.strip();
-
-    if (this.shouldUseSelectorsAPI()) {
-      this.mode = 'selectorsAPI';
-    } else if (this.shouldUseXPath()) {
-      this.mode = 'xpath';
-      this.compileXPathMatcher();
-    } else {
-      this.mode = "normal";
-      this.compileMatcher();
-    }
-
   },
 
-  shouldUseXPath: (function() {
-
-    var IS_DESCENDANT_SELECTOR_BUGGY = (function(){
-      var isBuggy = false;
-      if (document.evaluate && window.XPathResult) {
-        var el = document.createElement('div');
-        el.innerHTML = '<ul><li></li></ul><div><ul><li></li></ul></div>';
-
-        var xpath = ".//*[local-name()='ul' or local-name()='UL']" +
-          "//*[local-name()='li' or local-name()='LI']";
-
-        var result = document.evaluate(xpath, el, null,
-          XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
+  purge: function(element) {
+    if (!(element = $(element))) return;
+    var purgeElement = Element._purgeElement;
 
-        isBuggy = (result.snapshotLength !== 2);
-        el = null;
-      }
-      return isBuggy;
-    })();
+    purgeElement(element);
 
-    return function() {
-      if (!Prototype.BrowserFeatures.XPath) return false;
+    var descendants = element.getElementsByTagName('*'),
+     i = descendants.length;
 
-      var e = this.expression;
+    while (i--) purgeElement(descendants[i]);
 
-      if (Prototype.Browser.WebKit &&
-       (e.include("-of-type") || e.include(":empty")))
-        return false;
+    return null;
+  }
+});
 
-      if ((/(\[[\w-]*?:|:checked)/).test(e))
-        return false;
+(function() {
 
-      if (IS_DESCENDANT_SELECTOR_BUGGY) return false;
+  function toDecimal(pctString) {
+    var match = pctString.match(/^(\d+)%?$/i);
+    if (!match) return null;
+    return (Number(match[1]) / 100);
+  }
 
-      return true;
+  function getPixelValue(value, property, context) {
+    var element = null;
+    if (Object.isElement(value)) {
+      element = value;
+      value = element.getStyle(property);
     }
 
-  })(),
+    if (value === null) {
+      return null;
+    }
 
-  shouldUseSelectorsAPI: function() {
-    if (!Prototype.BrowserFeatures.SelectorsAPI) return false;
+    if ((/^(?:-)?\d+(\.\d+)?(px)?$/i).test(value)) {
+      return window.parseFloat(value);
+    }
 
-    if (Selector.CASE_INSENSITIVE_CLASS_NAMES) return false;
+    var isPercentage = value.include('%'), isViewport = (context === document.viewport);
 
-    if (!Selector._div) Selector._div = new Element('div');
+    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;
 
-    try {
-      Selector._div.querySelector(this.expression);
-    } catch(e) {
-      return false;
+      return value;
     }
 
-    return true;
-  },
-
-  compileMatcher: function() {
-    var e = this.expression, ps = Selector.patterns, h = Selector.handlers,
-        c = Selector.criteria, le, p, m, len = ps.length, name;
+    if (element && isPercentage) {
+      context = context || element.parentNode;
+      var decimal = toDecimal(value);
+      var whole = null;
+      var position = element.getStyle('position');
 
-    if (Selector._cache[e]) {
-      this.matcher = Selector._cache[e];
-      return;
-    }
+      var isHorizontal = property.include('left') || property.include('right') ||
+       property.include('width');
 
-    this.matcher = ["this.matcher = function(root) {",
-                    "var r = root, h = Selector.handlers, c = false, n;"];
+      var isVertical =  property.include('top') || property.include('bottom') ||
+        property.include('height');
 
-    while (e && le != e && (/\S/).test(e)) {
-      le = e;
-      for (var i = 0; i<len; i++) {
-        p = ps[i].re;
-        name = ps[i].name;
-        if (m = e.match(p)) {
-          this.matcher.push(Object.isFunction(c[name]) ? c[name](m) :
-            new Template(c[name]).evaluate(m));
-          e = e.replace(m[0], '');
-          break;
+      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');
         }
       }
-    }
 
-    this.matcher.push("return h.unique(n);\n}");
-    eval(this.matcher.join('\n'));
-    Selector._cache[this.expression] = this.matcher;
-  },
+      return (whole === null) ? 0 : whole * decimal;
+    }
 
-  compileXPathMatcher: function() {
-    var e = this.expression, ps = Selector.patterns,
-        x = Selector.xpath, le, m, len = ps.length, name;
+    return 0;
+  }
 
-    if (Selector._cache[e]) {
-      this.xpath = Selector._cache[e]; return;
+  function toCSSPixels(number) {
+    if (Object.isString(number) && number.endsWith('px')) {
+      return number;
     }
+    return number + 'px';
+  }
 
-    this.matcher = ['.//*'];
-    while (e && le != e && (/\S/).test(e)) {
-      le = e;
-      for (var i = 0; i<len; i++) {
-        name = ps[i].name;
-        if (m = e.match(ps[i].re)) {
-          this.matcher.push(Object.isFunction(x[name]) ? x[name](m) :
-            new Template(x[name]).evaluate(m));
-          e = e.replace(m[0], '');
-          break;
-        }
+  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;
+  }
 
-    this.xpath = this.matcher.join('');
-    Selector._cache[this.expression] = this.xpath;
-  },
+  var hasLayout = Prototype.K;
+  if ('currentStyle' in document.documentElement) {
+    hasLayout = function(element) {
+      if (!element.currentStyle.hasLayout) {
+        element.style.zoom = 1;
+      }
+      return element;
+    };
+  }
 
-  findElements: function(root) {
-    root = root || document;
-    var e = this.expression, results;
+  function cssNameFor(key) {
+    if (key.include('border')) key = key + '-width';
+    return key.camelize();
+  }
 
-    switch (this.mode) {
-      case 'selectorsAPI':
-        if (root !== document) {
-          var oldId = root.id, id = $(root).identify();
-          id = id.replace(/([\.:])/g, "\\$1");
-          e = "#" + id + " " + e;
-        }
+  Element.Layout = Class.create(Hash, {
+    initialize: function($super, element, preCompute) {
+      $super();
+      this.element = $(element);
 
-        results = $A(root.querySelectorAll(e)).map(Element.extend);
-        root.id = oldId;
+      Element.Layout.PROPERTIES.each( function(property) {
+        this._set(property, null);
+      }, this);
 
-        return results;
-      case 'xpath':
-        return document._getElementsByXPath(this.xpath, root);
-      default:
-       return this.matcher(root);
-    }
-  },
+      if (preCompute) {
+        this._preComputing = true;
+        this._begin();
+        Element.Layout.PROPERTIES.each( this._compute, this );
+        this._end();
+        this._preComputing = false;
+      }
+    },
 
-  match: function(element) {
-    this.tokens = [];
-
-    var e = this.expression, ps = Selector.patterns, as = Selector.assertions;
-    var le, p, m, len = ps.length, name;
-
-    while (e && le !== e && (/\S/).test(e)) {
-      le = e;
-      for (var i = 0; i<len; i++) {
-        p = ps[i].re;
-        name = ps[i].name;
-        if (m = e.match(p)) {
-          if (as[name]) {
-            this.tokens.push([name, Object.clone(m)]);
-            e = e.replace(m[0], '');
-          } else {
-            return this.findElements(document).include(element);
-          }
-        }
+    _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 match = true, name, matches;
-    for (var i = 0, token; token = this.tokens[i]; i++) {
-      name = token[0], matches = token[1];
-      if (!Selector.assertions[name](element, matches)) {
-        match = false; break;
+      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');
       }
-    }
 
-    return match;
-  },
+      var context = (position === 'fixed') ? document.viewport :
+       element.parentNode;
 
-  toString: function() {
-    return this.expression;
-  },
+      element.setStyle({
+        position:   'absolute',
+        visibility: 'hidden',
+        display:    'block'
+      });
 
-  inspect: function() {
-    return "#<Selector:" + this.expression.inspect() + ">";
-  }
-});
+      var positionedWidth = element.getStyle('width');
 
-if (Prototype.BrowserFeatures.SelectorsAPI &&
- document.compatMode === 'BackCompat') {
-  Selector.CASE_INSENSITIVE_CLASS_NAMES = (function(){
-    var div = document.createElement('div'),
-     span = document.createElement('span');
-
-    div.id = "prototype_test_id";
-    span.className = 'Test';
-    div.appendChild(span);
-    var isIgnored = (div.querySelector('#prototype_test_id .test') !== null);
-    div = span = null;
-    return isIgnored;
-  })();
-}
+      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');
+      }
 
-Object.extend(Selector, {
-  _cache: { },
-
-  xpath: {
-    descendant:   "//*",
-    child:        "/*",
-    adjacent:     "/following-sibling::*[1]",
-    laterSibling: '/following-sibling::*',
-    tagName:      function(m) {
-      if (m[1] == '*') return '';
-      return "[local-name()='" + m[1].toLowerCase() +
-             "' or local-name()='" + m[1].toUpperCase() + "']";
+      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;
     },
-    className:    "[contains(concat(' ', @class, ' '), ' #{1} ')]",
-    id:           "[@id='#{1}']",
-    attrPresence: function(m) {
-      m[1] = m[1].toLowerCase();
-      return new Template("[@#{1}]").evaluate(m);
+
+    _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));
     },
-    attr: function(m) {
-      m[1] = m[1].toLowerCase();
-      m[3] = m[5] || m[6];
-      return new Template(Selector.xpath.operators[m[2]]).evaluate(m);
+
+    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;
     },
-    pseudo: function(m) {
-      var h = Selector.xpath.pseudos[m[1]];
-      if (!h) return '';
-      if (Object.isFunction(h)) return h(m);
-      return new Template(Selector.xpath.pseudos[m[1]]).evaluate(m);
+
+    toHash: function() {
+      var obj = this.toObject.apply(this, arguments);
+      return new Hash(obj);
     },
-    operators: {
-      '=':  "[@#{1}='#{3}']",
-      '!=': "[@#{1}!='#{3}']",
-      '^=': "[starts-with(@#{1}, '#{3}')]",
-      '$=': "[substring(@#{1}, (string-length(@#{1}) - string-length('#{3}') + 1))='#{3}']",
-      '*=': "[contains(@#{1}, '#{3}')]",
-      '~=': "[contains(concat(' ', @#{1}, ' '), ' #{3} ')]",
-      '|=': "[contains(concat('-', @#{1}, '-'), '-#{3}-')]"
+
+    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;
     },
-    pseudos: {
-      'first-child': '[not(preceding-sibling::*)]',
-      'last-child':  '[not(following-sibling::*)]',
-      'only-child':  '[not(preceding-sibling::* or following-sibling::*)]',
-      'empty':       "[count(*) = 0 and (count(text()) = 0)]",
-      'checked':     "[@checked]",
-      'disabled':    "[(@disabled) and (@type!='hidden')]",
-      'enabled':     "[not(@disabled) and (@type!='hidden')]",
-      'not': function(m) {
-        var e = m[6], p = Selector.patterns,
-            x = Selector.xpath, le, v, len = p.length, name;
-
-        var exclusion = [];
-        while (e && le != e && (/\S/).test(e)) {
-          le = e;
-          for (var i = 0; i<len; i++) {
-            name = p[i].name
-            if (m = e.match(p[i].re)) {
-              v = Object.isFunction(x[name]) ? x[name](m) : new Template(x[name]).evaluate(m);
-              exclusion.push("(" + v.substring(1, v.length - 1) + ")");
-              e = e.replace(m[0], '');
-              break;
-            }
-          }
+
+    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;
         }
-        return "[not(" + exclusion.join(" and ") + ")]";
+
+        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;
       },
-      'nth-child':      function(m) {
-        return Selector.xpath.pseudos.nth("(count(./preceding-sibling::*) + 1) ", m);
+
+      '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;
       },
-      'nth-last-child': function(m) {
-        return Selector.xpath.pseudos.nth("(count(./following-sibling::*) + 1) ", m);
+
+      '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;
       },
-      'nth-of-type':    function(m) {
-        return Selector.xpath.pseudos.nth("position() ", m);
+
+      'top': function(element) {
+        var offset = element.positionedOffset();
+        return offset.top;
       },
-      'nth-last-of-type': function(m) {
-        return Selector.xpath.pseudos.nth("(last() + 1 - position()) ", m);
+
+      '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;
       },
-      'first-of-type':  function(m) {
-        m[6] = "1"; return Selector.xpath.pseudos['nth-of-type'](m);
+
+      'left': function(element) {
+        var offset = element.positionedOffset();
+        return offset.left;
       },
-      'last-of-type':   function(m) {
-        m[6] = "1"; return Selector.xpath.pseudos['nth-last-of-type'](m);
+
+      '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;
       },
-      'only-of-type':   function(m) {
-        var p = Selector.xpath.pseudos; return p['first-of-type'](m) + p['last-of-type'](m);
+
+      'padding-top': function(element) {
+        return getPixelValue(element, 'paddingTop');
       },
-      nth: function(fragment, m) {
-        var mm, formula = m[6], predicate;
-        if (formula == 'even') formula = '2n+0';
-        if (formula == 'odd')  formula = '2n+1';
-        if (mm = formula.match(/^(\d+)$/)) // digit only
-          return '[' + fragment + "= " + mm[1] + ']';
-        if (mm = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b
-          if (mm[1] == "-") mm[1] = -1;
-          var a = mm[1] ? Number(mm[1]) : 1;
-          var b = mm[2] ? Number(mm[2]) : 0;
-          predicate = "[((#{fragment} - #{b}) mod #{a} = 0) and " +
-          "((#{fragment} - #{b}) div #{a} >= 0)]";
-          return new Template(predicate).evaluate({
-            fragment: fragment, a: a, b: b });
-        }
+
+      '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
+  };
+})();
+/*!
+ * 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;
 
-  criteria: {
-    tagName:      'n = h.tagName(n, r, "#{1}", c);      c = false;',
-    className:    'n = h.className(n, r, "#{1}", c);    c = false;',
-    id:           'n = h.id(n, r, "#{1}", c);           c = false;',
-    attrPresence: 'n = h.attrPresence(n, r, "#{1}", c); c = false;',
-    attr: function(m) {
-      m[3] = (m[5] || m[6]);
-      return new Template('n = h.attr(n, r, "#{1}", "#{3}", "#{2}", c); c = false;').evaluate(m);
-    },
-    pseudo: function(m) {
-      if (m[6]) m[6] = m[6].replace(/"/g, '\\"');
-      return new Template('n = h.pseudo(n, "#{1}", "#{6}", r, c); c = false;').evaluate(m);
-    },
-    descendant:   'c = "descendant";',
-    child:        'c = "child";',
-    adjacent:     'c = "adjacent";',
-    laterSibling: 'c = "laterSibling";'
-  },
+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 );
+}
 
-  patterns: [
-    { name: 'laterSibling', re: /^\s*~\s*/ },
-    { name: 'child',        re: /^\s*>\s*/ },
-    { name: 'adjacent',     re: /^\s*\+\s*/ },
-    { name: 'descendant',   re: /^\s/ },
-
-    { name: 'tagName',      re: /^\s*(\*|[\w\-]+)(\b|$)?/ },
-    { name: 'id',           re: /^#([\w\-\*]+)(\b|$)/ },
-    { name: 'className',    re: /^\.([\w\-\*]+)(\b|$)/ },
-    { name: 'pseudo',       re: /^:((first|last|nth|nth-last|only)(-child|-of-type)|empty|checked|(en|dis)abled|not)(\((.*?)\))?(\b|$|(?=\s|[:+~>]))/ },
-    { name: 'attrPresence', re: /^\[((?:[\w-]+:)?[\w-]+)\]/ },
-    { name: 'attr',         re: /\[((?:[\w-]*:)?[\w-]+)\s*(?:([!^$*~|]?=)\s*((['"])([^\4]*?)\4|([^'"][^\]]*?)))?\]/ }
-  ],
-
-  assertions: {
-    tagName: function(element, matches) {
-      return matches[1].toUpperCase() == element.tagName.toUpperCase();
-    },
+var makeArray = function(array, results) {
+       array = Array.prototype.slice.call( array, 0 );
 
-    className: function(element, matches) {
-      return Element.hasClassName(element, matches[1]);
-    },
+       if ( results ) {
+               results.push.apply( results, array );
+               return results;
+       }
 
-    id: function(element, matches) {
-      return element.id === matches[1];
-    },
+       return array;
+};
 
-    attrPresence: function(element, matches) {
-      return Element.hasAttribute(element, matches[1]);
-    },
+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;
+       };
+}
 
-    attr: function(element, matches) {
-      var nodeValue = Element.readAttribute(element, matches[1]);
-      return nodeValue && Selector.operators[matches[2]](nodeValue, matches[5] || matches[6]);
-    }
-  },
+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;
+       };
+}
 
-  handlers: {
-    concat: function(a, b) {
-      for (var i = 0, node; node = b[i]; i++)
-        a.push(node);
-      return a;
-    },
+(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
+})();
 
-    mark: function(nodes) {
-      var _true = Prototype.emptyFunction;
-      for (var i = 0, node; node = nodes[i]; i++)
-        node._countedByPrototype = _true;
-      return nodes;
-    },
+(function(){
 
-    unmark: (function(){
+       var div = document.createElement("div");
+       div.appendChild( document.createComment("") );
 
-      var PROPERTIES_ATTRIBUTES_MAP = (function(){
-        var el = document.createElement('div'),
-            isBuggy = false,
-            propName = '_countedByPrototype',
-            value = 'x'
-        el[propName] = value;
-        isBuggy = (el.getAttribute(propName) === value);
-        el = null;
-        return isBuggy;
-      })();
-
-      return PROPERTIES_ATTRIBUTES_MAP ?
-        function(nodes) {
-          for (var i = 0, node; node = nodes[i]; i++)
-            node.removeAttribute('_countedByPrototype');
-          return nodes;
-        } :
-        function(nodes) {
-          for (var i = 0, node; node = nodes[i]; i++)
-            node._countedByPrototype = void 0;
-          return nodes;
-        }
-    })(),
+       if ( div.getElementsByTagName("*").length > 0 ) {
+               Expr.find.TAG = function(match, context){
+                       var results = context.getElementsByTagName(match[1]);
 
-    index: function(parentNode, reverse, ofType) {
-      parentNode._countedByPrototype = Prototype.emptyFunction;
-      if (reverse) {
-        for (var nodes = parentNode.childNodes, i = nodes.length - 1, j = 1; i >= 0; i--) {
-          var node = nodes[i];
-          if (node.nodeType == 1 && (!ofType || node._countedByPrototype)) node.nodeIndex = j++;
-        }
-      } else {
-        for (var i = 0, j = 1, nodes = parentNode.childNodes; node = nodes[i]; i++)
-          if (node.nodeType == 1 && (!ofType || node._countedByPrototype)) node.nodeIndex = j++;
-      }
-    },
+                       if ( match[1] === "*" ) {
+                               var tmp = [];
 
-    unique: function(nodes) {
-      if (nodes.length == 0) return nodes;
-      var results = [], n;
-      for (var i = 0, l = nodes.length; i < l; i++)
-        if (typeof (n = nodes[i])._countedByPrototype == 'undefined') {
-          n._countedByPrototype = Prototype.emptyFunction;
-          results.push(Element.extend(n));
-        }
-      return Selector.handlers.unmark(results);
-    },
+                               for ( var i = 0; results[i]; i++ ) {
+                                       if ( results[i].nodeType === 1 ) {
+                                               tmp.push( results[i] );
+                                       }
+                               }
 
-    descendant: function(nodes) {
-      var h = Selector.handlers;
-      for (var i = 0, results = [], node; node = nodes[i]; i++)
-        h.concat(results, node.getElementsByTagName('*'));
-      return results;
-    },
+                               results = tmp;
+                       }
 
-    child: function(nodes) {
-      var h = Selector.handlers;
-      for (var i = 0, results = [], node; node = nodes[i]; i++) {
-        for (var j = 0, child; child = node.childNodes[j]; j++)
-          if (child.nodeType == 1 && child.tagName != '!') results.push(child);
-      }
-      return results;
-    },
+                       return results;
+               };
+       }
 
-    adjacent: function(nodes) {
-      for (var i = 0, results = [], node; node = nodes[i]; i++) {
-        var next = this.nextElementSibling(node);
-        if (next) results.push(next);
-      }
-      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);
+               };
+       }
 
-    laterSibling: function(nodes) {
-      var h = Selector.handlers;
-      for (var i = 0, results = [], node; node = nodes[i]; i++)
-        h.concat(results, Element.nextSiblings(node));
-      return results;
-    },
+       div = null; // release memory in IE
+})();
 
-    nextElementSibling: function(node) {
-      while (node = node.nextSibling)
-        if (node.nodeType == 1) return node;
-      return null;
-    },
+if ( document.querySelectorAll ) (function(){
+       var oldSizzle = Sizzle, div = document.createElement("div");
+       div.innerHTML = "<p class='TEST'></p>";
 
-    previousElementSibling: function(node) {
-      while (node = node.previousSibling)
-        if (node.nodeType == 1) return node;
-      return null;
-    },
+       if ( div.querySelectorAll && div.querySelectorAll(".TEST").length === 0 ) {
+               return;
+       }
 
-    tagName: function(nodes, root, tagName, combinator) {
-      var uTagName = tagName.toUpperCase();
-      var results = [], h = Selector.handlers;
-      if (nodes) {
-        if (combinator) {
-          if (combinator == "descendant") {
-            for (var i = 0, node; node = nodes[i]; i++)
-              h.concat(results, node.getElementsByTagName(tagName));
-            return results;
-          } else nodes = this[combinator](nodes);
-          if (tagName == "*") return nodes;
-        }
-        for (var i = 0, node; node = nodes[i]; i++)
-          if (node.tagName.toUpperCase() === uTagName) results.push(node);
-        return results;
-      } else return root.getElementsByTagName(tagName);
-    },
+       Sizzle = function(query, context, extra, seed){
+               context = context || document;
 
-    id: function(nodes, root, id, combinator) {
-      var targetNode = $(id), h = Selector.handlers;
+               if ( !seed && context.nodeType === 9 && !isXML(context) ) {
+                       try {
+                               return makeArray( context.querySelectorAll(query), extra );
+                       } catch(e){}
+               }
 
-      if (root == document) {
-        if (!targetNode) return [];
-        if (!nodes) return [targetNode];
-      } else {
-        if (!root.sourceIndex || root.sourceIndex < 1) {
-          var nodes = root.getElementsByTagName('*');
-          for (var j = 0, node; node = nodes[j]; j++) {
-            if (node.id === id) return [node];
-          }
-        }
-      }
+               return oldSizzle(query, context, extra, seed);
+       };
 
-      if (nodes) {
-        if (combinator) {
-          if (combinator == 'child') {
-            for (var i = 0, node; node = nodes[i]; i++)
-              if (targetNode.parentNode == node) return [targetNode];
-          } else if (combinator == 'descendant') {
-            for (var i = 0, node; node = nodes[i]; i++)
-              if (Element.descendantOf(targetNode, node)) return [targetNode];
-          } else if (combinator == 'adjacent') {
-            for (var i = 0, node; node = nodes[i]; i++)
-              if (Selector.handlers.previousElementSibling(targetNode) == node)
-                return [targetNode];
-          } else nodes = h[combinator](nodes);
-        }
-        for (var i = 0, node; node = nodes[i]; i++)
-          if (node == targetNode) return [targetNode];
-        return [];
-      }
-      return (targetNode && Element.descendantOf(targetNode, root)) ? [targetNode] : [];
-    },
+       for ( var prop in oldSizzle ) {
+               Sizzle[ prop ] = oldSizzle[ prop ];
+       }
 
-    className: function(nodes, root, className, combinator) {
-      if (nodes && combinator) nodes = this[combinator](nodes);
-      return Selector.handlers.byClassName(nodes, root, className);
-    },
+       div = null; // release memory in IE
+})();
 
-    byClassName: function(nodes, root, className) {
-      if (!nodes) nodes = Selector.handlers.descendant([root]);
-      var needle = ' ' + className + ' ';
-      for (var i = 0, results = [], node, nodeClassName; node = nodes[i]; i++) {
-        nodeClassName = node.className;
-        if (nodeClassName.length == 0) continue;
-        if (nodeClassName == className || (' ' + nodeClassName + ' ').include(needle))
-          results.push(node);
-      }
-      return results;
-    },
+if ( document.getElementsByClassName && document.documentElement.getElementsByClassName ) (function(){
+       var div = document.createElement("div");
+       div.innerHTML = "<div class='test e'></div><div class='test'></div>";
 
-    attrPresence: function(nodes, root, attr, combinator) {
-      if (!nodes) nodes = root.getElementsByTagName("*");
-      if (nodes && combinator) nodes = this[combinator](nodes);
-      var results = [];
-      for (var i = 0, node; node = nodes[i]; i++)
-        if (Element.hasAttribute(node, attr)) results.push(node);
-      return results;
-    },
+       if ( div.getElementsByClassName("e").length === 0 )
+               return;
 
-    attr: function(nodes, root, attr, value, operator, combinator) {
-      if (!nodes) nodes = root.getElementsByTagName("*");
-      if (nodes && combinator) nodes = this[combinator](nodes);
-      var handler = Selector.operators[operator], results = [];
-      for (var i = 0, node; node = nodes[i]; i++) {
-        var nodeValue = Element.readAttribute(node, attr);
-        if (nodeValue === null) continue;
-        if (handler(nodeValue, value)) results.push(node);
-      }
-      return results;
-    },
+       div.lastChild.className = "e";
 
-    pseudo: function(nodes, name, value, root, combinator) {
-      if (nodes && combinator) nodes = this[combinator](nodes);
-      if (!nodes) nodes = root.getElementsByTagName("*");
-      return Selector.pseudos[name](nodes, value, root);
-    }
-  },
+       if ( div.getElementsByClassName("e").length === 1 )
+               return;
 
-  pseudos: {
-    'first-child': function(nodes, value, root) {
-      for (var i = 0, results = [], node; node = nodes[i]; i++) {
-        if (Selector.handlers.previousElementSibling(node)) continue;
-          results.push(node);
-      }
-      return results;
-    },
-    'last-child': function(nodes, value, root) {
-      for (var i = 0, results = [], node; node = nodes[i]; i++) {
-        if (Selector.handlers.nextElementSibling(node)) continue;
-          results.push(node);
-      }
-      return results;
-    },
-    'only-child': function(nodes, value, root) {
-      var h = Selector.handlers;
-      for (var i = 0, results = [], node; node = nodes[i]; i++)
-        if (!h.previousElementSibling(node) && !h.nextElementSibling(node))
-          results.push(node);
-      return results;
-    },
-    'nth-child':        function(nodes, formula, root) {
-      return Selector.pseudos.nth(nodes, formula, root);
-    },
-    'nth-last-child':   function(nodes, formula, root) {
-      return Selector.pseudos.nth(nodes, formula, root, true);
-    },
-    'nth-of-type':      function(nodes, formula, root) {
-      return Selector.pseudos.nth(nodes, formula, root, false, true);
-    },
-    'nth-last-of-type': function(nodes, formula, root) {
-      return Selector.pseudos.nth(nodes, formula, root, true, true);
-    },
-    'first-of-type':    function(nodes, formula, root) {
-      return Selector.pseudos.nth(nodes, "1", root, false, true);
-    },
-    'last-of-type':     function(nodes, formula, root) {
-      return Selector.pseudos.nth(nodes, "1", root, true, true);
-    },
-    'only-of-type':     function(nodes, formula, root) {
-      var p = Selector.pseudos;
-      return p['last-of-type'](p['first-of-type'](nodes, formula, root), formula, root);
-    },
+       Expr.order.splice(1, 0, "CLASS");
+       Expr.find.CLASS = function(match, context, isXML) {
+               if ( typeof context.getElementsByClassName !== "undefined" && !isXML ) {
+                       return context.getElementsByClassName(match[1]);
+               }
+       };
 
-    getIndices: function(a, b, total) {
-      if (a == 0) return b > 0 ? [b] : [];
-      return $R(1, total).inject([], function(memo, i) {
-        if (0 == (i - b) % a && (i - b) / a >= 0) memo.push(i);
-        return memo;
-      });
-    },
+       div = null; // release memory in IE
+})();
 
-    nth: function(nodes, formula, root, reverse, ofType) {
-      if (nodes.length == 0) return [];
-      if (formula == 'even') formula = '2n+0';
-      if (formula == 'odd')  formula = '2n+1';
-      var h = Selector.handlers, results = [], indexed = [], m;
-      h.mark(nodes);
-      for (var i = 0, node; node = nodes[i]; i++) {
-        if (!node.parentNode._countedByPrototype) {
-          h.index(node.parentNode, reverse, ofType);
-          indexed.push(node.parentNode);
-        }
-      }
-      if (formula.match(/^\d+$/)) { // just a number
-        formula = Number(formula);
-        for (var i = 0, node; node = nodes[i]; i++)
-          if (node.nodeIndex == formula) results.push(node);
-      } else if (m = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b
-        if (m[1] == "-") m[1] = -1;
-        var a = m[1] ? Number(m[1]) : 1;
-        var b = m[2] ? Number(m[2]) : 0;
-        var indices = Selector.pseudos.getIndices(a, b, nodes.length);
-        for (var i = 0, node, l = indices.length; node = nodes[i]; i++) {
-          for (var j = 0; j < l; j++)
-            if (node.nodeIndex == indices[j]) results.push(node);
-        }
-      }
-      h.unmark(nodes);
-      h.unmark(indexed);
-      return results;
-    },
+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;
+               }
+       }
+}
 
-    'empty': function(nodes, value, root) {
-      for (var i = 0, results = [], node; node = nodes[i]; i++) {
-        if (node.tagName == '!' || node.firstChild) continue;
-        results.push(node);
-      }
-      return results;
-    },
+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;
+               }
+       }
+}
 
-    'not': function(nodes, selector, root) {
-      var h = Selector.handlers, selectorType, m;
-      var exclusions = new Selector(selector).findElements(root);
-      h.mark(exclusions);
-      for (var i = 0, results = [], node; node = nodes[i]; i++)
-        if (!node._countedByPrototype) results.push(node);
-      h.unmark(exclusions);
-      return results;
-    },
+var contains = document.compareDocumentPosition ?  function(a, b){
+       return a.compareDocumentPosition(b) & 16;
+} : function(a, b){
+       return a !== b && (a.contains ? a.contains(b) : true);
+};
 
-    'enabled': function(nodes, value, root) {
-      for (var i = 0, results = [], node; node = nodes[i]; i++)
-        if (!node.disabled && (!node.type || node.type !== 'hidden'))
-          results.push(node);
-      return results;
-    },
+var isXML = function(elem){
+       return elem.nodeType === 9 && elem.documentElement.nodeName !== "HTML" ||
+               !!elem.ownerDocument && elem.ownerDocument.documentElement.nodeName !== "HTML";
+};
 
-    'disabled': function(nodes, value, root) {
-      for (var i = 0, results = [], node; node = nodes[i]; i++)
-        if (node.disabled) results.push(node);
-      return results;
-    },
+var posProcess = function(selector, context){
+       var tmpSet = [], later = "", match,
+               root = context.nodeType ? [context] : context;
 
-    'checked': function(nodes, value, root) {
-      for (var i = 0, results = [], node; node = nodes[i]; i++)
-        if (node.checked) results.push(node);
-      return results;
-    }
-  },
+       while ( (match = Expr.match.PSEUDO.exec( selector )) ) {
+               later += match[0];
+               selector = selector.replace( Expr.match.PSEUDO, "" );
+       }
 
-  operators: {
-    '=':  function(nv, v) { return nv == v; },
-    '!=': function(nv, v) { return nv != v; },
-    '^=': function(nv, v) { return nv == v || nv && nv.startsWith(v); },
-    '$=': function(nv, v) { return nv == v || nv && nv.endsWith(v); },
-    '*=': function(nv, v) { return nv == v || nv && nv.include(v); },
-    '~=': function(nv, v) { return (' ' + nv + ' ').include(' ' + v + ' '); },
-    '|=': function(nv, v) { return ('-' + (nv || "").toUpperCase() +
-     '-').include('-' + (v || "").toUpperCase() + '-'); }
-  },
+       selector = Expr.relative[selector] ? selector + "*" : selector;
 
-  split: function(expression) {
-    var expressions = [];
-    expression.scan(/(([\w#:.~>+()\s-]+|\*|\[.*?\])+)\s*(,|$)/, function(m) {
-      expressions.push(m[1].strip());
-    });
-    return expressions;
-  },
+       for ( var i = 0, l = root.length; i < l; i++ ) {
+               Sizzle( selector, root[i], tmpSet );
+       }
 
-  matchElements: function(elements, expression) {
-    var matches = $$(expression), h = Selector.handlers;
-    h.mark(matches);
-    for (var i = 0, results = [], element; element = elements[i]; i++)
-      if (element._countedByPrototype) results.push(element);
-    h.unmark(matches);
-    return results;
-  },
+       return Sizzle.filter( later, tmpSet );
+};
 
-  findElement: function(elements, expression, index) {
-    if (Object.isNumber(expression)) {
-      index = expression; expression = false;
-    }
-    return Selector.matchElements(elements, expression || '*')[index || 0];
-  },
 
-  findChildElements: function(element, expressions) {
-    expressions = Selector.split(expressions.join(','));
-    var results = [], h = Selector.handlers;
-    for (var i = 0, l = expressions.length, selector; i < l; i++) {
-      selector = new Selector(expressions[i].strip());
-      h.concat(results, selector.findElements(element));
-    }
-    return (l > 1) ? h.unique(results) : results;
+window.Sizzle = Sizzle;
+
+})();
+
+Prototype._original_property = window.Sizzle;
+
+;(function(engine) {
+  var extendElements = Prototype.Selector.extendElements;
+
+  function select(selector, scope) {
+    return extendElements(engine(selector, scope || document));
   }
-});
 
-if (Prototype.Browser.IE) {
-  Object.extend(Selector.handlers, {
-    concat: function(a, b) {
-      for (var i = 0, node; node = b[i]; i++)
-        if (node.tagName !== "!") a.push(node);
-      return a;
-    }
-  });
-}
+  function match(element, selector) {
+    return engine.matches(selector, [element]).length == 1;
+  }
 
-function $$() {
-  return Selector.findChildElements(document, $A(arguments));
-}
+  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) {
@@ -3924,24 +4969,34 @@ var 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;
+    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);
+      }
+    }
 
-    var data = elements.inject({ }, function(result, element) {
+    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)))) {
-          if (key in result) {
-            if (!Object.isArray(result[key])) result[key] = [result[key]];
-            result[key].push(value);
-          }
-          else result[key] = value;
+          result = accumulator(result, key, value);
         }
       }
       return result;
     });
-
-    return options.hash ? data : Object.toQueryString(data);
   }
 };
 
@@ -4008,7 +5063,8 @@ Form.Methods = {
 
   focusFirstElement: function(form) {
     form = $(form);
-    form.findFirstElement().activate();
+    var element = form.findFirstElement();
+    if (element) element.activate();
     return form;
   },
 
@@ -4115,67 +5171,77 @@ var $F = Form.Element.Methods.getValue;
 
 /*--------------------------------------------------------------------------*/
 
-Form.Element.Serializers = {
-  input: function(element, value) {
+Form.Element.Serializers = (function() {
+  function input(element, value) {
     switch (element.type.toLowerCase()) {
       case 'checkbox':
       case 'radio':
-        return Form.Element.Serializers.inputSelector(element, value);
+        return inputSelector(element, value);
       default:
-        return Form.Element.Serializers.textarea(element, value);
+        return valueSelector(element, value);
     }
-  },
+  }
 
-  inputSelector: function(element, value) {
-    if (Object.isUndefined(value)) return element.checked ? element.value : null;
+  function inputSelector(element, value) {
+    if (Object.isUndefined(value))
+      return element.checked ? element.value : null;
     else element.checked = !!value;
-  },
+  }
 
-  textarea: function(element, value) {
+  function valueSelector(element, value) {
     if (Object.isUndefined(value)) return element.value;
     else element.value = value;
-  },
+  }
 
-  select: function(element, value) {
+  function select(element, value) {
     if (Object.isUndefined(value))
-      return this[element.type == 'select-one' ?
-        'selectOne' : 'selectMany'](element);
-    else {
-      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;
-          }
+      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);
       }
+      else opt.selected = value.include(currentValue);
     }
-  },
+  }
 
-  selectOne: function(element) {
+  function selectOne(element) {
     var index = element.selectedIndex;
-    return index >= 0 ? this.optionValue(element.options[index]) : null;
-  },
+    return index >= 0 ? optionValue(element.options[index]) : null;
+  }
 
-  selectMany: function(element) {
+  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(this.optionValue(opt));
+      if (opt.selected) values.push(optionValue(opt));
     }
     return values;
-  },
+  }
 
-  optionValue: function(opt) {
-    return Element.extend(opt).hasAttribute('value') ? opt.value : opt.text;
+  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
+  };
+})();
 
 /*--------------------------------------------------------------------------*/
 
@@ -4286,24 +5352,53 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
   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;
-  if (Prototype.Browser.IE) {
-    var buttonMap = { 0: 1, 1: 4, 2: 2 };
-    _isButton = function(event, code) {
-      return event.button === buttonMap[code];
-    };
-  } else if (Prototype.Browser.WebKit) {
-    _isButton = function(event, code) {
-      switch (code) {
-        case 0: return event.which == 1 && !event.metaKey;
-        case 1: return event.which == 1 && event.metaKey;
-        default: return false;
+
+  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 = function(event, code) {
-      return event.which ? (event.which === code + 1) : (event.button === code);
-    };
+    _isButton = _isButtonForDOMEvents;
   }
 
   function isLeftClick(event)   { return _isButton(event, 0) }
@@ -4333,9 +5428,14 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
 
   function findElement(event, expression) {
     var element = Event.element(event);
+
     if (!expression) return element;
-    var elements = [element].concat(element.ancestors());
-    return Selector.findElement(elements, expression, 0);
+    while (element) {
+      if (Object.isElement(element) && Prototype.Selector.match(element, expression)) {
+        return Element.extend(element);
+      }
+      element = element.parentNode;
+    }
   }
 
   function pointer(event) {
@@ -4369,49 +5469,59 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
     event.stopped = true;
   }
 
+
   Event.Methods = {
-    isLeftClick: isLeftClick,
+    isLeftClick:   isLeftClick,
     isMiddleClick: isMiddleClick,
-    isRightClick: isRightClick,
+    isRightClick:  isRightClick,
 
-    element: element,
+    element:     element,
     findElement: findElement,
 
-    pointer: pointer,
+    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 (Prototype.Browser.IE) {
+  if (window.attachEvent) {
     function _relatedTarget(event) {
       var element;
       switch (event.type) {
-        case 'mouseover': element = event.fromElement; break;
-        case 'mouseout':  element = event.toElement;   break;
-        default: return null;
+        case 'mouseover':
+        case 'mouseenter':
+          element = event.fromElement;
+          break;
+        case 'mouseout':
+        case 'mouseleave':
+          element = event.toElement;
+          break;
+        default:
+          return null;
       }
       return Element.extend(element);
     }
 
-    Object.extend(methods, {
+    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 (event._extendedByPrototype) return event;
 
+      if (!isIELegacyEvent(event)) return event;
+
+      if (event._extendedByPrototype) return event;
       event._extendedByPrototype = Prototype.emptyFunction;
+
       var pointer = Event.pointer(event);
 
       Object.extend(event, {
@@ -4421,12 +5531,18 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
         pageY:  pointer.y
       });
 
-      return Object.extend(event, methods);
+      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);
-    Event.extend = Prototype.K;
   }
 
   function _createResponder(element, eventName, handler) {
@@ -4504,12 +5620,12 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
     window.addEventListener('unload', Prototype.emptyFunction, false);
 
 
-  var _getDOMEventName = Prototype.K;
+  var _getDOMEventName = Prototype.K,
+      translations = { mouseenter: "mouseover", mouseleave: "mouseout" };
 
   if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED) {
     _getDOMEventName = function(eventName) {
-      var translations = { mouseenter: "mouseover", mouseleave: "mouseout" };
-      return eventName in translations ? translations[eventName] : eventName;
+      return (translations[eventName] || eventName);
     };
   }
 
@@ -4525,7 +5641,7 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
         element.addEventListener("dataavailable", responder, false);
       else {
         element.attachEvent("ondataavailable", responder);
-        element.attachEvent("onfilterchange", responder);
+        element.attachEvent("onlosecapture", responder);
       }
     } else {
       var actualEventName = _getDOMEventName(eventName);
@@ -4543,46 +5659,44 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
     element = $(element);
 
     var registry = Element.retrieve(element, 'prototype_event_registry');
+    if (!registry) return element;
 
-    if (Object.isUndefined(registry)) return element;
-
-    if (eventName && !handler) {
-      var responders = registry.get(eventName);
-
-      if (Object.isUndefined(responders)) return element;
-
-      responders.each( function(r) {
-        Element.stopObserving(element, eventName, r.handler);
-      });
-      return element;
-    } else if (!eventName) {
+    if (!eventName) {
       registry.each( function(pair) {
-        var eventName = pair.key, responders = pair.value;
-
-        responders.each( function(r) {
-          Element.stopObserving(element, eventName, r.handler);
-        });
+        var eventName = pair.key;
+        stopObserving(element, eventName);
       });
       return element;
     }
 
     var responders = registry.get(eventName);
+    if (!responders) return element;
 
-    if (!responders) return;
+    if (!handler) {
+      responders.each(function(r) {
+        stopObserving(element, eventName, r.handler);
+      });
+      return element;
+    }
 
-    var responder = responders.find( function(r) { return r.handler === handler; });
+    var i = responders.length, responder;
+    while (i--) {
+      if (responders[i].handler === handler) {
+        responder = responders[i];
+        break;
+      }
+    }
     if (!responder) return element;
 
-    var actualEventName = _getDOMEventName(eventName);
-
     if (eventName.include(':')) {
       if (element.removeEventListener)
         element.removeEventListener("dataavailable", responder, false);
       else {
         element.detachEvent("ondataavailable", responder);
-        element.detachEvent("onfilterchange",  responder);
+        element.detachEvent("onlosecapture", responder);
       }
     } else {
+      var actualEventName = _getDOMEventName(eventName);
       if (element.removeEventListener)
         element.removeEventListener(actualEventName, responder, false);
       else
@@ -4606,10 +5720,10 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
     var event;
     if (document.createEvent) {
       event = document.createEvent('HTMLEvents');
-      event.initEvent('dataavailable', true, true);
+      event.initEvent('dataavailable', bubble, true);
     } else {
       event = document.createEventObject();
-      event.eventType = bubble ? 'ondataavailable' : 'onfilterchange';
+      event.eventType = bubble ? 'ondataavailable' : 'onlosecapture';
     }
 
     event.eventName = eventName;
@@ -4623,13 +5737,47 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
     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
+    stopObserving: stopObserving,
+    on:            on
   });
 
   Element.addMethods({
@@ -4637,7 +5785,9 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
 
     observe:       observe,
 
-    stopObserving: stopObserving
+    stopObserving: stopObserving,
+
+    on:            on
   });
 
   Object.extend(document, {
@@ -4647,6 +5797,8 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
 
     stopObserving: stopObserving.methodize(),
 
+    on:            on.methodize(),
+
     loaded:        false
   });
 
@@ -4694,8 +5846,8 @@ Form.EventObserver = Class.create(Abstract.EventObserver, {
   Event.observe(window, 'load', fireContentLoadedEvent);
 })();
 
-Element.addMethods();
 
+Element.addMethods();
 /*------------------------------- DEPRECATED -------------------------------*/
 
 Hash.toQueryString = Object.toQueryString;
@@ -4871,4 +6023,59 @@ Element.ClassNames.prototype = {
 
 Object.extend(Element.ClassNames.prototype, Enumerable);
 
-/*--------------------------------------------------------------------------*/
\ No newline at end of file
+/*--------------------------------------------------------------------------*/
+
+(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/sonar-server/src/main/webapp/javascripts/protovis-msie-shim.js b/sonar-server/src/main/webapp/javascripts/protovis-msie-shim.js
new file mode 100644 (file)
index 0000000..d89ce57
--- /dev/null
@@ -0,0 +1,6 @@
+/*
+ * Protovis MSIE VML shim
+ * (c) 2011 DataMarket (datamarket@datamarket.com)
+ * 
+ */
+pv.have_SVG=!!document.createElementNS&&!!document.createElementNS("http://www.w3.org/2000/svg","svg").createSVGRect,pv.have_VML=function(a,b,c){b=a.createElement("div"),b.innerHTML='<v:shape adj="1" />',c=b.firstChild,c.style.behavior="url(#default#VML)";return c?typeof c.adj==="object":!0}(document),!pv.have_SVG&&pv.have_VML&&function(){function B(a){var b=a;a=new A(b);for(var c=0,d=z.length;c<d;c++){var e=z[c];a[e]=b[e]}a.target||(a.target=a.srcElement||document),a.target.nodeType===3&&(a.target=a.target.parentNode),!a.relatedTarget&&a.fromElement&&(a.relatedTarget=a.fromElement===a.target?a.toElement:a.fromElement);if(a.pageX==null&&a.clientX!=null){var f=document.documentElement,g=document.body;a.pageX=a.clientX+(f&&f.scrollLeft||g&&g.scrollLeft||0)-(f&&f.clientLeft||g&&g.clientLeft||0),a.pageY=a.clientY+(f&&f.scrollTop||g&&g.scrollTop||0)-(f&&f.clientTop||g&&g.clientTop||0)}a.which==null&&(a.charCode!=null||a.keyCode!=null)&&(a.which=a.charCode!=null?a.charCode:a.keyCode),!a.metaKey&&a.ctrlKey&&(a.metaKey=a.ctrlKey),!a.which&&a.button!==undefined&&(a.which=a.button&1?1:a.button&2?3:a.button&4?2:0);return a}function A(a){if(a&&a.type){this.originalEvent=a,this.type=a.type,this.isDefaultPrevented=y;if(a.defaultPrevented||a.returnValue===!1||a.getPreventDefault&&a.getPreventDefault())this.isDefaultPrevented=x}else this.type=a;this.timeStamp=new Date*1}function m(a,b){var c=a.getElementsByTagName("stroke")[0];c||(c=j("stroke"),a.appendChild(c)),!b.stroke||b.stroke==="none"?(c.on="false",c.weight="0"):(c.on="true",c.weight=parseFloat(b["stroke-width"]||"1")/1.25,c.color=f(b.stroke)||"black",c.opacity=parseFloat(b["stroke-opacity"]||"1")||"1",c.joinstyle=v[b["stroke-linejoin"]]||"miter")}function l(a,b){var c=a.getElementsByTagName("fill")[0];c||(c=j("fill"),a.appendChild(c)),!b.fill||b.fill==="none"?c.on=!1:(c.on="true",c.color=f(b.fill),c.opacity=parseFloat(b["fill-opacity"]||"1")||"1")}function k(a,b){var c=b||{};c.translate_x=0,c.translate_y=0;if(a.transform){var d=/translate\((\d+(?:\.\d+)?)(?:,(\d+(?:\.\d+)?))?\)/.exec(a.transform);d&&d[1]&&(c.translate_x=parseFloat(d[1])),d&&d[2]&&(c.translate_y=parseFloat(d[2]));var e=/rotate\((\d+\.\d+|\d+)\)/.exec(a.transform);e&&(c.rotation=parseFloat(e[1])%360)}c.x=parseFloat(a.x||0),c.y=parseFloat(a.y||0),"width"in a&&(c.width=parseInt(a.width,10)),"height"in a&&(c.height=parseInt(a.height,10));return c}function j(a){a in i||(i[a]=document.createElement(q+a+r));return i[a].cloneNode(!1)}function h(a,b){var c=0,d=0,e=Math.round;a=a.replace(/(\d*)((\.*\d*)(e ?-?\d*))/g,"$1");if(a in g)return g[a];var f=a.match(/([MLHVCSQTAZ].*?)(?=[MLHVCSQTAZ]|$)/gi),h=[];for(var i=0,j=f.length;i<j;i++){var k=f[i],l=k.charAt(0),m=k.substring(1).split(/[, ]/);switch(l){case"M":l="m",c=e(m[0]),d=e(m[1]),m=[c,d];break;case"m":l="m",c+=e(m[0]),d+=e(m[1]),m=[c,d];break;case"A":c=e(m[5]),d=e(m[6]),l="l",m=[c,d];break;case"L":l="l",c=e(m[0]),d=e(m[1]),m=[c,d];break;case"l":l="l",c+=e(m[0]),d+=e(m[1]),m=[c,d];break;case"H":l="l",c=e(m[0]),m=[c,d];break;case"h":l="l",c+=e(m[0]),m=[c,d];break;case"V":l="l",d=e(m[0]),m=[c,d];break;case"v":l="l",d+=e(m[0]),m=[c,d];break;case"C":l="l",c=e(m[4]),d=e(m[5]),m=[c,d];break;case"c":l="l",c+=e(m[4]),d+=e(m[5]),m=[c,d];break;case"Z":case"z":l="xe",m=[];default:}h.push(l+m.join(","))}return g[a]=h.join("")+"e"}function f(a,b){!(a in c)&&(b=/^rgb\((\d+),(\d+),(\d+)\)$/i.exec(a))&&(c[a]="#"+d[b[1]]+d[b[2]]+d[b[3]]);return c[a]||a}pv.VmlScene={scale:1,events:["DOMMouseScroll","mousewheel","mousedown","mouseup","mouseover","mouseout","mousemove","click","dblclick"],implicit:{css:{font:"10px sans-serif"}}};for(var a in pv.SvgScene)a in pv.SvgScene&&typeof pv.SvgScene[a]==="function"&&!(a in pv.VmlScene)&&(pv.VmlScene[a]=pv.SvgScene[a]);pv.Scene=pv.VmlScene;var b={crosshair:1,pointer:1,move:1,hand:"pointer",text:1,wait:1,help:1,progress:1,"n-resize":1,"ne-resize":1,"nw-resize":1,"s-resize":1,"se-resize":1,"sw-resize":1,"e-resize":1,"w-resize":1},c={},d=[];for(var e=0;e<256;e++)d[e]=e===0?"00":e<16?"0"+e.toString(16):e.toString(16);var g={},i={span:document.createElement("span"),div:document.createElement("div")},n=Math.PI*2/360,o=null,p=null,q="<v:",r=' class="msvml">',s="px",t={group:1,shape:1,shapetype:1,line:1,polyline:1,curve:1,rect:1,roundrect:1,oval:1,arc:1,image:1},u={butt:"flat",round:"round",square:"square",flat:"flat"},v={bevel:"bevel",round:"round",miter:"miter"},w={g:{rewrite:"div",attr:function(a,b,c){var d=k(a);c.style.position="absolute",c.style.zoom=1,c.style.left=d.translate_x+d.x+s,c.style.top=d.translate_y+d.y+s}},line:{rewrite:"shape",attr:function(a,b,c){var d=parseFloat(a.x1||0),e=parseFloat(a.y1||0),f=parseFloat(a.x2||0),g=parseFloat(a.y2||0);c.style.top=0+s,c.style.left=0+s,c.style.width=1e3+s,c.style.height=1e3+s;var h=c.getElementsByTagName("path")[0];h||(h=j("path"),c.appendChild(h));var i=Math.round;h.v="M "+i(d)+" "+i(e)+" L "+i(f)+" "+i(g)+" E",m(c,a)}},rect:{rewrite:"rect",attr:function(a,b,c){var d=k(a),e=c.style;e.position="absolute",e.left=d.translate_x+d.x+s,e.top=d.translate_y+d.y+s,d.width&&(e.width=d.width+s),d.height&&(e.height=d.height+s),l(c,a),m(c,a)}},path:{rewrite:"shape",attr:function(a,b,c){var d=k(a);c.style.left=d.translate_x+d.x+s,c.style.top=d.translate_y+d.y+s,c.style.width=1e3+s,c.style.height=1e3+s;var e=c.getElementsByTagName("path")[0];e||(e=j("path"),c.appendChild(e)),e.v=h(a.d),l(c,a),m(c,a)}},circle:{rewrite:"oval",attr:function(a,b,c){var d=k(a),e=parseFloat(a.cx||0),f=parseFloat(a.cy||0),g=parseFloat(a.r||0);c.style.top=d.translate_y+f-g+s,c.style.left=d.translate_x+e-g+s,c.style.width=g*2+s,c.style.height=g*2+s,l(c,a),m(c,a)}},text:{rewrite:"span"},svg:{rewrite:"span",oncreate:function(a){p||(p=document.createElement("span"),p.style.cssText="position:absolute;left:-9999em;top:-9999em;padding:0;margin:0;line-height:1;display:inline-block;white-space:nowrap;",document.body.appendChild(p));if(!b){var b=document.createElement("style");b.id="protovis_vml_styles",document.documentElement.firstChild.appendChild(b),b.styleSheet.addRule(".msvml","behavior:url(#default#VML);"),b.styleSheet.addRule(".msvml_block","position:absolute;top:0;left:0;");try{document.namespaces.v||document.namespaces.add("v","urn:schemas-microsoft-com:vml")}catch(c){q="<",r=' class="msvml" xmlns="urn:schemas-microsoft.com:vml">'}}},css:"position:relative;overflow:hidden;display:inline-block;~display:block;"}},x=function(){return!0},y=function(){return!1},z=["altKey","attrChange","attrName","bubbles","button","cancelable","charCode","clientX","clientY","ctrlKey","currentTarget","data","detail","eventPhase","fromElement","handler","keyCode","layerX","layerY","metaKey","newValue","offsetX","offsetY","pageX","pageY","prevValue","relatedNode","relatedTarget","screenX","screenY","shiftKey","srcElement","target","toElement","view","wheelDelta","which"];A.prototype={preventDefault:function(){this.isDefaultPrevented=x;var a=this.originalEvent;!a||(a.preventDefault?a.preventDefault():a.returnValue=!1)},stopPropagation:function(){this.isPropagationStopped=x;var a=this.originalEvent;!a||(a.stopPropagation&&a.stopPropagation(),a.cancelBubble=!0)},stopImmediatePropagation:function(){this.isImmediatePropagationStopped=x,this.stopPropagation()},isDefaultPrevented:y,isPropagationStopped:y,isImmediatePropagationStopped:y},pv.listener=function(a,b){return a.$listener||(a.$listener=function(b){try{pv.event=B(b||window.event);if(1)return a.call(this,pv.event)}finally{delete pv.event}})},pv.listen=function(a,b,c){c=pv.listener(c,a);return a.addEventListener?a.addEventListener(b,c,!1):a.attachEvent("on"+b,c)},pv.VmlScene.updateAll=function(a){if(a.length&&a[0].reverse&&a.type!="line"&&a.type!="area"){var b=pv.extend(a);for(var c=0,d=a.length-1;d>=0;c++,d--)b[c]=a[d];a=b}this.removeSiblings(this[a.type](a))},pv.VmlScene.create=function(a){var b;if(a in w){var c=w[a]||{},d=c.rewrite||a;b=j(d),b.style.cssText=c.css||"";if(d in t||a==="span"||a==="div")b.className+=" msvml_block";"oncreate"in c&&c.oncreate(b)}else b=j(d),a in t&&(b.className+=" msvml_block");a!==d&&(b.svgtype=a);return b},pv.VmlScene.expect=function(a,c,d,e){e=e||{};var f=w[c]||{},g=f.rewrite||c;if(a){if(a.tagName.toUpperCase()!==g.toUpperCase()){var h=this.create(c);a.parentNode.replaceChild(h,a),a=h}}else a=this.create(c);f.attr&&f.attr(d,e,a),d.cursor in b&&(e.cursor=b[d.cursor]===1?d.cursor:b[d.cursor]);for(var i in e){var j=e[i];j==null?a.style.removeAttribute(i):a.style[i]=j}return a},pv.VmlScene.append=function(a,b,c){a.$scene={scenes:b,index:c},a=this.title(a,b[c]);if(!a.parentNode||a.parentNode.nodeType===11){b.$g.appendChild(a);var d=w[a.svgtype];d&&typeof d.onappend==="function"&&d.onappend(a,b[c])}return a.nextSibling},pv.VmlScene.title=function(a,b){b.title&&(a.title=b.title);return a},pv.VmlScene.dispatch=pv.listener(function(a){var b=a.target.$scene;if(b){var c=a.type;switch(c){case"DOMMouseScroll":c="mousewheel",a.wheel=-480*a.detail;break;case"mousewheel":a.wheel=(window.opera?12:1)*a.wheelDelta}pv.Mark.dispatch(c,b.scenes,b.index)&&a.preventDefault()}}),pv.VmlScene.panel=function(a){var b=a.$g,c=b&&b.firstChild;for(var d=0;d<a.length;d++){var e=a[d];if(!e.visible)continue;if(!a.parent){e.canvas.style.display="inline-block",e.canvas.style.zoom=1,b&&b.parentNode!=e.canvas&&(b=e.canvas.firstChild,c=b&&b.firstChild);if(!b){b=e.canvas.appendChild(this.create("svg"));for(var f=0;f<this.events.length;f++)b.addEventListener?b.addEventListener(this.events[f],this.dispatch,!1):b.attachEvent("on"+this.events[f],this.dispatch);c=b.firstChild}a.$g=b;var g=e.width+e.left+e.right,h=e.height+e.top+e.bottom;b.style.width=g+s,b.style.height=h+s,b.style.clip="rect(0 "+g+s+" "+h+s+" 0)"}c=this.fill(c,a,d);var i=this.scale,j=e.transform,k=e.left+j.x,l=e.top+j.y;this.scale*=j.k;for(var f=0;f<e.children.length;f++){e.children[f].$g=c=this.expect(c,"g",{transform:"translate("+k+","+l+")"+(j.k!=1?" scale("+j.k+")":"")}),this.updateAll(e.children[f]);if(!c.parentNode||c.parentNode.nodeType===11){b.appendChild(c);var m=w[c.svgtype];m&&typeof m.onappend==="function"&&m.onappend(c,a[d])}c=c.nextSibling}this.scale=i,c=this.stroke(c,a,d)}return c},pv.VmlScene.image=function(a){var b=a.$g.firstChild;for(var c=0;c<a.length;c++){var d=a[c];if(!d.visible)continue;b=this.fill(b,a,c);if(!d.image){b=new Image,b.src=d.url;var e=b.style;e.position="absolute",e.top=d.top,e.left=d.left,e.width=d.width,e.height=d.height,e.cursor=d.cursor,e.msInterpolationMode="bicubic"}b=this.append(b,a,c),b=this.stroke(b,a,c)}return b},pv.VmlScene.label=function(a){var b=a.$g.firstChild;for(var c=0;c<a.length;c++){var d=a[c];if(!d.visible)continue;var e=d.textStyle;if(!e.opacity||!d.text)continue;var g={};d.cursor&&(g.cursor=d.cursor),b=this.expect(b,"text",g,{font:d.font,textDecoration:d.textDecoration,top:d.top+s,left:d.left+s,position:"absolute",display:"block",lineHeight:1,whiteSpace:"nowrap",zoom:1,cursor:"default",color:f(e.color)||"black"});var h=180*d.textAngle/Math.PI;if(h){var i=~~h%360*n,j=Math.cos(i),k=Math.sin(i);b.style.filter="progid:DXImageTransform.Microsoft.Matrix(M11="+j.toFixed(8)+","+"M12="+ -k.toFixed(8)+","+"M21="+k.toFixed(8)+","+"M22="+j.toFixed(8)+",sizingMethod='auto expand')\";"}var l=d.text.replace(/\s+/g," ");p.style.font=d.font,p.innerText=l;var m=b.style;d.textAlign==="center"?m.marginLeft=-Math.ceil(p.offsetWidth/2)+s:d.textAlign==="right"?m.marginLeft=-(p.offsetWidth+d.textMargin)+s:d.textAlign==="left"&&(m.marginLeft=d.textMargin+s),d.textBaseline==="middle"?m.marginTop=-Math.floor(p.offsetHeight*.45)+s:d.textBaseline==="top"?m.marginTop=1+s:d.textBaseline==="bottom"&&(m.marginTop=-p.offsetHeight+s),b.innerText=l,b=this.append(b,a,c)}return b},"now"in Date||(Date.now=function(){return(new Date).valueOf()})}()
\ No newline at end of file
diff --git a/sonar-server/src/main/webapp/javascripts/protovis.js b/sonar-server/src/main/webapp/javascripts/protovis.js
new file mode 100755 (executable)
index 0000000..c3eff74
--- /dev/null
@@ -0,0 +1,15326 @@
+/**
+ * @class The built-in Array class.
+ * @name Array
+ */
+
+/**
+ * Creates a new array with the results of calling a provided function on every
+ * element in this array. Implemented in Javascript 1.6.
+ *
+ * @function
+ * @name Array.prototype.map
+ * @see <a
+ * href="https://developer.mozilla.org/En/Core_JavaScript_1.5_Reference/Objects/Array/Map">map</a>
+ * documentation.
+ * @param {function} f function that produces an element of the new Array from
+ * an element of the current one.
+ * @param [o] object to use as <tt>this</tt> when executing <tt>f</tt>.
+ */
+if (!Array.prototype.map) Array.prototype.map = function(f, o) {
+  var n = this.length;
+  var result = new Array(n);
+  for (var i = 0; i < n; i++) {
+    if (i in this) {
+      result[i] = f.call(o, this[i], i, this);
+    }
+  }
+  return result;
+};
+
+/**
+ * Creates a new array with all elements that pass the test implemented by the
+ * provided function. Implemented in Javascript 1.6.
+ *
+ * @function
+ * @name Array.prototype.filter
+ * @see <a
+ * href="https://developer.mozilla.org/En/Core_JavaScript_1.5_Reference/Objects/Array/filter">filter</a>
+ * documentation.
+ * @param {function} f function to test each element of the array.
+ * @param [o] object to use as <tt>this</tt> when executing <tt>f</tt>.
+ */
+if (!Array.prototype.filter) Array.prototype.filter = function(f, o) {
+  var n = this.length;
+  var result = new Array();
+  for (var i = 0; i < n; i++) {
+    if (i in this) {
+      var v = this[i];
+      if (f.call(o, v, i, this)) result.push(v);
+    }
+  }
+  return result;
+};
+
+/**
+ * Executes a provided function once per array element. Implemented in
+ * Javascript 1.6.
+ *
+ * @function
+ * @name Array.prototype.forEach
+ * @see <a
+ * href="https://developer.mozilla.org/En/Core_JavaScript_1.5_Reference/Objects/Array/ForEach">forEach</a>
+ * documentation.
+ * @param {function} f function to execute for each element.
+ * @param [o] object to use as <tt>this</tt> when executing <tt>f</tt>.
+ */
+if (!Array.prototype.forEach) Array.prototype.forEach = function(f, o) {
+  var n = this.length >>> 0;
+  for (var i = 0; i < n; i++) {
+    if (i in this) f.call(o, this[i], i, this);
+  }
+};
+
+/**
+ * Apply a function against an accumulator and each value of the array (from
+ * left-to-right) as to reduce it to a single value. Implemented in Javascript
+ * 1.8.
+ *
+ * @function
+ * @name Array.prototype.reduce
+ * @see <a
+ * href="https://developer.mozilla.org/En/Core_JavaScript_1.5_Reference/Objects/Array/Reduce">reduce</a>
+ * documentation.
+ * @param {function} f function to execute on each value in the array.
+ * @param [v] object to use as the first argument to the first call of
+ * <tt>t</tt>.
+ */
+if (!Array.prototype.reduce) Array.prototype.reduce = function(f, v) {
+  var len = this.length;
+  if (!len && (arguments.length == 1)) {
+    throw new Error("reduce: empty array, no initial value");
+  }
+
+  var i = 0;
+  if (arguments.length < 2) {
+    while (true) {
+      if (i in this) {
+        v = this[i++];
+        break;
+      }
+      if (++i >= len) {
+        throw new Error("reduce: no values, no initial value");
+      }
+    }
+  }
+
+  for (; i < len; i++) {
+    if (i in this) {
+      v = f(v, this[i], i, this);
+    }
+  }
+  return v;
+};
+/**
+ * The top-level Protovis namespace. All public methods and fields should be
+ * registered on this object. Note that core Protovis source is surrounded by an
+ * anonymous function, so any other declared globals will not be visible outside
+ * of core methods. This also allows multiple versions of Protovis to coexist,
+ * since each version will see their own <tt>pv</tt> namespace.
+ *
+ * @namespace The top-level Protovis namespace, <tt>pv</tt>.
+ */
+var pv = {};
+
+/**
+ * Protovis version number. See <a href="http://semver.org">semver.org</a>.
+ *
+ * @type string
+ * @constant
+ */
+pv.version = "3.3.1";
+
+/**
+ * Returns the passed-in argument, <tt>x</tt>; the identity function. This method
+ * is provided for convenience since it is used as the default behavior for a
+ * number of property functions.
+ *
+ * @param x a value.
+ * @returns the value <tt>x</tt>.
+ */
+pv.identity = function(x) { return x; };
+
+/**
+ * Returns <tt>this.index</tt>. This method is provided for convenience for use
+ * with scales. For example, to color bars by their index, say:
+ *
+ * <pre>.fillStyle(pv.Colors.category10().by(pv.index))</pre>
+ *
+ * This method is equivalent to <tt>function() this.index</tt>, but more
+ * succinct. Note that the <tt>index</tt> property is also supported for
+ * accessor functions with {@link pv.max}, {@link pv.min} and other array
+ * utility methods.
+ *
+ * @see pv.Scale
+ * @see pv.Mark#index
+ */
+pv.index = function() { return this.index; };
+
+/**
+ * Returns <tt>this.childIndex</tt>. This method is provided for convenience for
+ * use with scales. For example, to color bars by their child index, say:
+ *
+ * <pre>.fillStyle(pv.Colors.category10().by(pv.child))</pre>
+ *
+ * This method is equivalent to <tt>function() this.childIndex</tt>, but more
+ * succinct.
+ *
+ * @see pv.Scale
+ * @see pv.Mark#childIndex
+ */
+pv.child = function() { return this.childIndex; };
+
+/**
+ * Returns <tt>this.parent.index</tt>. This method is provided for convenience
+ * for use with scales. This method is provided for convenience for use with
+ * scales. For example, to color bars by their parent index, say:
+ *
+ * <pre>.fillStyle(pv.Colors.category10().by(pv.parent))</pre>
+ *
+ * Tthis method is equivalent to <tt>function() this.parent.index</tt>, but more
+ * succinct.
+ *
+ * @see pv.Scale
+ * @see pv.Mark#index
+ */
+pv.parent = function() { return this.parent.index; };
+
+/**
+ * Stores the current event. This field is only set within event handlers.
+ *
+ * @type Event
+ * @name pv.event
+ */
+/**
+ * @private Returns a prototype object suitable for extending the given class
+ * <tt>f</tt>. Rather than constructing a new instance of <tt>f</tt> to serve as
+ * the prototype (which unnecessarily runs the constructor on the created
+ * prototype object, potentially polluting it), an anonymous function is
+ * generated internally that shares the same prototype:
+ *
+ * <pre>function g() {}
+ * g.prototype = f.prototype;
+ * return new g();</pre>
+ *
+ * For more details, see Douglas Crockford's essay on prototypal inheritance.
+ *
+ * @param {function} f a constructor.
+ * @returns a suitable prototype object.
+ * @see Douglas Crockford's essay on <a
+ * href="http://javascript.crockford.com/prototypal.html">prototypal
+ * inheritance</a>.
+ */
+pv.extend = function(f) {
+  function g() {}
+  g.prototype = f.prototype || f;
+  return new g();
+};
+
+try {
+  eval("pv.parse = function(x) x;"); // native support
+} catch (e) {
+
+/**
+ * @private Parses a Protovis specification, which may use JavaScript 1.8
+ * function expresses, replacing those function expressions with proper
+ * functions such that the code can be run by a JavaScript 1.6 interpreter. This
+ * hack only supports function expressions (using clumsy regular expressions, no
+ * less), and not other JavaScript 1.8 features such as let expressions.
+ *
+ * @param {string} s a Protovis specification (i.e., a string of JavaScript 1.8
+ * source code).
+ * @returns {string} a conformant JavaScript 1.6 source code.
+ */
+  pv.parse = function(js) { // hacky regex support
+    var re = new RegExp("function\\s*(\\b\\w+)?\\s*\\([^)]*\\)\\s*", "mg"), m, d, i = 0, s = "";
+    while (m = re.exec(js)) {
+      var j = m.index + m[0].length;
+      if (js.charAt(j) != '{') {
+        s += js.substring(i, j) + "{return ";
+        i = j;
+        for (var p = 0; p >= 0 && j < js.length; j++) {
+          var c = js.charAt(j);
+          switch (c) {
+            case '"': case '\'': {
+              while (++j < js.length && (d = js.charAt(j)) != c) {
+                if (d == '\\') j++;
+              }
+              break;
+            }
+            case '[': case '(': p++; break;
+            case ']': case ')': p--; break;
+            case ';':
+            case ',': if (p == 0) p--; break;
+          }
+        }
+        s += pv.parse(js.substring(i, --j)) + ";}";
+        i = j;
+      }
+      re.lastIndex = j;
+    }
+    s += js.substring(i);
+    return s;
+  };
+}
+
+/**
+ * @private Computes the value of the specified CSS property <tt>p</tt> on the
+ * specified element <tt>e</tt>.
+ *
+ * @param {string} p the name of the CSS property.
+ * @param e the element on which to compute the CSS property.
+ */
+pv.css = function(e, p) {
+  return window.getComputedStyle
+      ? window.getComputedStyle(e, null).getPropertyValue(p)
+      : e.currentStyle[p];
+};
+
+/**
+ * @private Reports the specified error to the JavaScript console. Mozilla only
+ * allows logging to the console for privileged code; if the console is
+ * unavailable, the alert dialog box is used instead.
+ *
+ * @param e the exception that triggered the error.
+ */
+pv.error = function(e) {
+  (typeof console == "undefined") ? alert(e) : console.error(e);
+};
+
+/**
+ * @private Registers the specified listener for events of the specified type on
+ * the specified target. For standards-compliant browsers, this method uses
+ * <tt>addEventListener</tt>; for Internet Explorer, <tt>attachEvent</tt>.
+ *
+ * @param target a DOM element.
+ * @param {string} type the type of event, such as "click".
+ * @param {function} the event handler callback.
+ */
+pv.listen = function(target, type, listener) {
+  listener = pv.listener(listener);
+  return target.addEventListener
+      ? target.addEventListener(type, listener, false)
+      : target.attachEvent("on" + type, listener);
+};
+
+/**
+ * @private Returns a wrapper for the specified listener function such that the
+ * {@link pv.event} is set for the duration of the listener's invocation. The
+ * wrapper is cached on the returned function, such that duplicate registrations
+ * of the wrapped event handler are ignored.
+ *
+ * @param {function} f an event handler.
+ * @returns {function} the wrapped event handler.
+ */
+pv.listener = function(f) {
+  return f.$listener || (f.$listener = function(e) {
+      try {
+        pv.event = e;
+        return f.call(this, e);
+      } finally {
+        delete pv.event;
+      }
+    });
+};
+
+/**
+ * @private Returns true iff <i>a</i> is an ancestor of <i>e</i>. This is useful
+ * for ignoring mouseout and mouseover events that are contained within the
+ * target element.
+ */
+pv.ancestor = function(a, e) {
+  while (e) {
+    if (e == a) return true;
+    e = e.parentNode;
+  }
+  return false;
+};
+
+/** @private Returns a locally-unique positive id. */
+pv.id = function() {
+  var id = 1; return function() { return id++; };
+}();
+
+/** @private Returns a function wrapping the specified constant. */
+pv.functor = function(v) {
+  return typeof v == "function" ? v : function() { return v; };
+};
+/*
+ * Parses the Protovis specifications on load, allowing the use of JavaScript
+ * 1.8 function expressions on browsers that only support JavaScript 1.6.
+ *
+ * @see pv.parse
+ */
+pv.listen(window, "load", function() {
+   /*
+    * Note: in Firefox any variables declared here are visible to the eval'd
+    * script below. Even worse, any global variables declared by the script
+    * could overwrite local variables here (such as the index, `i`)!  To protect
+    * against this, all variables are explicitly scoped on a pv.$ object.
+    */
+    pv.$ = {i:0, x:document.getElementsByTagName("script")};
+    for (; pv.$.i < pv.$.x.length; pv.$.i++) {
+      pv.$.s = pv.$.x[pv.$.i];
+      if (pv.$.s.type == "text/javascript+protovis") {
+        try {
+          window.eval(pv.parse(pv.$.s.text));
+        } catch (e) {
+          pv.error(e);
+        }
+      }
+    }
+    delete pv.$;
+  });
+/**
+ * Abstract; see an implementing class.
+ *
+ * @class Represents an abstract text formatter and parser. A <i>format</i> is a
+ * function that converts an object of a given type, such as a <tt>Date</tt>, to
+ * a human-readable string representation. The format may also have a
+ * {@link #parse} method for converting a string representation back to the
+ * given object type.
+ *
+ * <p>Because formats are themselves functions, they can be used directly as
+ * mark properties. For example, if the data associated with a label are dates,
+ * a date format can be used as label text:
+ *
+ * <pre>    .text(pv.Format.date("%m/%d/%y"))</pre>
+ *
+ * And as with scales, if the format is used in multiple places, it can be
+ * convenient to declare it as a global variable and then reference it from the
+ * appropriate property functions. For example, if the data has a <tt>date</tt>
+ * attribute, and <tt>format</tt> references a given date format:
+ *
+ * <pre>    .text(function(d) format(d.date))</pre>
+ *
+ * Similarly, to parse a string into a date:
+ *
+ * <pre>var date = format.parse("4/30/2010");</pre>
+ *
+ * Not all format implementations support parsing. See the implementing class
+ * for details.
+ *
+ * @see pv.Format.date
+ * @see pv.Format.number
+ * @see pv.Format.time
+ */
+pv.Format = {};
+
+/**
+ * Formats the specified object, returning the string representation.
+ *
+ * @function
+ * @name pv.Format.prototype.format
+ * @param {object} x the object to format.
+ * @returns {string} the formatted string.
+ */
+
+/**
+ * Parses the specified string, returning the object representation.
+ *
+ * @function
+ * @name pv.Format.prototype.parse
+ * @param {string} x the string to parse.
+ * @returns {object} the parsed object.
+ */
+
+/**
+ * @private Given a string that may be used as part of a regular expression,
+ * this methods returns an appropriately quoted version of the specified string,
+ * with any special characters escaped.
+ *
+ * @param {string} s a string to quote.
+ * @returns {string} the quoted string.
+ */
+pv.Format.re = function(s) {
+  return s.replace(/[\\\^\$\*\+\?\[\]\(\)\.\{\}]/g, "\\$&");
+};
+
+/**
+ * @private Optionally pads the specified string <i>s</i> so that it is at least
+ * <i>n</i> characters long, using the padding character <i>c</i>.
+ *
+ * @param {string} c the padding character.
+ * @param {number} n the minimum string length.
+ * @param {string} s the string to pad.
+ * @returns {string} the padded string.
+ */
+pv.Format.pad = function(c, n, s) {
+  var m = n - String(s).length;
+  return (m < 1) ? s : new Array(m + 1).join(c) + s;
+};
+/**
+ * Constructs a new date format with the specified string pattern.
+ *
+ * @class The format string is in the same format expected by the
+ * <tt>strftime</tt> function in C. The following conversion specifications are
+ * supported:<ul>
+ *
+ * <li>%a - abbreviated weekday name.</li>
+ * <li>%A - full weekday name.</li>
+ * <li>%b - abbreviated month names.</li>
+ * <li>%B - full month names.</li>
+ * <li>%c - locale's appropriate date and time.</li>
+ * <li>%C - century number.</li>
+ * <li>%d - day of month [01,31] (zero padded).</li>
+ * <li>%D - same as %m/%d/%y.</li>
+ * <li>%e - day of month [ 1,31] (space padded).</li>
+ * <li>%h - same as %b.</li>
+ * <li>%H - hour (24-hour clock) [00,23] (zero padded).</li>
+ * <li>%I - hour (12-hour clock) [01,12] (zero padded).</li>
+ * <li>%m - month number [01,12] (zero padded).</li>
+ * <li>%M - minute [0,59] (zero padded).</li>
+ * <li>%n - newline character.</li>
+ * <li>%p - locale's equivalent of a.m. or p.m.</li>
+ * <li>%r - same as %I:%M:%S %p.</li>
+ * <li>%R - same as %H:%M.</li>
+ * <li>%S - second [00,61] (zero padded).</li>
+ * <li>%t - tab character.</li>
+ * <li>%T - same as %H:%M:%S.</li>
+ * <li>%x - same as %m/%d/%y.</li>
+ * <li>%X - same as %I:%M:%S %p.</li>
+ * <li>%y - year with century [00,99] (zero padded).</li>
+ * <li>%Y - year including century.</li>
+ * <li>%% - %.</li>
+ *
+ * </ul>The following conversion specifications are currently <i>unsupported</i>
+ * for formatting:<ul>
+ *
+ * <li>%j - day number [1,366].</li>
+ * <li>%u - weekday number [1,7].</li>
+ * <li>%U - week number [00,53].</li>
+ * <li>%V - week number [01,53].</li>
+ * <li>%w - weekday number [0,6].</li>
+ * <li>%W - week number [00,53].</li>
+ * <li>%Z - timezone name or abbreviation.</li>
+ *
+ * </ul>In addition, the following conversion specifications are currently
+ * <i>unsupported</i> for parsing:<ul>
+ *
+ * <li>%a - day of week, either abbreviated or full name.</li>
+ * <li>%A - same as %a.</li>
+ * <li>%c - locale's appropriate date and time.</li>
+ * <li>%C - century number.</li>
+ * <li>%D - same as %m/%d/%y.</li>
+ * <li>%I - hour (12-hour clock) [1,12].</li>
+ * <li>%n - any white space.</li>
+ * <li>%p - locale's equivalent of a.m. or p.m.</li>
+ * <li>%r - same as %I:%M:%S %p.</li>
+ * <li>%R - same as %H:%M.</li>
+ * <li>%t - same as %n.</li>
+ * <li>%T - same as %H:%M:%S.</li>
+ * <li>%x - locale's equivalent to %m/%d/%y.</li>
+ * <li>%X - locale's equivalent to %I:%M:%S %p.</li>
+ *
+ * </ul>
+ *
+ * @see <a
+ * href="http://www.opengroup.org/onlinepubs/007908799/xsh/strftime.html">strftime</a>
+ * documentation.
+ * @see <a
+ * href="http://www.opengroup.org/onlinepubs/007908799/xsh/strptime.html">strptime</a>
+ * documentation.
+ * @extends pv.Format
+ * @param {string} pattern the format pattern.
+ */
+pv.Format.date = function(pattern) {
+  var pad = pv.Format.pad;
+
+  /** @private */
+  function format(d) {
+    return pattern.replace(/%[a-zA-Z0-9]/g, function(s) {
+        switch (s) {
+          case '%a': return [
+              "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"
+            ][d.getDay()];
+          case '%A': return [
+              "Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday",
+              "Saturday"
+            ][d.getDay()];
+          case '%h':
+          case '%b': return [
+              "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep",
+              "Oct", "Nov", "Dec"
+            ][d.getMonth()];
+          case '%B': return [
+              "January", "February", "March", "April", "May", "June", "July",
+              "August", "September", "October", "November", "December"
+            ][d.getMonth()];
+          case '%c': return d.toLocaleString();
+          case '%C': return pad("0", 2, Math.floor(d.getFullYear() / 100) % 100);
+          case '%d': return pad("0", 2, d.getDate());
+          case '%x':
+          case '%D': return pad("0", 2, d.getMonth() + 1)
+                    + "/" + pad("0", 2, d.getDate())
+                    + "/" + pad("0", 2, d.getFullYear() % 100);
+          case '%e': return pad(" ", 2, d.getDate());
+          case '%H': return pad("0", 2, d.getHours());
+          case '%I': {
+            var h = d.getHours() % 12;
+            return h ? pad("0", 2, h) : 12;
+          }
+          // TODO %j: day of year as a decimal number [001,366]
+          case '%m': return pad("0", 2, d.getMonth() + 1);
+          case '%M': return pad("0", 2, d.getMinutes());
+          case '%n': return "\n";
+          case '%p': return d.getHours() < 12 ? "AM" : "PM";
+          case '%T':
+          case '%X':
+          case '%r': {
+            var h = d.getHours() % 12;
+            return (h ? pad("0", 2, h) : 12)
+                    + ":" + pad("0", 2, d.getMinutes())
+                    + ":" + pad("0", 2, d.getSeconds())
+                    + " " + (d.getHours() < 12 ? "AM" : "PM");
+          }
+          case '%R': return pad("0", 2, d.getHours()) + ":" + pad("0", 2, d.getMinutes());
+          case '%S': return pad("0", 2, d.getSeconds());
+          case '%Q': return pad("0", 3, d.getMilliseconds());
+          case '%t': return "\t";
+          case '%u': {
+            var w = d.getDay();
+            return w ? w : 1;
+          }
+          // TODO %U: week number (sunday first day) [00,53]
+          // TODO %V: week number (monday first day) [01,53] ... with weirdness
+          case '%w': return d.getDay();
+          // TODO %W: week number (monday first day) [00,53] ... with weirdness
+          case '%y': return pad("0", 2, d.getFullYear() % 100);
+          case '%Y': return d.getFullYear();
+          // TODO %Z: timezone name or abbreviation
+          case '%%': return "%";
+        }
+        return s;
+      });
+  }
+
+  /**
+   * Converts a date to a string using the associated formatting pattern.
+   *
+   * @function
+   * @name pv.Format.date.prototype.format
+   * @param {Date} date a date to format.
+   * @returns {string} the formatted date as a string.
+   */
+  format.format = format;
+
+  /**
+   * Parses a date from a string using the associated formatting pattern.
+   *
+   * @function
+   * @name pv.Format.date.prototype.parse
+   * @param {string} s the string to parse as a date.
+   * @returns {Date} the parsed date.
+   */
+  format.parse = function(s) {
+    var year = 1970, month = 0, date = 1, hour = 0, minute = 0, second = 0;
+    var fields = [function() {}];
+
+    /* Register callbacks for each field in the format pattern. */
+    var re = pv.Format.re(pattern).replace(/%[a-zA-Z0-9]/g, function(s) {
+        switch (s) {
+          // TODO %a: day of week, either abbreviated or full name
+          // TODO %A: same as %a
+          case '%b': {
+            fields.push(function(x) { month = {
+                  Jan: 0, Feb: 1, Mar: 2, Apr: 3, May: 4, Jun: 5, Jul: 6, Aug: 7,
+                  Sep: 8, Oct: 9, Nov: 10, Dec: 11
+                }[x]; });
+            return "([A-Za-z]+)";
+          }
+          case '%h':
+          case '%B': {
+            fields.push(function(x) { month = {
+                  January: 0, February: 1, March: 2, April: 3, May: 4, June: 5,
+                  July: 6, August: 7, September: 8, October: 9, November: 10,
+                  December: 11
+                }[x]; });
+            return "([A-Za-z]+)";
+          }
+          // TODO %c: locale's appropriate date and time
+          // TODO %C: century number[0,99]
+          case '%e':
+          case '%d': {
+            fields.push(function(x) { date = x; });
+            return "([0-9]+)";
+          }
+          // TODO %D: same as %m/%d/%y
+          case '%I':
+          case '%H': {
+            fields.push(function(x) { hour = x; });
+            return "([0-9]+)";
+          }
+          // TODO %j: day number [1,366]
+          case '%m': {
+            fields.push(function(x) { month = x - 1; });
+            return "([0-9]+)";
+          }
+          case '%M': {
+            fields.push(function(x) { minute = x; });
+            return "([0-9]+)";
+          }
+          // TODO %n: any white space
+          // TODO %p: locale's equivalent of a.m. or p.m.
+          case '%p': { // TODO this is a hack
+            fields.push(function(x) {
+              if (hour == 12) {
+                if (x == "am") hour = 0;
+              } else if (x == "pm") {
+                hour = Number(hour) + 12;
+              }
+            });
+            return "(am|pm)";
+          }
+          // TODO %r: %I:%M:%S %p
+          // TODO %R: %H:%M
+          case '%S': {
+            fields.push(function(x) { second = x; });
+            return "([0-9]+)";
+          }
+          // TODO %t: any white space
+          // TODO %T: %H:%M:%S
+          // TODO %U: week number [00,53]
+          // TODO %w: weekday [0,6]
+          // TODO %W: week number [00, 53]
+          // TODO %x: locale date (%m/%d/%y)
+          // TODO %X: locale time (%I:%M:%S %p)
+          case '%y': {
+            fields.push(function(x) {
+                x = Number(x);
+                year = x + (((0 <= x) && (x < 69)) ? 2000
+                    : (((x >= 69) && (x < 100) ? 1900 : 0)));
+              });
+            return "([0-9]+)";
+          }
+          case '%Y': {
+            fields.push(function(x) { year = x; });
+            return "([0-9]+)";
+          }
+          case '%%': {
+            fields.push(function() {});
+            return "%";
+          }
+        }
+        return s;
+      });
+
+    var match = s.match(re);
+    if (match) match.forEach(function(m, i) { fields[i](m); });
+    return new Date(year, month, date, hour, minute, second);
+  };
+
+  return format;
+};
+/**
+ * Returns a time format of the given type, either "short" or "long".
+ *
+ * @class Represents a time format, converting between a <tt>number</tt>
+ * representing a duration in milliseconds, and a <tt>string</tt>. Two types of
+ * time formats are supported: "short" and "long". The <i>short</i> format type
+ * returns a string such as "3.3 days" or "12.1 minutes", while the <i>long</i>
+ * format returns "13:04:12" or similar.
+ *
+ * @extends pv.Format
+ * @param {string} type the type; "short" or "long".
+ */
+pv.Format.time = function(type) {
+  var pad = pv.Format.pad;
+
+  /*
+   * MILLISECONDS = 1
+   * SECONDS = 1e3
+   * MINUTES = 6e4
+   * HOURS = 36e5
+   * DAYS = 864e5
+   * WEEKS = 6048e5
+   * MONTHS = 2592e6
+   * YEARS = 31536e6
+   */
+
+  /** @private */
+  function format(t) {
+    t = Number(t); // force conversion from Date
+    switch (type) {
+      case "short": {
+        if (t >= 31536e6) {
+          return (t / 31536e6).toFixed(1) + " years";
+        } else if (t >= 6048e5) {
+          return (t / 6048e5).toFixed(1) + " weeks";
+        } else if (t >= 864e5) {
+          return (t / 864e5).toFixed(1) + " days";
+        } else if (t >= 36e5) {
+          return (t / 36e5).toFixed(1) + " hours";
+        } else if (t >= 6e4) {
+          return (t / 6e4).toFixed(1) + " minutes";
+        }
+        return (t / 1e3).toFixed(1) + " seconds";
+      }
+      case "long": {
+        var a = [],
+            s = ((t % 6e4) / 1e3) >> 0,
+            m = ((t % 36e5) / 6e4) >> 0;
+        a.push(pad("0", 2, s));
+        if (t >= 36e5) {
+          var h = ((t % 864e5) / 36e5) >> 0;
+          a.push(pad("0", 2, m));
+          if (t >= 864e5) {
+            a.push(pad("0", 2, h));
+            a.push(Math.floor(t / 864e5).toFixed());
+          } else {
+            a.push(h.toFixed());
+          }
+        } else {
+          a.push(m.toFixed());
+        }
+        return a.reverse().join(":");
+      }
+    }
+  }
+
+  /**
+   * Formats the specified time, returning the string representation.
+   *
+   * @function
+   * @name pv.Format.time.prototype.format
+   * @param {number} t the duration in milliseconds. May also be a <tt>Date</tt>.
+   * @returns {string} the formatted string.
+   */
+  format.format = format;
+
+  /**
+   * Parses the specified string, returning the time in milliseconds.
+   *
+   * @function
+   * @name pv.Format.time.prototype.parse
+   * @param {string} s a formatted string.
+   * @returns {number} the parsed duration in milliseconds.
+   */
+  format.parse = function(s) {
+    switch (type) {
+      case "short": {
+        var re = /([0-9,.]+)\s*([a-z]+)/g, a, t = 0;
+        while (a = re.exec(s)) {
+          var f = parseFloat(a[0].replace(",", "")), u = 0;
+          switch (a[2].toLowerCase()) {
+            case "year": case "years": u = 31536e6; break;
+            case "week": case "weeks": u = 6048e5; break;
+            case "day": case "days": u = 864e5; break;
+            case "hour": case "hours": u = 36e5; break;
+            case "minute": case "minutes": u = 6e4; break;
+            case "second": case "seconds": u = 1e3; break;
+          }
+          t += f * u;
+        }
+        return t;
+      }
+      case "long": {
+        var a = s.replace(",", "").split(":").reverse(), t = 0;
+        if (a.length) t += parseFloat(a[0]) * 1e3;
+        if (a.length > 1) t += parseFloat(a[1]) * 6e4;
+        if (a.length > 2) t += parseFloat(a[2]) * 36e5;
+        if (a.length > 3) t += parseFloat(a[3]) * 864e5;
+        return t;
+      }
+    }
+  }
+
+  return format;
+};
+/**
+ * Returns a default number format.
+ *
+ * @class Represents a number format, converting between a <tt>number</tt> and a
+ * <tt>string</tt>. This class allows numbers to be formatted with variable
+ * precision (both for the integral and fractional part of the number), optional
+ * thousands grouping, and optional padding. The thousands (",") and decimal
+ * (".") separator can be customized.
+ *
+ * @returns {pv.Format.number} a number format.
+ */
+pv.Format.number = function() {
+  var mini = 0, // default minimum integer digits
+      maxi = Infinity, // default maximum integer digits
+      mins = 0, // mini, including group separators
+      minf = 0, // default minimum fraction digits
+      maxf = 0, // default maximum fraction digits
+      maxk = 1, // 10^maxf
+      padi = "0", // default integer pad
+      padf = "0", // default fraction pad
+      padg = true, // whether group separator affects integer padding
+      decimal = ".", // default decimal separator
+      group = ",", // default group separator
+      np = "\u2212", // default negative prefix
+      ns = ""; // default negative suffix
+
+  /** @private */
+  function format(x) {
+    /* Round the fractional part, and split on decimal separator. */
+    if (Infinity > maxf) x = Math.round(x * maxk) / maxk;
+    var s = String(Math.abs(x)).split(".");
+
+    /* Pad, truncate and group the integral part. */
+    var i = s[0];
+    if (i.length > maxi) i = i.substring(i.length - maxi);
+    if (padg && (i.length < mini)) i = new Array(mini - i.length + 1).join(padi) + i;
+    if (i.length > 3) i = i.replace(/\B(?=(?:\d{3})+(?!\d))/g, group);
+    if (!padg && (i.length < mins)) i = new Array(mins - i.length + 1).join(padi) + i;
+    s[0] = x < 0 ? np + i + ns : i;
+
+    /* Pad the fractional part. */
+    var f = s[1] || "";
+    if (f.length < minf) s[1] = f + new Array(minf - f.length + 1).join(padf);
+
+    return s.join(decimal);
+  }
+
+  /**
+   * @function
+   * @name pv.Format.number.prototype.format
+   * @param {number} x
+   * @returns {string}
+   */
+  format.format = format;
+
+  /**
+   * Parses the specified string as a number. Before parsing, leading and
+   * trailing padding is removed. Group separators are also removed, and the
+   * decimal separator is replaced with the standard point ("."). The integer
+   * part is truncated per the maximum integer digits, and the fraction part is
+   * rounded per the maximum fraction digits.
+   *
+   * @function
+   * @name pv.Format.number.prototype.parse
+   * @param {string} x the string to parse.
+   * @returns {number} the parsed number.
+   */
+  format.parse = function(x) {
+    var re = pv.Format.re;
+
+    /* Remove leading and trailing padding. Split on the decimal separator. */
+    var s = String(x)
+        .replace(new RegExp("^(" + re(padi) + ")*"), "")
+        .replace(new RegExp("(" + re(padf) + ")*$"), "")
+        .split(decimal);
+
+    /* Remove grouping and truncate the integral part. */
+    var i = s[0].replace(new RegExp(re(group), "g"), "");
+    if (i.length > maxi) i = i.substring(i.length - maxi);
+
+    /* Round the fractional part. */
+    var f = s[1] ? Number("0." + s[1]) : 0;
+    if (Infinity > maxf) f = Math.round(f * maxk) / maxk;
+
+    return Math.round(i) + f;
+  };
+
+  /**
+   * Sets or gets the minimum and maximum number of integer digits. This
+   * controls the number of decimal digits to display before the decimal
+   * separator for the integral part of the number. If the number of digits is
+   * smaller than the minimum, the digits are padded; if the number of digits is
+   * larger, the digits are truncated, showing only the lower-order digits. The
+   * default range is [0, Infinity].
+   *
+   * <p>If only one argument is specified to this method, this value is used as
+   * both the minimum and maximum number. If no arguments are specified, a
+   * two-element array is returned containing the minimum and the maximum.
+   *
+   * @function
+   * @name pv.Format.number.prototype.integerDigits
+   * @param {number} [min] the minimum integer digits.
+   * @param {number} [max] the maximum integer digits.
+   * @returns {pv.Format.number} <tt>this</tt>, or the current integer digits.
+   */
+  format.integerDigits = function(min, max) {
+    if (arguments.length) {
+      mini = Number(min);
+      maxi = (arguments.length > 1) ? Number(max) : mini;
+      mins = mini + Math.floor(mini / 3) * group.length;
+      return this;
+    }
+    return [mini, maxi];
+  };
+
+  /**
+   * Sets or gets the minimum and maximum number of fraction digits. The
+   * controls the number of decimal digits to display after the decimal
+   * separator for the fractional part of the number. If the number of digits is
+   * smaller than the minimum, the digits are padded; if the number of digits is
+   * larger, the fractional part is rounded, showing only the higher-order
+   * digits. The default range is [0, 0].
+   *
+   * <p>If only one argument is specified to this method, this value is used as
+   * both the minimum and maximum number. If no arguments are specified, a
+   * two-element array is returned containing the minimum and the maximum.
+   *
+   * @function
+   * @name pv.Format.number.prototype.fractionDigits
+   * @param {number} [min] the minimum fraction digits.
+   * @param {number} [max] the maximum fraction digits.
+   * @returns {pv.Format.number} <tt>this</tt>, or the current fraction digits.
+   */
+  format.fractionDigits = function(min, max) {
+    if (arguments.length) {
+      minf = Number(min);
+      maxf = (arguments.length > 1) ? Number(max) : minf;
+      maxk = Math.pow(10, maxf);
+      return this;
+    }
+    return [minf, maxf];
+  };
+
+  /**
+   * Sets or gets the character used to pad the integer part. The integer pad is
+   * used when the number of integer digits is smaller than the minimum. The
+   * default pad character is "0" (zero).
+   *
+   * @param {string} [x] the new pad character.
+   * @returns {pv.Format.number} <tt>this</tt> or the current pad character.
+   */
+  format.integerPad = function(x) {
+    if (arguments.length) {
+      padi = String(x);
+      padg = /\d/.test(padi);
+      return this;
+    }
+    return padi;
+  };
+
+  /**
+   * Sets or gets the character used to pad the fration part. The fraction pad
+   * is used when the number of fraction digits is smaller than the minimum. The
+   * default pad character is "0" (zero).
+   *
+   * @param {string} [x] the new pad character.
+   * @returns {pv.Format.number} <tt>this</tt> or the current pad character.
+   */
+  format.fractionPad = function(x) {
+    if (arguments.length) {
+      padf = String(x);
+      return this;
+    }
+    return padf;
+  };
+
+  /**
+   * Sets or gets the character used as the decimal point, separating the
+   * integer and fraction parts of the number. The default decimal point is ".".
+   *
+   * @param {string} [x] the new decimal separator.
+   * @returns {pv.Format.number} <tt>this</tt> or the current decimal separator.
+   */
+  format.decimal = function(x) {
+    if (arguments.length) {
+      decimal = String(x);
+      return this;
+    }
+    return decimal;
+  };
+
+  /**
+   * Sets or gets the character used as the group separator, grouping integer
+   * digits by thousands. The default decimal point is ",". Grouping can be
+   * disabled by using "" for the separator.
+   *
+   * @param {string} [x] the new group separator.
+   * @returns {pv.Format.number} <tt>this</tt> or the current group separator.
+   */
+  format.group = function(x) {
+    if (arguments.length) {
+      group = x ? String(x) : "";
+      mins = mini + Math.floor(mini / 3) * group.length;
+      return this;
+    }
+    return group;
+  };
+
+  /**
+   * Sets or gets the negative prefix and suffix. The default negative prefix is
+   * "&minus;", and the default negative suffix is the empty string.
+   *
+   * @param {string} [x] the negative prefix.
+   * @param {string} [y] the negative suffix.
+   * @returns {pv.Format.number} <tt>this</tt> or the current negative format.
+   */
+  format.negativeAffix = function(x, y) {
+    if (arguments.length) {
+      np = String(x || "");
+      ns = String(y || "");
+      return this;
+    }
+    return [np, ns];
+  };
+
+  return format;
+};
+/**
+ * @private A private variant of Array.prototype.map that supports the index
+ * property.
+ */
+pv.map = function(array, f) {
+  var o = {};
+  return f
+      ? array.map(function(d, i) { o.index = i; return f.call(o, d); })
+      : array.slice();
+};
+
+/**
+ * Concatenates the specified array with itself <i>n</i> times. For example,
+ * <tt>pv.repeat([1, 2])</tt> returns [1, 2, 1, 2].
+ *
+ * @param {array} a an array.
+ * @param {number} [n] the number of times to repeat; defaults to two.
+ * @returns {array} an array that repeats the specified array.
+ */
+pv.repeat = function(array, n) {
+  if (arguments.length == 1) n = 2;
+  return pv.blend(pv.range(n).map(function() { return array; }));
+};
+
+/**
+ * Given two arrays <tt>a</tt> and <tt>b</tt>, <style
+ * type="text/css">sub{line-height:0}</style> returns an array of all possible
+ * pairs of elements [a<sub>i</sub>, b<sub>j</sub>]. The outer loop is on array
+ * <i>a</i>, while the inner loop is on <i>b</i>, such that the order of
+ * returned elements is [a<sub>0</sub>, b<sub>0</sub>], [a<sub>0</sub>,
+ * b<sub>1</sub>], ... [a<sub>0</sub>, b<sub>m</sub>], [a<sub>1</sub>,
+ * b<sub>0</sub>], [a<sub>1</sub>, b<sub>1</sub>], ... [a<sub>1</sub>,
+ * b<sub>m</sub>], ... [a<sub>n</sub>, b<sub>m</sub>]. If either array is empty,
+ * an empty array is returned.
+ *
+ * @param {array} a an array.
+ * @param {array} b an array.
+ * @returns {array} an array of pairs of elements in <tt>a</tt> and <tt>b</tt>.
+ */
+pv.cross = function(a, b) {
+  var array = [];
+  for (var i = 0, n = a.length, m = b.length; i < n; i++) {
+    for (var j = 0, x = a[i]; j < m; j++) {
+      array.push([x, b[j]]);
+    }
+  }
+  return array;
+};
+
+/**
+ * Given the specified array of arrays, concatenates the arrays into a single
+ * array. If the individual arrays are explicitly known, an alternative to blend
+ * is to use JavaScript's <tt>concat</tt> method directly. These two equivalent
+ * expressions:<ul>
+ *
+ * <li><tt>pv.blend([[1, 2, 3], ["a", "b", "c"]])</tt>
+ * <li><tt>[1, 2, 3].concat(["a", "b", "c"])</tt>
+ *
+ * </ul>return [1, 2, 3, "a", "b", "c"].
+ *
+ * @param {array[]} arrays an array of arrays.
+ * @returns {array} an array containing all the elements of each array in
+ * <tt>arrays</tt>.
+ */
+pv.blend = function(arrays) {
+  return Array.prototype.concat.apply([], arrays);
+};
+
+/**
+ * Given the specified array of arrays, <style
+ * type="text/css">sub{line-height:0}</style> transposes each element
+ * array<sub>ij</sub> with array<sub>ji</sub>. If the array has dimensions
+ * <i>n</i>&times;<i>m</i>, it will have dimensions <i>m</i>&times;<i>n</i>
+ * after this method returns. This method transposes the elements of the array
+ * in place, mutating the array, and returning a reference to the array.
+ *
+ * @param {array[]} arrays an array of arrays.
+ * @returns {array[]} the passed-in array, after transposing the elements.
+ */
+pv.transpose = function(arrays) {
+  var n = arrays.length, m = pv.max(arrays, function(d) { return d.length; });
+
+  if (m > n) {
+    arrays.length = m;
+    for (var i = n; i < m; i++) {
+      arrays[i] = new Array(n);
+    }
+    for (var i = 0; i < n; i++) {
+      for (var j = i + 1; j < m; j++) {
+        var t = arrays[i][j];
+        arrays[i][j] = arrays[j][i];
+        arrays[j][i] = t;
+      }
+    }
+  } else {
+    for (var i = 0; i < m; i++) {
+      arrays[i].length = n;
+    }
+    for (var i = 0; i < n; i++) {
+      for (var j = 0; j < i; j++) {
+        var t = arrays[i][j];
+        arrays[i][j] = arrays[j][i];
+        arrays[j][i] = t;
+      }
+    }
+  }
+
+  arrays.length = m;
+  for (var i = 0; i < m; i++) {
+    arrays[i].length = n;
+  }
+
+  return arrays;
+};
+
+/**
+ * Returns a normalized copy of the specified array, such that the sum of the
+ * returned elements sum to one. If the specified array is not an array of
+ * numbers, an optional accessor function <tt>f</tt> can be specified to map the
+ * elements to numbers. For example, if <tt>array</tt> is an array of objects,
+ * and each object has a numeric property "foo", the expression
+ *
+ * <pre>pv.normalize(array, function(d) d.foo)</pre>
+ *
+ * returns a normalized array on the "foo" property. If an accessor function is
+ * not specified, the identity function is used. Accessor functions can refer to
+ * <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number[]} an array of numbers that sums to one.
+ */
+pv.normalize = function(array, f) {
+  var norm = pv.map(array, f), sum = pv.sum(norm);
+  for (var i = 0; i < norm.length; i++) norm[i] /= sum;
+  return norm;
+};
+
+/**
+ * Returns a permutation of the specified array, using the specified array of
+ * indexes. The returned array contains the corresponding element in
+ * <tt>array</tt> for each index in <tt>indexes</tt>, in order. For example,
+ *
+ * <pre>pv.permute(["a", "b", "c"], [1, 2, 0])</pre>
+ *
+ * returns <tt>["b", "c", "a"]</tt>. It is acceptable for the array of indexes
+ * to be a different length from the array of elements, and for indexes to be
+ * duplicated or omitted. The optional accessor function <tt>f</tt> can be used
+ * to perform a simultaneous mapping of the array elements. Accessor functions
+ * can refer to <tt>this.index</tt>.
+ *
+ * @param {array} array an array.
+ * @param {number[]} indexes an array of indexes into <tt>array</tt>.
+ * @param {function} [f] an optional accessor function.
+ * @returns {array} an array of elements from <tt>array</tt>; a permutation.
+ */
+pv.permute = function(array, indexes, f) {
+  if (!f) f = pv.identity;
+  var p = new Array(indexes.length), o = {};
+  indexes.forEach(function(j, i) { o.index = j; p[i] = f.call(o, array[j]); });
+  return p;
+};
+
+/**
+ * Returns a map from key to index for the specified <tt>keys</tt> array. For
+ * example,
+ *
+ * <pre>pv.numerate(["a", "b", "c"])</pre>
+ *
+ * returns <tt>{a: 0, b: 1, c: 2}</tt>. Note that since JavaScript maps only
+ * support string keys, <tt>keys</tt> must contain strings, or other values that
+ * naturally map to distinct string values. Alternatively, an optional accessor
+ * function <tt>f</tt> can be specified to compute the string key for the given
+ * element. Accessor functions can refer to <tt>this.index</tt>.
+ *
+ * @param {array} keys an array, usually of string keys.
+ * @param {function} [f] an optional key function.
+ * @returns a map from key to index.
+ */
+pv.numerate = function(keys, f) {
+  if (!f) f = pv.identity;
+  var map = {}, o = {};
+  keys.forEach(function(x, i) { o.index = i; map[f.call(o, x)] = i; });
+  return map;
+};
+
+/**
+ * Returns the unique elements in the specified array, in the order they appear.
+ * Note that since JavaScript maps only support string keys, <tt>array</tt> must
+ * contain strings, or other values that naturally map to distinct string
+ * values. Alternatively, an optional accessor function <tt>f</tt> can be
+ * specified to compute the string key for the given element. Accessor functions
+ * can refer to <tt>this.index</tt>.
+ *
+ * @param {array} array an array, usually of string keys.
+ * @param {function} [f] an optional key function.
+ * @returns {array} the unique values.
+ */
+pv.uniq = function(array, f) {
+  if (!f) f = pv.identity;
+  var map = {}, keys = [], o = {}, y;
+  array.forEach(function(x, i) {
+    o.index = i;
+    y = f.call(o, x);
+    if (!(y in map)) map[y] = keys.push(y);
+  });
+  return keys;
+};
+
+/**
+ * The comparator function for natural order. This can be used in conjunction with
+ * the built-in array <tt>sort</tt> method to sort elements by their natural
+ * order, ascending. Note that if no comparator function is specified to the
+ * built-in <tt>sort</tt> method, the default order is lexicographic, <i>not</i>
+ * natural!
+ *
+ * @see <a
+ * href="http://developer.mozilla.org/en/Core_JavaScript_1.5_Reference/Global_Objects/Array/sort">Array.sort</a>.
+ * @param a an element to compare.
+ * @param b an element to compare.
+ * @returns {number} negative if a &lt; b; positive if a &gt; b; otherwise 0.
+ */
+pv.naturalOrder = function(a, b) {
+  return (a < b) ? -1 : ((a > b) ? 1 : 0);
+};
+
+/**
+ * The comparator function for reverse natural order. This can be used in
+ * conjunction with the built-in array <tt>sort</tt> method to sort elements by
+ * their natural order, descending. Note that if no comparator function is
+ * specified to the built-in <tt>sort</tt> method, the default order is
+ * lexicographic, <i>not</i> natural!
+ *
+ * @see #naturalOrder
+ * @param a an element to compare.
+ * @param b an element to compare.
+ * @returns {number} negative if a &lt; b; positive if a &gt; b; otherwise 0.
+ */
+pv.reverseOrder = function(b, a) {
+  return (a < b) ? -1 : ((a > b) ? 1 : 0);
+};
+
+/**
+ * Searches the specified array of numbers for the specified value using the
+ * binary search algorithm. The array must be sorted (as by the <tt>sort</tt>
+ * method) prior to making this call. If it is not sorted, the results are
+ * undefined. If the array contains multiple elements with the specified value,
+ * there is no guarantee which one will be found.
+ *
+ * <p>The <i>insertion point</i> is defined as the point at which the value
+ * would be inserted into the array: the index of the first element greater than
+ * the value, or <tt>array.length</tt>, if all elements in the array are less
+ * than the specified value. Note that this guarantees that the return value
+ * will be nonnegative if and only if the value is found.
+ *
+ * @param {number[]} array the array to be searched.
+ * @param {number} value the value to be searched for.
+ * @returns the index of the search value, if it is contained in the array;
+ * otherwise, (-(<i>insertion point</i>) - 1).
+ * @param {function} [f] an optional key function.
+ */
+pv.search = function(array, value, f) {
+  if (!f) f = pv.identity;
+  var low = 0, high = array.length - 1;
+  while (low <= high) {
+    var mid = (low + high) >> 1, midValue = f(array[mid]);
+    if (midValue < value) low = mid + 1;
+    else if (midValue > value) high = mid - 1;
+    else return mid;
+  }
+  return -low - 1;
+};
+
+pv.search.index = function(array, value, f) {
+  var i = pv.search(array, value, f);
+  return (i < 0) ? (-i - 1) : i;
+};
+/**
+ * Returns an array of numbers, starting at <tt>start</tt>, incrementing by
+ * <tt>step</tt>, until <tt>stop</tt> is reached. The stop value is
+ * exclusive. If only a single argument is specified, this value is interpeted
+ * as the <i>stop</i> value, with the <i>start</i> value as zero. If only two
+ * arguments are specified, the step value is implied to be one.
+ *
+ * <p>The method is modeled after the built-in <tt>range</tt> method from
+ * Python. See the Python documentation for more details.
+ *
+ * @see <a href="http://docs.python.org/library/functions.html#range">Python range</a>
+ * @param {number} [start] the start value.
+ * @param {number} stop the stop value.
+ * @param {number} [step] the step value.
+ * @returns {number[]} an array of numbers.
+ */
+pv.range = function(start, stop, step) {
+  if (arguments.length == 1) {
+    stop = start;
+    start = 0;
+  }
+  if (step == undefined) step = 1;
+  if ((stop - start) / step == Infinity) throw new Error("range must be finite");
+  var array = [], i = 0, j;
+  stop -= (stop - start) * 1e-10; // floating point precision!
+  if (step < 0) {
+    while ((j = start + step * i++) > stop) {
+      array.push(j);
+    }
+  } else {
+    while ((j = start + step * i++) < stop) {
+      array.push(j);
+    }
+  }
+  return array;
+};
+
+/**
+ * Returns a random number in the range [<tt>start</tt>, <tt>stop</tt>) that is
+ * a multiple of <tt>step</tt>. More specifically, the returned number is of the
+ * form <tt>start</tt> + <i>n</i> * <tt>step</tt>, where <i>n</i> is a
+ * nonnegative integer. If <tt>step</tt> is not specified, it defaults to 1,
+ * returning a random integer if <tt>start</tt> is also an integer.
+ *
+ * @param {number} [start] the start value value.
+ * @param {number} stop the stop value.
+ * @param {number} [step] the step value.
+ * @returns {number} a random number between <i>start</i> and <i>stop</i>.
+ */
+pv.random = function(start, stop, step) {
+  if (arguments.length == 1) {
+    stop = start;
+    start = 0;
+  }
+  if (step == undefined) step = 1;
+  return step
+      ? (Math.floor(Math.random() * (stop - start) / step) * step + start)
+      : (Math.random() * (stop - start) + start);
+};
+
+/**
+ * Returns the sum of the specified array. If the specified array is not an
+ * array of numbers, an optional accessor function <tt>f</tt> can be specified
+ * to map the elements to numbers. See {@link #normalize} for an example.
+ * Accessor functions can refer to <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the sum of the specified array.
+ */
+pv.sum = function(array, f) {
+  var o = {};
+  return array.reduce(f
+      ? function(p, d, i) { o.index = i; return p + f.call(o, d); }
+      : function(p, d) { return p + d; }, 0);
+};
+
+/**
+ * Returns the maximum value of the specified array. If the specified array is
+ * not an array of numbers, an optional accessor function <tt>f</tt> can be
+ * specified to map the elements to numbers. See {@link #normalize} for an
+ * example. Accessor functions can refer to <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the maximum value of the specified array.
+ */
+pv.max = function(array, f) {
+  if (f == pv.index) return array.length - 1;
+  return Math.max.apply(null, f ? pv.map(array, f) : array);
+};
+
+/**
+ * Returns the index of the maximum value of the specified array. If the
+ * specified array is not an array of numbers, an optional accessor function
+ * <tt>f</tt> can be specified to map the elements to numbers. See
+ * {@link #normalize} for an example. Accessor functions can refer to
+ * <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the index of the maximum value of the specified array.
+ */
+pv.max.index = function(array, f) {
+  if (!array.length) return -1;
+  if (f == pv.index) return array.length - 1;
+  if (!f) f = pv.identity;
+  var maxi = 0, maxx = -Infinity, o = {};
+  for (var i = 0; i < array.length; i++) {
+    o.index = i;
+    var x = f.call(o, array[i]);
+    if (x > maxx) {
+      maxx = x;
+      maxi = i;
+    }
+  }
+  return maxi;
+}
+
+/**
+ * Returns the minimum value of the specified array of numbers. If the specified
+ * array is not an array of numbers, an optional accessor function <tt>f</tt>
+ * can be specified to map the elements to numbers. See {@link #normalize} for
+ * an example. Accessor functions can refer to <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the minimum value of the specified array.
+ */
+pv.min = function(array, f) {
+  if (f == pv.index) return 0;
+  return Math.min.apply(null, f ? pv.map(array, f) : array);
+};
+
+/**
+ * Returns the index of the minimum value of the specified array. If the
+ * specified array is not an array of numbers, an optional accessor function
+ * <tt>f</tt> can be specified to map the elements to numbers. See
+ * {@link #normalize} for an example. Accessor functions can refer to
+ * <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the index of the minimum value of the specified array.
+ */
+pv.min.index = function(array, f) {
+  if (!array.length) return -1;
+  if (f == pv.index) return 0;
+  if (!f) f = pv.identity;
+  var mini = 0, minx = Infinity, o = {};
+  for (var i = 0; i < array.length; i++) {
+    o.index = i;
+    var x = f.call(o, array[i]);
+    if (x < minx) {
+      minx = x;
+      mini = i;
+    }
+  }
+  return mini;
+}
+
+/**
+ * Returns the arithmetic mean, or average, of the specified array. If the
+ * specified array is not an array of numbers, an optional accessor function
+ * <tt>f</tt> can be specified to map the elements to numbers. See
+ * {@link #normalize} for an example. Accessor functions can refer to
+ * <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the mean of the specified array.
+ */
+pv.mean = function(array, f) {
+  return pv.sum(array, f) / array.length;
+};
+
+/**
+ * Returns the median of the specified array. If the specified array is not an
+ * array of numbers, an optional accessor function <tt>f</tt> can be specified
+ * to map the elements to numbers. See {@link #normalize} for an example.
+ * Accessor functions can refer to <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the median of the specified array.
+ */
+pv.median = function(array, f) {
+  if (f == pv.index) return (array.length - 1) / 2;
+  array = pv.map(array, f).sort(pv.naturalOrder);
+  if (array.length % 2) return array[Math.floor(array.length / 2)];
+  var i = array.length / 2;
+  return (array[i - 1] + array[i]) / 2;
+};
+
+/**
+ * Returns the unweighted variance of the specified array. If the specified
+ * array is not an array of numbers, an optional accessor function <tt>f</tt>
+ * can be specified to map the elements to numbers. See {@link #normalize} for
+ * an example. Accessor functions can refer to <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the variance of the specified array.
+ */
+pv.variance = function(array, f) {
+  if (array.length < 1) return NaN;
+  if (array.length == 1) return 0;
+  var mean = pv.mean(array, f), sum = 0, o = {};
+  if (!f) f = pv.identity;
+  for (var i = 0; i < array.length; i++) {
+    o.index = i;
+    var d = f.call(o, array[i]) - mean;
+    sum += d * d;
+  }
+  return sum;
+};
+
+/**
+ * Returns an unbiased estimation of the standard deviation of a population,
+ * given the specified random sample. If the specified array is not an array of
+ * numbers, an optional accessor function <tt>f</tt> can be specified to map the
+ * elements to numbers. See {@link #normalize} for an example. Accessor
+ * functions can refer to <tt>this.index</tt>.
+ *
+ * @param {array} array an array of objects, or numbers.
+ * @param {function} [f] an optional accessor function.
+ * @returns {number} the standard deviation of the specified array.
+ */
+pv.deviation = function(array, f) {
+  return Math.sqrt(pv.variance(array, f) / (array.length - 1));
+};
+
+/**
+ * Returns the logarithm with a given base value.
+ *
+ * @param {number} x the number for which to compute the logarithm.
+ * @param {number} b the base of the logarithm.
+ * @returns {number} the logarithm value.
+ */
+pv.log = function(x, b) {
+  return Math.log(x) / Math.log(b);
+};
+
+/**
+ * Computes a zero-symmetric logarithm. Computes the logarithm of the absolute
+ * value of the input, and determines the sign of the output according to the
+ * sign of the input value.
+ *
+ * @param {number} x the number for which to compute the logarithm.
+ * @param {number} b the base of the logarithm.
+ * @returns {number} the symmetric log value.
+ */
+pv.logSymmetric = function(x, b) {
+  return (x == 0) ? 0 : ((x < 0) ? -pv.log(-x, b) : pv.log(x, b));
+};
+
+/**
+ * Computes a zero-symmetric logarithm, with adjustment to values between zero
+ * and the logarithm base. This adjustment introduces distortion for values less
+ * than the base number, but enables simultaneous plotting of log-transformed
+ * data involving both positive and negative numbers.
+ *
+ * @param {number} x the number for which to compute the logarithm.
+ * @param {number} b the base of the logarithm.
+ * @returns {number} the adjusted, symmetric log value.
+ */
+pv.logAdjusted = function(x, b) {
+  if (!isFinite(x)) return x;
+  var negative = x < 0;
+  if (x < b) x += (b - x) / b;
+  return negative ? -pv.log(x, b) : pv.log(x, b);
+};
+
+/**
+ * Rounds an input value down according to its logarithm. The method takes the
+ * floor of the logarithm of the value and then uses the resulting value as an
+ * exponent for the base value.
+ *
+ * @param {number} x the number for which to compute the logarithm floor.
+ * @param {number} b the base of the logarithm.
+ * @returns {number} the rounded-by-logarithm value.
+ */
+pv.logFloor = function(x, b) {
+  return (x > 0)
+      ? Math.pow(b, Math.floor(pv.log(x, b)))
+      : -Math.pow(b, -Math.floor(-pv.log(-x, b)));
+};
+
+/**
+ * Rounds an input value up according to its logarithm. The method takes the
+ * ceiling of the logarithm of the value and then uses the resulting value as an
+ * exponent for the base value.
+ *
+ * @param {number} x the number for which to compute the logarithm ceiling.
+ * @param {number} b the base of the logarithm.
+ * @returns {number} the rounded-by-logarithm value.
+ */
+pv.logCeil = function(x, b) {
+  return (x > 0)
+      ? Math.pow(b, Math.ceil(pv.log(x, b)))
+      : -Math.pow(b, -Math.ceil(-pv.log(-x, b)));
+};
+
+(function() {
+  var radians = Math.PI / 180,
+      degrees = 180 / Math.PI;
+
+  /** Returns the number of radians corresponding to the specified degrees. */
+  pv.radians = function(degrees) { return radians * degrees; };
+
+  /** Returns the number of degrees corresponding to the specified radians. */
+  pv.degrees = function(radians) { return degrees * radians; };
+})();
+/**
+ * Returns all of the property names (keys) of the specified object (a map). The
+ * order of the returned array is not defined.
+ *
+ * @param map an object.
+ * @returns {string[]} an array of strings corresponding to the keys.
+ * @see #entries
+ */
+pv.keys = function(map) {
+  var array = [];
+  for (var key in map) {
+    array.push(key);
+  }
+  return array;
+};
+
+/**
+ * Returns all of the entries (key-value pairs) of the specified object (a
+ * map). The order of the returned array is not defined. Each key-value pair is
+ * represented as an object with <tt>key</tt> and <tt>value</tt> attributes,
+ * e.g., <tt>{key: "foo", value: 42}</tt>.
+ *
+ * @param map an object.
+ * @returns {array} an array of key-value pairs corresponding to the keys.
+ */
+pv.entries = function(map) {
+  var array = [];
+  for (var key in map) {
+    array.push({ key: key, value: map[key] });
+  }
+  return array;
+};
+
+/**
+ * Returns all of the values (attribute values) of the specified object (a
+ * map). The order of the returned array is not defined.
+ *
+ * @param map an object.
+ * @returns {array} an array of objects corresponding to the values.
+ * @see #entries
+ */
+pv.values = function(map) {
+  var array = [];
+  for (var key in map) {
+    array.push(map[key]);
+  }
+  return array;
+};
+
+/**
+ * Returns a map constructed from the specified <tt>keys</tt>, using the
+ * function <tt>f</tt> to compute the value for each key. The single argument to
+ * the value function is the key. The callback is invoked only for indexes of
+ * the array which have assigned values; it is not invoked for indexes which
+ * have been deleted or which have never been assigned values.
+ *
+ * <p>For example, this expression creates a map from strings to string length:
+ *
+ * <pre>pv.dict(["one", "three", "seventeen"], function(s) s.length)</pre>
+ *
+ * The returned value is <tt>{one: 3, three: 5, seventeen: 9}</tt>. Accessor
+ * functions can refer to <tt>this.index</tt>.
+ *
+ * @param {array} keys an array.
+ * @param {function} f a value function.
+ * @returns a map from keys to values.
+ */
+pv.dict = function(keys, f) {
+  var m = {}, o = {};
+  for (var i = 0; i < keys.length; i++) {
+    if (i in keys) {
+      var k = keys[i];
+      o.index = i;
+      m[k] = f.call(o, k);
+    }
+  }
+  return m;
+};
+/**
+ * Returns a {@link pv.Dom} operator for the given map. This is a convenience
+ * factory method, equivalent to <tt>new pv.Dom(map)</tt>. To apply the operator
+ * and retrieve the root node, call {@link pv.Dom#root}; to retrieve all nodes
+ * flattened, use {@link pv.Dom#nodes}.
+ *
+ * @see pv.Dom
+ * @param map a map from which to construct a DOM.
+ * @returns {pv.Dom} a DOM operator for the specified map.
+ */
+pv.dom = function(map) {
+  return new pv.Dom(map);
+};
+
+/**
+ * Constructs a DOM operator for the specified map. This constructor should not
+ * be invoked directly; use {@link pv.dom} instead.
+ *
+ * @class Represets a DOM operator for the specified map. This allows easy
+ * transformation of a hierarchical JavaScript object (such as a JSON map) to a
+ * W3C Document Object Model hierarchy. For more information on which attributes
+ * and methods from the specification are supported, see {@link pv.Dom.Node}.
+ *
+ * <p>Leaves in the map are determined using an associated <i>leaf</i> function;
+ * see {@link #leaf}. By default, leaves are any value whose type is not
+ * "object", such as numbers or strings.
+ *
+ * @param map a map from which to construct a DOM.
+ */
+pv.Dom = function(map) {
+  this.$map = map;
+};
+
+/** @private The default leaf function. */
+pv.Dom.prototype.$leaf = function(n) {
+  return typeof n != "object";
+};
+
+/**
+ * Sets or gets the leaf function for this DOM operator. The leaf function
+ * identifies which values in the map are leaves, and which are internal nodes.
+ * By default, objects are considered internal nodes, and primitives (such as
+ * numbers and strings) are considered leaves.
+ *
+ * @param {function} f the new leaf function.
+ * @returns the current leaf function, or <tt>this</tt>.
+ */
+pv.Dom.prototype.leaf = function(f) {
+  if (arguments.length) {
+    this.$leaf = f;
+    return this;
+  }
+  return this.$leaf;
+};
+
+/**
+ * Applies the DOM operator, returning the root node.
+ *
+ * @returns {pv.Dom.Node} the root node.
+ * @param {string} [nodeName] optional node name for the root.
+ */
+pv.Dom.prototype.root = function(nodeName) {
+  var leaf = this.$leaf, root = recurse(this.$map);
+
+  /** @private */
+  function recurse(map) {
+    var n = new pv.Dom.Node();
+    for (var k in map) {
+      var v = map[k];
+      n.appendChild(leaf(v) ? new pv.Dom.Node(v) : recurse(v)).nodeName = k;
+    }
+    return n;
+  }
+
+  root.nodeName = nodeName;
+  return root;
+};
+
+/**
+ * Applies the DOM operator, returning the array of all nodes in preorder
+ * traversal.
+ *
+ * @returns {array} the array of nodes in preorder traversal.
+ */
+pv.Dom.prototype.nodes = function() {
+  return this.root().nodes();
+};
+
+/**
+ * Constructs a DOM node for the specified value. Instances of this class are
+ * not typically created directly; instead they are generated from a JavaScript
+ * map using the {@link pv.Dom} operator.
+ *
+ * @class Represents a <tt>Node</tt> in the W3C Document Object Model.
+ */
+pv.Dom.Node = function(value) {
+  this.nodeValue = value;
+  this.childNodes = [];
+};
+
+/**
+ * The node name. When generated from a map, the node name corresponds to the
+ * key at the given level in the map. Note that the root node has no associated
+ * key, and thus has an undefined node name (and no <tt>parentNode</tt>).
+ *
+ * @type string
+ * @field pv.Dom.Node.prototype.nodeName
+ */
+
+/**
+ * The node value. When generated from a map, node value corresponds to the leaf
+ * value for leaf nodes, and is undefined for internal nodes.
+ *
+ * @field pv.Dom.Node.prototype.nodeValue
+ */
+
+/**
+ * The array of child nodes. This array is empty for leaf nodes. An easy way to
+ * check if child nodes exist is to query <tt>firstChild</tt>.
+ *
+ * @type array
+ * @field pv.Dom.Node.prototype.childNodes
+ */
+
+/**
+ * The parent node, which is null for root nodes.
+ *
+ * @type pv.Dom.Node
+ */
+pv.Dom.Node.prototype.parentNode = null;
+
+/**
+ * The first child, which is null for leaf nodes.
+ *
+ * @type pv.Dom.Node
+ */
+pv.Dom.Node.prototype.firstChild = null;
+
+/**
+ * The last child, which is null for leaf nodes.
+ *
+ * @type pv.Dom.Node
+ */
+pv.Dom.Node.prototype.lastChild = null;
+
+/**
+ * The previous sibling node, which is null for the first child.
+ *
+ * @type pv.Dom.Node
+ */
+pv.Dom.Node.prototype.previousSibling = null;
+
+/**
+ * The next sibling node, which is null for the last child.
+ *
+ * @type pv.Dom.Node
+ */
+pv.Dom.Node.prototype.nextSibling = null;
+
+/**
+ * Removes the specified child node from this node.
+ *
+ * @throws Error if the specified child is not a child of this node.
+ * @returns {pv.Dom.Node} the removed child.
+ */
+pv.Dom.Node.prototype.removeChild = function(n) {
+  var i = this.childNodes.indexOf(n);
+  if (i == -1) throw new Error("child not found");
+  this.childNodes.splice(i, 1);
+  if (n.previousSibling) n.previousSibling.nextSibling = n.nextSibling;
+  else this.firstChild = n.nextSibling;
+  if (n.nextSibling) n.nextSibling.previousSibling = n.previousSibling;
+  else this.lastChild = n.previousSibling;
+  delete n.nextSibling;
+  delete n.previousSibling;
+  delete n.parentNode;
+  return n;
+};
+
+/**
+ * Appends the specified child node to this node. If the specified child is
+ * already part of the DOM, the child is first removed before being added to
+ * this node.
+ *
+ * @returns {pv.Dom.Node} the appended child.
+ */
+pv.Dom.Node.prototype.appendChild = function(n) {
+  if (n.parentNode) n.parentNode.removeChild(n);
+  n.parentNode = this;
+  n.previousSibling = this.lastChild;
+  if (this.lastChild) this.lastChild.nextSibling = n;
+  else this.firstChild = n;
+  this.lastChild = n;
+  this.childNodes.push(n);
+  return n;
+};
+
+/**
+ * Inserts the specified child <i>n</i> before the given reference child
+ * <i>r</i> of this node. If <i>r</i> is null, this method is equivalent to
+ * {@link #appendChild}. If <i>n</i> is already part of the DOM, it is first
+ * removed before being inserted.
+ *
+ * @throws Error if <i>r</i> is non-null and not a child of this node.
+ * @returns {pv.Dom.Node} the inserted child.
+ */
+pv.Dom.Node.prototype.insertBefore = function(n, r) {
+  if (!r) return this.appendChild(n);
+  var i = this.childNodes.indexOf(r);
+  if (i == -1) throw new Error("child not found");
+  if (n.parentNode) n.parentNode.removeChild(n);
+  n.parentNode = this;
+  n.nextSibling = r;
+  n.previousSibling = r.previousSibling;
+  if (r.previousSibling) {
+    r.previousSibling.nextSibling = n;
+  } else {
+    if (r == this.lastChild) this.lastChild = n;
+    this.firstChild = n;
+  }
+  this.childNodes.splice(i, 0, n);
+  return n;
+};
+
+/**
+ * Replaces the specified child <i>r</i> of this node with the node <i>n</i>. If
+ * <i>n</i> is already part of the DOM, it is first removed before being added.
+ *
+ * @throws Error if <i>r</i> is not a child of this node.
+ */
+pv.Dom.Node.prototype.replaceChild = function(n, r) {
+  var i = this.childNodes.indexOf(r);
+  if (i == -1) throw new Error("child not found");
+  if (n.parentNode) n.parentNode.removeChild(n);
+  n.parentNode = this;
+  n.nextSibling = r.nextSibling;
+  n.previousSibling = r.previousSibling;
+  if (r.previousSibling) r.previousSibling.nextSibling = n;
+  else this.firstChild = n;
+  if (r.nextSibling) r.nextSibling.previousSibling = n;
+  else this.lastChild = n;
+  this.childNodes[i] = n;
+  return r;
+};
+
+/**
+ * Visits each node in the tree in preorder traversal, applying the specified
+ * function <i>f</i>. The arguments to the function are:<ol>
+ *
+ * <li>The current node.
+ * <li>The current depth, starting at 0 for the root node.</ol>
+ *
+ * @param {function} f a function to apply to each node.
+ */
+pv.Dom.Node.prototype.visitBefore = function(f) {
+  function visit(n, i) {
+    f(n, i);
+    for (var c = n.firstChild; c; c = c.nextSibling) {
+      visit(c, i + 1);
+    }
+  }
+  visit(this, 0);
+};
+
+/**
+ * Visits each node in the tree in postorder traversal, applying the specified
+ * function <i>f</i>. The arguments to the function are:<ol>
+ *
+ * <li>The current node.
+ * <li>The current depth, starting at 0 for the root node.</ol>
+ *
+ * @param {function} f a function to apply to each node.
+ */
+pv.Dom.Node.prototype.visitAfter = function(f) {
+  function visit(n, i) {
+    for (var c = n.firstChild; c; c = c.nextSibling) {
+      visit(c, i + 1);
+    }
+    f(n, i);
+  }
+  visit(this, 0);
+};
+
+/**
+ * Sorts child nodes of this node, and all descendent nodes recursively, using
+ * the specified comparator function <tt>f</tt>. The comparator function is
+ * passed two nodes to compare.
+ *
+ * <p>Note: during the sort operation, the comparator function should not rely
+ * on the tree being well-formed; the values of <tt>previousSibling</tt> and
+ * <tt>nextSibling</tt> for the nodes being compared are not defined during the
+ * sort operation.
+ *
+ * @param {function} f a comparator function.
+ * @returns this.
+ */
+pv.Dom.Node.prototype.sort = function(f) {
+  if (this.firstChild) {
+    this.childNodes.sort(f);
+    var p = this.firstChild = this.childNodes[0], c;
+    delete p.previousSibling;
+    for (var i = 1; i < this.childNodes.length; i++) {
+      p.sort(f);
+      c = this.childNodes[i];
+      c.previousSibling = p;
+      p = p.nextSibling = c;
+    }
+    this.lastChild = p;
+    delete p.nextSibling;
+    p.sort(f);
+  }
+  return this;
+};
+
+/**
+ * Reverses all sibling nodes.
+ *
+ * @returns this.
+ */
+pv.Dom.Node.prototype.reverse = function() {
+  var childNodes = [];
+  this.visitAfter(function(n) {
+      while (n.lastChild) childNodes.push(n.removeChild(n.lastChild));
+      for (var c; c = childNodes.pop();) n.insertBefore(c, n.firstChild);
+    });
+  return this;
+};
+
+/** Returns all descendants of this node in preorder traversal. */
+pv.Dom.Node.prototype.nodes = function() {
+  var array = [];
+
+  /** @private */
+  function flatten(node) {
+    array.push(node);
+    node.childNodes.forEach(flatten);
+  }
+
+  flatten(this, array);
+  return array;
+};
+
+/**
+ * Toggles the child nodes of this node. If this node is not yet toggled, this
+ * method removes all child nodes and appends them to a new <tt>toggled</tt>
+ * array attribute on this node. Otherwise, if this node is toggled, this method
+ * re-adds all toggled child nodes and deletes the <tt>toggled</tt> attribute.
+ *
+ * <p>This method has no effect if the node has no child nodes.
+ *
+ * @param {boolean} [recursive] whether the toggle should apply to descendants.
+ */
+pv.Dom.Node.prototype.toggle = function(recursive) {
+  if (recursive) return this.toggled
+      ? this.visitBefore(function(n) { if (n.toggled) n.toggle(); })
+      : this.visitAfter(function(n) { if (!n.toggled) n.toggle(); });
+  var n = this;
+  if (n.toggled) {
+    for (var c; c = n.toggled.pop();) n.appendChild(c);
+    delete n.toggled;
+  } else if (n.lastChild) {
+    n.toggled = [];
+    while (n.lastChild) n.toggled.push(n.removeChild(n.lastChild));
+  }
+};
+
+/**
+ * Given a flat array of values, returns a simple DOM with each value wrapped by
+ * a node that is a child of the root node.
+ *
+ * @param {array} values.
+ * @returns {array} nodes.
+ */
+pv.nodes = function(values) {
+  var root = new pv.Dom.Node();
+  for (var i = 0; i < values.length; i++) {
+    root.appendChild(new pv.Dom.Node(values[i]));
+  }
+  return root.nodes();
+};
+/**
+ * Returns a {@link pv.Tree} operator for the specified array. This is a
+ * convenience factory method, equivalent to <tt>new pv.Tree(array)</tt>.
+ *
+ * @see pv.Tree
+ * @param {array} array an array from which to construct a tree.
+ * @returns {pv.Tree} a tree operator for the specified array.
+ */
+pv.tree = function(array) {
+  return new pv.Tree(array);
+};
+
+/**
+ * Constructs a tree operator for the specified array. This constructor should
+ * not be invoked directly; use {@link pv.tree} instead.
+ *
+ * @class Represents a tree operator for the specified array. The tree operator
+ * allows a hierarchical map to be constructed from an array; it is similar to
+ * the {@link pv.Nest} operator, except the hierarchy is derived dynamically
+ * from the array elements.
+ *
+ * <p>For example, given an array of size information for ActionScript classes:
+ *
+ * <pre>{ name: "flare.flex.FlareVis", size: 4116 },
+ * { name: "flare.physics.DragForce", size: 1082 },
+ * { name: "flare.physics.GravityForce", size: 1336 }, ...</pre>
+ *
+ * To facilitate visualization, it may be useful to nest the elements by their
+ * package hierarchy:
+ *
+ * <pre>var tree = pv.tree(classes)
+ *     .keys(function(d) d.name.split("."))
+ *     .map();</pre>
+ *
+ * The resulting tree is:
+ *
+ * <pre>{ flare: {
+ *     flex: {
+ *       FlareVis: {
+ *         name: "flare.flex.FlareVis",
+ *         size: 4116 } },
+ *     physics: {
+ *       DragForce: {
+ *         name: "flare.physics.DragForce",
+ *         size: 1082 },
+ *       GravityForce: {
+ *         name: "flare.physics.GravityForce",
+ *         size: 1336 } },
+ *     ... } }</pre>
+ *
+ * By specifying a value function,
+ *
+ * <pre>var tree = pv.tree(classes)
+ *     .keys(function(d) d.name.split("."))
+ *     .value(function(d) d.size)
+ *     .map();</pre>
+ *
+ * we can further eliminate redundant data:
+ *
+ * <pre>{ flare: {
+ *     flex: {
+ *       FlareVis: 4116 },
+ *     physics: {
+ *       DragForce: 1082,
+ *       GravityForce: 1336 },
+ *   ... } }</pre>
+ *
+ * For visualizations with large data sets, performance improvements may be seen
+ * by storing the data in a tree format, and then flattening it into an array at
+ * runtime with {@link pv.Flatten}.
+ *
+ * @param {array} array an array from which to construct a tree.
+ */
+pv.Tree = function(array) {
+  this.array = array;
+};
+
+/**
+ * Assigns a <i>keys</i> function to this operator; required. The keys function
+ * returns an array of <tt>string</tt>s for each element in the associated
+ * array; these keys determine how the elements are nested in the tree. The
+ * returned keys should be unique for each element in the array; otherwise, the
+ * behavior of this operator is undefined.
+ *
+ * @param {function} k the keys function.
+ * @returns {pv.Tree} this.
+ */
+pv.Tree.prototype.keys = function(k) {
+  this.k = k;
+  return this;
+};
+
+/**
+ * Assigns a <i>value</i> function to this operator; optional. The value
+ * function specifies an optional transformation of the element in the array
+ * before it is inserted into the map. If no value function is specified, it is
+ * equivalent to using the identity function.
+ *
+ * @param {function} k the value function.
+ * @returns {pv.Tree} this.
+ */
+pv.Tree.prototype.value = function(v) {
+  this.v = v;
+  return this;
+};
+
+/**
+ * Returns a hierarchical map of values. The hierarchy is determined by the keys
+ * function; the values in the map are determined by the value function.
+ *
+ * @returns a hierarchical map of values.
+ */
+pv.Tree.prototype.map = function() {
+  var map = {}, o = {};
+  for (var i = 0; i < this.array.length; i++) {
+    o.index = i;
+    var value = this.array[i], keys = this.k.call(o, value), node = map;
+    for (var j = 0; j < keys.length - 1; j++) {
+      node = node[keys[j]] || (node[keys[j]] = {});
+    }
+    node[keys[j]] = this.v ? this.v.call(o, value) : value;
+  }
+  return map;
+};
+/**
+ * Returns a {@link pv.Nest} operator for the specified array. This is a
+ * convenience factory method, equivalent to <tt>new pv.Nest(array)</tt>.
+ *
+ * @see pv.Nest
+ * @param {array} array an array of elements to nest.
+ * @returns {pv.Nest} a nest operator for the specified array.
+ */
+pv.nest = function(array) {
+  return new pv.Nest(array);
+};
+
+/**
+ * Constructs a nest operator for the specified array. This constructor should
+ * not be invoked directly; use {@link pv.nest} instead.
+ *
+ * @class Represents a {@link Nest} operator for the specified array. Nesting
+ * allows elements in an array to be grouped into a hierarchical tree
+ * structure. The levels in the tree are specified by <i>key</i> functions. The
+ * leaf nodes of the tree can be sorted by value, while the internal nodes can
+ * be sorted by key. Finally, the tree can be returned either has a
+ * multidimensional array via {@link #entries}, or as a hierarchical map via
+ * {@link #map}. The {@link #rollup} routine similarly returns a map, collapsing
+ * the elements in each leaf node using a summary function.
+ *
+ * <p>For example, consider the following tabular data structure of Barley
+ * yields, from various sites in Minnesota during 1931-2:
+ *
+ * <pre>{ yield: 27.00, variety: "Manchuria", year: 1931, site: "University Farm" },
+ * { yield: 48.87, variety: "Manchuria", year: 1931, site: "Waseca" },
+ * { yield: 27.43, variety: "Manchuria", year: 1931, site: "Morris" }, ...</pre>
+ *
+ * To facilitate visualization, it may be useful to nest the elements first by
+ * year, and then by variety, as follows:
+ *
+ * <pre>var nest = pv.nest(yields)
+ *     .key(function(d) d.year)
+ *     .key(function(d) d.variety)
+ *     .entries();</pre>
+ *
+ * This returns a nested array. Each element of the outer array is a key-values
+ * pair, listing the values for each distinct key:
+ *
+ * <pre>{ key: 1931, values: [
+ *   { key: "Manchuria", values: [
+ *       { yield: 27.00, variety: "Manchuria", year: 1931, site: "University Farm" },
+ *       { yield: 48.87, variety: "Manchuria", year: 1931, site: "Waseca" },
+ *       { yield: 27.43, variety: "Manchuria", year: 1931, site: "Morris" },
+ *       ...
+ *     ] },
+ *   { key: "Glabron", values: [
+ *       { yield: 43.07, variety: "Glabron", year: 1931, site: "University Farm" },
+ *       { yield: 55.20, variety: "Glabron", year: 1931, site: "Waseca" },
+ *       ...
+ *     ] },
+ *   ] },
+ * { key: 1932, values: ... }</pre>
+ *
+ * Further details, including sorting and rollup, is provided below on the
+ * corresponding methods.
+ *
+ * @param {array} array an array of elements to nest.
+ */
+pv.Nest = function(array) {
+  this.array = array;
+  this.keys = [];
+};
+
+/**
+ * Nests using the specified key function. Multiple keys may be added to the
+ * nest; the array elements will be nested in the order keys are specified.
+ *
+ * @param {function} key a key function; must return a string or suitable map
+ * key.
+ * @returns {pv.Nest} this.
+ */
+pv.Nest.prototype.key = function(key) {
+  this.keys.push(key);
+  return this;
+};
+
+/**
+ * Sorts the previously-added keys. The natural sort order is used by default
+ * (see {@link pv.naturalOrder}); if an alternative order is desired,
+ * <tt>order</tt> should be a comparator function. If this method is not called
+ * (i.e., keys are <i>unsorted</i>), keys will appear in the order they appear
+ * in the underlying elements array. For example,
+ *
+ * <pre>pv.nest(yields)
+ *     .key(function(d) d.year)
+ *     .key(function(d) d.variety)
+ *     .sortKeys()
+ *     .entries()</pre>
+ *
+ * groups yield data by year, then variety, and sorts the variety groups
+ * lexicographically (since the variety attribute is a string).
+ *
+ * <p>Key sort order is only used in conjunction with {@link #entries}, which
+ * returns an array of key-values pairs. If the nest is used to construct a
+ * {@link #map} instead, keys are unsorted.
+ *
+ * @param {function} [order] an optional comparator function.
+ * @returns {pv.Nest} this.
+ */
+pv.Nest.prototype.sortKeys = function(order) {
+  this.keys[this.keys.length - 1].order = order || pv.naturalOrder;
+  return this;
+};
+
+/**
+ * Sorts the leaf values. The natural sort order is used by default (see
+ * {@link pv.naturalOrder}); if an alternative order is desired, <tt>order</tt>
+ * should be a comparator function. If this method is not called (i.e., values
+ * are <i>unsorted</i>), values will appear in the order they appear in the
+ * underlying elements array. For example,
+ *
+ * <pre>pv.nest(yields)
+ *     .key(function(d) d.year)
+ *     .key(function(d) d.variety)
+ *     .sortValues(function(a, b) a.yield - b.yield)
+ *     .entries()</pre>
+ *
+ * groups yield data by year, then variety, and sorts the values for each
+ * variety group by yield.
+ *
+ * <p>Value sort order, unlike keys, applies to both {@link #entries} and
+ * {@link #map}. It has no effect on {@link #rollup}.
+ *
+ * @param {function} [order] an optional comparator function.
+ * @returns {pv.Nest} this.
+ */
+pv.Nest.prototype.sortValues = function(order) {
+  this.order = order || pv.naturalOrder;
+  return this;
+};
+
+/**
+ * Returns a hierarchical map of values. Each key adds one level to the
+ * hierarchy. With only a single key, the returned map will have a key for each
+ * distinct value of the key function; the correspond value with be an array of
+ * elements with that key value. If a second key is added, this will be a nested
+ * map. For example:
+ *
+ * <pre>pv.nest(yields)
+ *     .key(function(d) d.variety)
+ *     .key(function(d) d.site)
+ *     .map()</pre>
+ *
+ * returns a map <tt>m</tt> such that <tt>m[variety][site]</tt> is an array, a subset of
+ * <tt>yields</tt>, with each element having the given variety and site.
+ *
+ * @returns a hierarchical map of values.
+ */
+pv.Nest.prototype.map = function() {
+  var map = {}, values = [];
+
+  /* Build the map. */
+  for (var i, j = 0; j < this.array.length; j++) {
+    var x = this.array[j];
+    var m = map;
+    for (i = 0; i < this.keys.length - 1; i++) {
+      var k = this.keys[i](x);
+      if (!m[k]) m[k] = {};
+      m = m[k];
+    }
+    k = this.keys[i](x);
+    if (!m[k]) {
+      var a = [];
+      values.push(a);
+      m[k] = a;
+    }
+    m[k].push(x);
+  }
+
+  /* Sort each leaf array. */
+  if (this.order) {
+    for (var i = 0; i < values.length; i++) {
+      values[i].sort(this.order);
+    }
+  }
+
+  return map;
+};
+
+/**
+ * Returns a hierarchical nested array. This method is similar to
+ * {@link pv.entries}, but works recursively on the entire hierarchy. Rather
+ * than returning a map like {@link #map}, this method returns a nested
+ * array. Each element of the array has a <tt>key</tt> and <tt>values</tt>
+ * field. For leaf nodes, the <tt>values</tt> array will be a subset of the
+ * underlying elements array; for non-leaf nodes, the <tt>values</tt> array will
+ * contain more key-values pairs.
+ *
+ * <p>For an example usage, see the {@link Nest} constructor.
+ *
+ * @returns a hierarchical nested array.
+ */
+pv.Nest.prototype.entries = function() {
+
+  /** Recursively extracts the entries for the given map. */
+  function entries(map) {
+    var array = [];
+    for (var k in map) {
+      var v = map[k];
+      array.push({ key: k, values: (v instanceof Array) ? v : entries(v) });
+    };
+    return array;
+  }
+
+  /** Recursively sorts the values for the given key-values array. */
+  function sort(array, i) {
+    var o = this.keys[i].order;
+    if (o) array.sort(function(a, b) { return o(a.key, b.key); });
+    if (++i < this.keys.length) {
+      for (var j = 0; j < array.length; j++) {
+        sort.call(this, array[j].values, i);
+      }
+    }
+    return array;
+  }
+
+  return sort.call(this, entries(this.map()), 0);
+};
+
+/**
+ * Returns a rollup map. The behavior of this method is the same as
+ * {@link #map}, except that the leaf values are replaced with the return value
+ * of the specified rollup function <tt>f</tt>. For example,
+ *
+ * <pre>pv.nest(yields)
+ *      .key(function(d) d.site)
+ *      .rollup(function(v) pv.median(v, function(d) d.yield))</pre>
+ *
+ * first groups yield data by site, and then returns a map from site to median
+ * yield for the given site.
+ *
+ * @see #map
+ * @param {function} f a rollup function.
+ * @returns a hierarchical map, with the leaf values computed by <tt>f</tt>.
+ */
+pv.Nest.prototype.rollup = function(f) {
+
+  /** Recursively descends to the leaf nodes (arrays) and does rollup. */
+  function rollup(map) {
+    for (var key in map) {
+      var value = map[key];
+      if (value instanceof Array) {
+        map[key] = f(value);
+      } else {
+        rollup(value);
+      }
+    }
+    return map;
+  }
+
+  return rollup(this.map());
+};
+/**
+ * Returns a {@link pv.Flatten} operator for the specified map. This is a
+ * convenience factory method, equivalent to <tt>new pv.Flatten(map)</tt>.
+ *
+ * @see pv.Flatten
+ * @param map a map to flatten.
+ * @returns {pv.Flatten} a flatten operator for the specified map.
+ */
+pv.flatten = function(map) {
+  return new pv.Flatten(map);
+};
+
+/**
+ * Constructs a flatten operator for the specified map. This constructor should
+ * not be invoked directly; use {@link pv.flatten} instead.
+ *
+ * @class Represents a flatten operator for the specified array. Flattening
+ * allows hierarchical maps to be flattened into an array. The levels in the
+ * input tree are specified by <i>key</i> functions.
+ *
+ * <p>For example, consider the following hierarchical data structure of Barley
+ * yields, from various sites in Minnesota during 1931-2:
+ *
+ * <pre>{ 1931: {
+ *     Manchuria: {
+ *       "University Farm": 27.00,
+ *       "Waseca": 48.87,
+ *       "Morris": 27.43,
+ *       ... },
+ *     Glabron: {
+ *       "University Farm": 43.07,
+ *       "Waseca": 55.20,
+ *       ... } },
+ *   1932: {
+ *     ... } }</pre>
+ *
+ * To facilitate visualization, it may be useful to flatten the tree into a
+ * tabular array:
+ *
+ * <pre>var array = pv.flatten(yields)
+ *     .key("year")
+ *     .key("variety")
+ *     .key("site")
+ *     .key("yield")
+ *     .array();</pre>
+ *
+ * This returns an array of object elements. Each element in the array has
+ * attributes corresponding to this flatten operator's keys:
+ *
+ * <pre>{ site: "University Farm", variety: "Manchuria", year: 1931, yield: 27 },
+ * { site: "Waseca", variety: "Manchuria", year: 1931, yield: 48.87 },
+ * { site: "Morris", variety: "Manchuria", year: 1931, yield: 27.43 },
+ * { site: "University Farm", variety: "Glabron", year: 1931, yield: 43.07 },
+ * { site: "Waseca", variety: "Glabron", year: 1931, yield: 55.2 }, ...</pre>
+ *
+ * <p>The flatten operator is roughly the inverse of the {@link pv.Nest} and
+ * {@link pv.Tree} operators.
+ *
+ * @param map a map to flatten.
+ */
+pv.Flatten = function(map) {
+  this.map = map;
+  this.keys = [];
+};
+
+/**
+ * Flattens using the specified key function. Multiple keys may be added to the
+ * flatten; the tiers of the underlying tree must correspond to the specified
+ * keys, in order. The order of the returned array is undefined; however, you
+ * can easily sort it.
+ *
+ * @param {string} key the key name.
+ * @param {function} [f] an optional value map function.
+ * @returns {pv.Nest} this.
+ */
+pv.Flatten.prototype.key = function(key, f) {
+  this.keys.push({name: key, value: f});
+  delete this.$leaf;
+  return this;
+};
+
+/**
+ * Flattens using the specified leaf function. This is an alternative to
+ * specifying an explicit set of keys; the tiers of the underlying tree will be
+ * determined dynamically by recursing on the values, and the resulting keys
+ * will be stored in the entries <tt>keys</tt> attribute. The leaf function must
+ * return true for leaves, and false for internal nodes.
+ *
+ * @param {function} f a leaf function.
+ * @returns {pv.Nest} this.
+ */
+pv.Flatten.prototype.leaf = function(f) {
+  this.keys.length = 0;
+  this.$leaf = f;
+  return this;
+};
+
+/**
+ * Returns the flattened array. Each entry in the array is an object; each
+ * object has attributes corresponding to this flatten operator's keys.
+ *
+ * @returns an array of elements from the flattened map.
+ */
+pv.Flatten.prototype.array = function() {
+  var entries = [], stack = [], keys = this.keys, leaf = this.$leaf;
+
+  /* Recursively visit using the leaf function. */
+  if (leaf) {
+    function recurse(value, i) {
+      if (leaf(value)) {
+        entries.push({keys: stack.slice(), value: value});
+      } else {
+        for (var key in value) {
+          stack.push(key);
+          recurse(value[key], i + 1);
+          stack.pop();
+        }
+      }
+    }
+    recurse(this.map, 0);
+    return entries;
+  }
+
+  /* Recursively visits the specified value. */
+  function visit(value, i) {
+    if (i < keys.length - 1) {
+      for (var key in value) {
+        stack.push(key);
+        visit(value[key], i + 1);
+        stack.pop();
+      }
+    } else {
+      entries.push(stack.concat(value));
+    }
+  }
+
+  visit(this.map, 0);
+  return entries.map(function(stack) {
+      var m = {};
+      for (var i = 0; i < keys.length; i++) {
+        var k = keys[i], v = stack[i];
+        m[k.name] = k.value ? k.value.call(null, v) : v;
+      }
+      return m;
+    });
+};
+/**
+ * Returns a {@link pv.Vector} for the specified <i>x</i> and <i>y</i>
+ * coordinate. This is a convenience factory method, equivalent to <tt>new
+ * pv.Vector(x, y)</tt>.
+ *
+ * @see pv.Vector
+ * @param {number} x the <i>x</i> coordinate.
+ * @param {number} y the <i>y</i> coordinate.
+ * @returns {pv.Vector} a vector for the specified coordinates.
+ */
+pv.vector = function(x, y) {
+  return new pv.Vector(x, y);
+};
+
+/**
+ * Constructs a {@link pv.Vector} for the specified <i>x</i> and <i>y</i>
+ * coordinate. This constructor should not be invoked directly; use
+ * {@link pv.vector} instead.
+ *
+ * @class Represents a two-dimensional vector; a 2-tuple <i>&#x27e8;x,
+ * y&#x27e9;</i>. The intent of this class is to simplify vector math. Note that
+ * in performance-sensitive cases it may be more efficient to represent 2D
+ * vectors as simple objects with <tt>x</tt> and <tt>y</tt> attributes, rather
+ * than using instances of this class.
+ *
+ * @param {number} x the <i>x</i> coordinate.
+ * @param {number} y the <i>y</i> coordinate.
+ */
+pv.Vector = function(x, y) {
+  this.x = x;
+  this.y = y;
+};
+
+/**
+ * Returns a vector perpendicular to this vector: <i>&#x27e8;-y, x&#x27e9;</i>.
+ *
+ * @returns {pv.Vector} a perpendicular vector.
+ */
+pv.Vector.prototype.perp = function() {
+  return new pv.Vector(-this.y, this.x);
+};
+
+/**
+ * Returns a normalized copy of this vector: a vector with the same direction,
+ * but unit length. If this vector has zero length this method returns a copy of
+ * this vector.
+ *
+ * @returns {pv.Vector} a unit vector.
+ */
+pv.Vector.prototype.norm = function() {
+  var l = this.length();
+  return this.times(l ? (1 / l) : 1);
+};
+
+/**
+ * Returns the magnitude of this vector, defined as <i>sqrt(x * x + y * y)</i>.
+ *
+ * @returns {number} a length.
+ */
+pv.Vector.prototype.length = function() {
+  return Math.sqrt(this.x * this.x + this.y * this.y);
+};
+
+/**
+ * Returns a scaled copy of this vector: <i>&#x27e8;x * k, y * k&#x27e9;</i>.
+ * To perform the equivalent divide operation, use <i>1 / k</i>.
+ *
+ * @param {number} k the scale factor.
+ * @returns {pv.Vector} a scaled vector.
+ */
+pv.Vector.prototype.times = function(k) {
+  return new pv.Vector(this.x * k, this.y * k);
+};
+
+/**
+ * Returns this vector plus the vector <i>v</i>: <i>&#x27e8;x + v.x, y +
+ * v.y&#x27e9;</i>. If only one argument is specified, it is interpreted as the
+ * vector <i>v</i>.
+ *
+ * @param {number} x the <i>x</i> coordinate to add.
+ * @param {number} y the <i>y</i> coordinate to add.
+ * @returns {pv.Vector} a new vector.
+ */
+pv.Vector.prototype.plus = function(x, y) {
+  return (arguments.length == 1)
+      ? new pv.Vector(this.x + x.x, this.y + x.y)
+      : new pv.Vector(this.x + x, this.y + y);
+};
+
+/**
+ * Returns this vector minus the vector <i>v</i>: <i>&#x27e8;x - v.x, y -
+ * v.y&#x27e9;</i>. If only one argument is specified, it is interpreted as the
+ * vector <i>v</i>.
+ *
+ * @param {number} x the <i>x</i> coordinate to subtract.
+ * @param {number} y the <i>y</i> coordinate to subtract.
+ * @returns {pv.Vector} a new vector.
+ */
+pv.Vector.prototype.minus = function(x, y) {
+  return (arguments.length == 1)
+      ? new pv.Vector(this.x - x.x, this.y - x.y)
+      : new pv.Vector(this.x - x, this.y - y);
+};
+
+/**
+ * Returns the dot product of this vector and the vector <i>v</i>: <i>x * v.x +
+ * y * v.y</i>. If only one argument is specified, it is interpreted as the
+ * vector <i>v</i>.
+ *
+ * @param {number} x the <i>x</i> coordinate to dot.
+ * @param {number} y the <i>y</i> coordinate to dot.
+ * @returns {number} a dot product.
+ */
+pv.Vector.prototype.dot = function(x, y) {
+  return (arguments.length == 1)
+      ? this.x * x.x + this.y * x.y
+      : this.x * x + this.y * y;
+};
+/**
+ * Returns a new identity transform.
+ *
+ * @class Represents a transformation matrix. The transformation matrix is
+ * limited to expressing translate and uniform scale transforms only; shearing,
+ * rotation, general affine, and other transforms are not supported.
+ *
+ * <p>The methods on this class treat the transform as immutable, returning a
+ * copy of the transformation matrix with the specified transform applied. Note,
+ * alternatively, that the matrix fields can be get and set directly.
+ */
+pv.Transform = function() {};
+pv.Transform.prototype = {k: 1, x: 0, y: 0};
+
+/**
+ * The scale magnitude; defaults to 1.
+ *
+ * @type number
+ * @name pv.Transform.prototype.k
+ */
+
+/**
+ * The x-offset; defaults to 0.
+ *
+ * @type number
+ * @name pv.Transform.prototype.x
+ */
+
+/**
+ * The y-offset; defaults to 0.
+ *
+ * @type number
+ * @name pv.Transform.prototype.y
+ */
+
+/**
+ * @private The identity transform.
+ *
+ * @type pv.Transform
+ */
+pv.Transform.identity = new pv.Transform();
+
+// k 0 x   1 0 a   k 0 ka+x
+// 0 k y * 0 1 b = 0 k kb+y
+// 0 0 1   0 0 1   0 0 1
+
+/**
+ * Returns a translated copy of this transformation matrix.
+ *
+ * @param {number} x the x-offset.
+ * @param {number} y the y-offset.
+ * @returns {pv.Transform} the translated transformation matrix.
+ */
+pv.Transform.prototype.translate = function(x, y) {
+  var v = new pv.Transform();
+  v.k = this.k;
+  v.x = this.k * x + this.x;
+  v.y = this.k * y + this.y;
+  return v;
+};
+
+// k 0 x   d 0 0   kd  0 x
+// 0 k y * 0 d 0 =  0 kd y
+// 0 0 1   0 0 1    0  0 1
+
+/**
+ * Returns a scaled copy of this transformation matrix.
+ *
+ * @param {number} k
+ * @returns {pv.Transform} the scaled transformation matrix.
+ */
+pv.Transform.prototype.scale = function(k) {
+  var v = new pv.Transform();
+  v.k = this.k * k;
+  v.x = this.x;
+  v.y = this.y;
+  return v;
+};
+
+/**
+ * Returns the inverse of this transformation matrix.
+ *
+ * @returns {pv.Transform} the inverted transformation matrix.
+ */
+pv.Transform.prototype.invert = function() {
+  var v = new pv.Transform(), k = 1 / this.k;
+  v.k = k;
+  v.x = -this.x * k;
+  v.y = -this.y * k;
+  return v;
+};
+
+// k 0 x   d 0 a   kd  0 ka+x
+// 0 k y * 0 d b =  0 kd kb+y
+// 0 0 1   0 0 1    0  0    1
+
+/**
+ * Returns this matrix post-multiplied by the specified matrix <i>m</i>.
+ *
+ * @param {pv.Transform} m
+ * @returns {pv.Transform} the post-multiplied transformation matrix.
+ */
+pv.Transform.prototype.times = function(m) {
+  var v = new pv.Transform();
+  v.k = this.k * m.k;
+  v.x = this.k * m.x + this.x;
+  v.y = this.k * m.y + this.y;
+  return v;
+};
+/**
+ * Abstract; see the various scale implementations.
+ *
+ * @class Represents a scale; a function that performs a transformation from
+ * data domain to visual range. For quantitative and quantile scales, the domain
+ * is expressed as numbers; for ordinal scales, the domain is expressed as
+ * strings (or equivalently objects with unique string representations). The
+ * "visual range" may correspond to pixel space, colors, font sizes, and the
+ * like.
+ *
+ * <p>Note that scales are functions, and thus can be used as properties
+ * directly, assuming that the data associated with a mark is a number. While
+ * this is convenient for single-use scales, frequently it is desirable to
+ * define scales globally:
+ *
+ * <pre>var y = pv.Scale.linear(0, 100).range(0, 640);</pre>
+ *
+ * The <tt>y</tt> scale can now be equivalently referenced within a property:
+ *
+ * <pre>    .height(function(d) y(d))</pre>
+ *
+ * Alternatively, if the data are not simple numbers, the appropriate value can
+ * be passed to the <tt>y</tt> scale (e.g., <tt>d.foo</tt>). The {@link #by}
+ * method similarly allows the data to be mapped to a numeric value before
+ * performing the linear transformation.
+ *
+ * @see pv.Scale.quantitative
+ * @see pv.Scale.quantile
+ * @see pv.Scale.ordinal
+ * @extends function
+ */
+pv.Scale = function() {};
+
+/**
+ * @private Returns a function that interpolators from the start value to the
+ * end value, given a parameter <i>t</i> in [0, 1].
+ *
+ * @param start the start value.
+ * @param end the end value.
+ */
+pv.Scale.interpolator = function(start, end) {
+  if (typeof start == "number") {
+    return function(t) {
+      return t * (end - start) + start;
+    };
+  }
+
+  /* For now, assume color. */
+  start = pv.color(start).rgb();
+  end = pv.color(end).rgb();
+  return function(t) {
+    var a = start.a * (1 - t) + end.a * t;
+    if (a < 1e-5) a = 0; // avoid scientific notation
+    return (start.a == 0) ? pv.rgb(end.r, end.g, end.b, a)
+        : ((end.a == 0) ? pv.rgb(start.r, start.g, start.b, a)
+        : pv.rgb(
+            Math.round(start.r * (1 - t) + end.r * t),
+            Math.round(start.g * (1 - t) + end.g * t),
+            Math.round(start.b * (1 - t) + end.b * t), a));
+  };
+};
+
+/**
+ * Returns a view of this scale by the specified accessor function <tt>f</tt>.
+ * Given a scale <tt>y</tt>, <tt>y.by(function(d) d.foo)</tt> is equivalent to
+ * <tt>function(d) y(d.foo)</tt>.
+ *
+ * <p>This method is provided for convenience, such that scales can be
+ * succinctly defined inline. For example, given an array of data elements that
+ * have a <tt>score</tt> attribute with the domain [0, 1], the height property
+ * could be specified as:
+ *
+ * <pre>    .height(pv.Scale.linear().range(0, 480).by(function(d) d.score))</pre>
+ *
+ * This is equivalent to:
+ *
+ * <pre>    .height(function(d) d.score * 480)</pre>
+ *
+ * This method should be used judiciously; it is typically more clear to invoke
+ * the scale directly, passing in the value to be scaled.
+ *
+ * @function
+ * @name pv.Scale.prototype.by
+ * @param {function} f an accessor function.
+ * @returns {pv.Scale} a view of this scale by the specified accessor function.
+ */
+/**
+ * Returns a default quantitative, linear, scale for the specified domain. The
+ * arguments to this constructor are optional, and equivalent to calling
+ * {@link #domain}. The default domain and range are [0,1].
+ *
+ * <p>This constructor is typically not used directly; see one of the
+ * quantitative scale implementations instead.
+ *
+ * @class Represents an abstract quantitative scale; a function that performs a
+ * numeric transformation. This class is typically not used directly; see one of
+ * the quantitative scale implementations (linear, log, root, etc.)
+ * instead. <style type="text/css">sub{line-height:0}</style> A quantitative
+ * scale represents a 1-dimensional transformation from a numeric domain of
+ * input data [<i>d<sub>0</sub></i>, <i>d<sub>1</sub></i>] to a numeric range of
+ * pixels [<i>r<sub>0</sub></i>, <i>r<sub>1</sub></i>]. In addition to
+ * readability, scales offer several useful features:
+ *
+ * <p>1. The range can be expressed in colors, rather than pixels. For example:
+ *
+ * <pre>    .fillStyle(pv.Scale.linear(0, 100).range("red", "green"))</pre>
+ *
+ * will fill the marks "red" on an input value of 0, "green" on an input value
+ * of 100, and some color in-between for intermediate values.
+ *
+ * <p>2. The domain and range can be subdivided for a non-uniform
+ * transformation. For example, you may want a diverging color scale that is
+ * increasingly red for negative values, and increasingly green for positive
+ * values:
+ *
+ * <pre>    .fillStyle(pv.Scale.linear(-1, 0, 1).range("red", "white", "green"))</pre>
+ *
+ * The domain can be specified as a series of <i>n</i> monotonically-increasing
+ * values; the range must also be specified as <i>n</i> values, resulting in
+ * <i>n - 1</i> contiguous linear scales.
+ *
+ * <p>3. Quantitative scales can be inverted for interaction. The
+ * {@link #invert} method takes a value in the output range, and returns the
+ * corresponding value in the input domain. This is frequently used to convert
+ * the mouse location (see {@link pv.Mark#mouse}) to a value in the input
+ * domain. Note that inversion is only supported for numeric ranges, and not
+ * colors.
+ *
+ * <p>4. A scale can be queried for reasonable "tick" values. The {@link #ticks}
+ * method provides a convenient way to get a series of evenly-spaced rounded
+ * values in the input domain. Frequently these are used in conjunction with
+ * {@link pv.Rule} to display tick marks or grid lines.
+ *
+ * <p>5. A scale can be "niced" to extend the domain to suitable rounded
+ * numbers. If the minimum and maximum of the domain are messy because they are
+ * derived from data, you can use {@link #nice} to round these values down and
+ * up to even numbers.
+ *
+ * @param {number...} domain... optional domain values.
+ * @see pv.Scale.linear
+ * @see pv.Scale.log
+ * @see pv.Scale.root
+ * @extends pv.Scale
+ */
+pv.Scale.quantitative = function() {
+  var d = [0, 1], // default domain
+      l = [0, 1], // default transformed domain
+      r = [0, 1], // default range
+      i = [pv.identity], // default interpolators
+      type = Number, // default type
+      n = false, // whether the domain is negative
+      f = pv.identity, // default forward transform
+      g = pv.identity, // default inverse transform
+      tickFormat = String; // default tick formatting function
+
+  /** @private */
+  function newDate(x) {
+    return new Date(x);
+  }
+
+  /** @private */
+  function scale(x) {
+    var j = pv.search(d, x);
+    if (j < 0) j = -j - 2;
+    j = Math.max(0, Math.min(i.length - 1, j));
+    return i[j]((f(x) - l[j]) / (l[j + 1] - l[j]));
+  }
+
+  /** @private */
+  scale.transform = function(forward, inverse) {
+    /** @ignore */ f = function(x) { return n ? -forward(-x) : forward(x); };
+    /** @ignore */ g = function(y) { return n ? -inverse(-y) : inverse(y); };
+    l = d.map(f);
+    return this;
+  };
+
+  /**
+   * Sets or gets the input domain. This method can be invoked several ways:
+   *
+   * <p>1. <tt>domain(min, ..., max)</tt>
+   *
+   * <p>Specifying the domain as a series of numbers is the most explicit and
+   * recommended approach. Most commonly, two numbers are specified: the minimum
+   * and maximum value. However, for a diverging scale, or other subdivided
+   * non-uniform scales, multiple values can be specified. Values can be derived
+   * from data using {@link pv.min} and {@link pv.max}. For example:
+   *
+   * <pre>    .domain(0, pv.max(array))</pre>
+   *
+   * An alternative method for deriving minimum and maximum values from data
+   * follows.
+   *
+   * <p>2. <tt>domain(array, minf, maxf)</tt>
+   *
+   * <p>When both the minimum and maximum value are derived from data, the
+   * arguments to the <tt>domain</tt> method can be specified as the array of
+   * data, followed by zero, one or two accessor functions. For example, if the
+   * array of data is just an array of numbers:
+   *
+   * <pre>    .domain(array)</pre>
+   *
+   * On the other hand, if the array elements are objects representing stock
+   * values per day, and the domain should consider the stock's daily low and
+   * daily high:
+   *
+   * <pre>    .domain(array, function(d) d.low, function(d) d.high)</pre>
+   *
+   * The first method of setting the domain is preferred because it is more
+   * explicit; setting the domain using this second method should be used only
+   * if brevity is required.
+   *
+   * <p>3. <tt>domain()</tt>
+   *
+   * <p>Invoking the <tt>domain</tt> method with no arguments returns the
+   * current domain as an array of numbers.
+   *
+   * @function
+   * @name pv.Scale.quantitative.prototype.domain
+   * @param {number...} domain... domain values.
+   * @returns {pv.Scale.quantitative} <tt>this</tt>, or the current domain.
+   */
+  scale.domain = function(array, min, max) {
+    if (arguments.length) {
+      var o; // the object we use to infer the domain type
+      if (array instanceof Array) {
+        if (arguments.length < 2) min = pv.identity;
+        if (arguments.length < 3) max = min;
+        o = array.length && min(array[0]);
+        d = array.length ? [pv.min(array, min), pv.max(array, max)] : [];
+      } else {
+        o = array;
+        d = Array.prototype.slice.call(arguments).map(Number);
+      }
+      if (!d.length) d = [-Infinity, Infinity];
+      else if (d.length == 1) d = [d[0], d[0]];
+      n = (d[0] || d[d.length - 1]) < 0;
+      l = d.map(f);
+      type = (o instanceof Date) ? newDate : Number;
+      return this;
+    }
+    return d.map(type);
+  };
+
+  /**
+   * Sets or gets the output range. This method can be invoked several ways:
+   *
+   * <p>1. <tt>range(min, ..., max)</tt>
+   *
+   * <p>The range may be specified as a series of numbers or colors. Most
+   * commonly, two numbers are specified: the minimum and maximum pixel values.
+   * For a color scale, values may be specified as {@link pv.Color}s or
+   * equivalent strings. For a diverging scale, or other subdivided non-uniform
+   * scales, multiple values can be specified. For example:
+   *
+   * <pre>    .range("red", "white", "green")</pre>
+   *
+   * <p>Currently, only numbers and colors are supported as range values. The
+   * number of range values must exactly match the number of domain values, or
+   * the behavior of the scale is undefined.
+   *
+   * <p>2. <tt>range()</tt>
+   *
+   * <p>Invoking the <tt>range</tt> method with no arguments returns the current
+   * range as an array of numbers or colors.
+   *
+   * @function
+   * @name pv.Scale.quantitative.prototype.range
+   * @param {...} range... range values.
+   * @returns {pv.Scale.quantitative} <tt>this</tt>, or the current range.
+   */
+  scale.range = function() {
+    if (arguments.length) {
+      r = Array.prototype.slice.call(arguments);
+      if (!r.length) r = [-Infinity, Infinity];
+      else if (r.length == 1) r = [r[0], r[0]];
+      i = [];
+      for (var j = 0; j < r.length - 1; j++) {
+        i.push(pv.Scale.interpolator(r[j], r[j + 1]));
+      }
+      return this;
+    }
+    return r;
+  };
+
+  /**
+   * Inverts the specified value in the output range, returning the
+   * corresponding value in the input domain. This is frequently used to convert
+   * the mouse location (see {@link pv.Mark#mouse}) to a value in the input
+   * domain. Inversion is only supported for numeric ranges, and not colors.
+   *
+   * <p>Note that this method does not do any rounding or bounds checking. If
+   * the input domain is discrete (e.g., an array index), the returned value
+   * should be rounded. If the specified <tt>y</tt> value is outside the range,
+   * the returned value may be equivalently outside the input domain.
+   *
+   * @function
+   * @name pv.Scale.quantitative.prototype.invert
+   * @param {number} y a value in the output range (a pixel location).
+   * @returns {number} a value in the input domain.
+   */
+  scale.invert = function(y) {
+    var j = pv.search(r, y);
+    if (j < 0) j = -j - 2;
+    j = Math.max(0, Math.min(i.length - 1, j));
+    return type(g(l[j] + (y - r[j]) / (r[j + 1] - r[j]) * (l[j + 1] - l[j])));
+  };
+
+  /**
+   * Returns an array of evenly-spaced, suitably-rounded values in the input
+   * domain. This method attempts to return between 5 and 10 tick values. These
+   * values are frequently used in conjunction with {@link pv.Rule} to display
+   * tick marks or grid lines.
+   *
+   * @function
+   * @name pv.Scale.quantitative.prototype.ticks
+   * @param {number} [m] optional number of desired ticks.
+   * @returns {number[]} an array input domain values to use as ticks.
+   */
+  scale.ticks = function(m) {
+    var start = d[0],
+        end = d[d.length - 1],
+        reverse = end < start,
+        min = reverse ? end : start,
+        max = reverse ? start : end,
+        span = max - min;
+
+    /* Special case: empty, invalid or infinite span. */
+    if (!span || !isFinite(span)) {
+      if (type == newDate) tickFormat = pv.Format.date("%x");
+      return [type(min)];
+    }
+
+    /* Special case: dates. */
+    if (type == newDate) {
+      /* Floor the date d given the precision p. */
+      function floor(d, p) {
+        switch (p) {
+          case 31536e6: d.setMonth(0);
+          case 2592e6: d.setDate(1);
+          case 6048e5: if (p == 6048e5) d.setDate(d.getDate() - d.getDay());
+          case 864e5: d.setHours(0);
+          case 36e5: d.setMinutes(0);
+          case 6e4: d.setSeconds(0);
+          case 1e3: d.setMilliseconds(0);
+        }
+      }
+
+      var precision, format, increment, step = 1;
+      if (span >= 3 * 31536e6) {
+        precision = 31536e6;
+        format = "%Y";
+        /** @ignore */ increment = function(d) { d.setFullYear(d.getFullYear() + step); };
+      } else if (span >= 3 * 2592e6) {
+        precision = 2592e6;
+        format = "%m/%Y";
+        /** @ignore */ increment = function(d) { d.setMonth(d.getMonth() + step); };
+      } else if (span >= 3 * 6048e5) {
+        precision = 6048e5;
+        format = "%m/%d";
+        /** @ignore */ increment = function(d) { d.setDate(d.getDate() + 7 * step); };
+      } else if (span >= 3 * 864e5) {
+        precision = 864e5;
+        format = "%m/%d";
+        /** @ignore */ increment = function(d) { d.setDate(d.getDate() + step); };
+      } else if (span >= 3 * 36e5) {
+        precision = 36e5;
+        format = "%I:%M %p";
+        /** @ignore */ increment = function(d) { d.setHours(d.getHours() + step); };
+      } else if (span >= 3 * 6e4) {
+        precision = 6e4;
+        format = "%I:%M %p";
+        /** @ignore */ increment = function(d) { d.setMinutes(d.getMinutes() + step); };
+      } else if (span >= 3 * 1e3) {
+        precision = 1e3;
+        format = "%I:%M:%S";
+        /** @ignore */ increment = function(d) { d.setSeconds(d.getSeconds() + step); };
+      } else {
+        precision = 1;
+        format = "%S.%Qs";
+        /** @ignore */ increment = function(d) { d.setTime(d.getTime() + step); };
+      }
+      tickFormat = pv.Format.date(format);
+
+      var date = new Date(min), dates = [];
+      floor(date, precision);
+
+      /* If we'd generate too many ticks, skip some!. */
+      var n = span / precision;
+      if (n > 10) {
+        switch (precision) {
+          case 36e5: {
+            step = (n > 20) ? 6 : 3;
+            date.setHours(Math.floor(date.getHours() / step) * step);
+            break;
+          }
+          case 2592e6: {
+            step = 3; // seasons
+            date.setMonth(Math.floor(date.getMonth() / step) * step);
+            break;
+          }
+          case 6e4: {
+            step = (n > 30) ? 15 : ((n > 15) ? 10 : 5);
+            date.setMinutes(Math.floor(date.getMinutes() / step) * step);
+            break;
+          }
+          case 1e3: {
+            step = (n > 90) ? 15 : ((n > 60) ? 10 : 5);
+            date.setSeconds(Math.floor(date.getSeconds() / step) * step);
+            break;
+          }
+          case 1: {
+            step = (n > 1000) ? 250 : ((n > 200) ? 100 : ((n > 100) ? 50 : ((n > 50) ? 25 : 5)));
+            date.setMilliseconds(Math.floor(date.getMilliseconds() / step) * step);
+            break;
+          }
+          default: {
+            step = pv.logCeil(n / 15, 10);
+            if (n / step < 2) step /= 5;
+            else if (n / step < 5) step /= 2;
+            date.setFullYear(Math.floor(date.getFullYear() / step) * step);
+            break;
+          }
+        }
+      }
+
+      while (true) {
+        increment(date);
+        if (date > max) break;
+        dates.push(new Date(date));
+      }
+      return reverse ? dates.reverse() : dates;
+    }
+
+    /* Normal case: numbers. */
+    if (!arguments.length) m = 10;
+    var step = pv.logFloor(span / m, 10),
+        err = m / (span / step);
+    if (err <= .15) step *= 10;
+    else if (err <= .35) step *= 5;
+    else if (err <= .75) step *= 2;
+    var start = Math.ceil(min / step) * step,
+        end = Math.floor(max / step) * step;
+    tickFormat = pv.Format.number()
+        .fractionDigits(Math.max(0, -Math.floor(pv.log(step, 10) + .01)));
+    var ticks = pv.range(start, end + step, step);
+    return reverse ? ticks.reverse() : ticks;
+  };
+
+  /**
+   * Formats the specified tick value using the appropriate precision, based on
+   * the step interval between tick marks. If {@link #ticks} has not been called,
+   * the argument is converted to a string, but no formatting is applied.
+   *
+   * @function
+   * @name pv.Scale.quantitative.prototype.tickFormat
+   * @param {number} t a tick value.
+   * @returns {string} a formatted tick value.
+   */
+  scale.tickFormat = function (t) { return tickFormat(t); };
+
+  /**
+   * "Nices" this scale, extending the bounds of the input domain to
+   * evenly-rounded values. Nicing is useful if the domain is computed
+   * dynamically from data, and may be irregular. For example, given a domain of
+   * [0.20147987687960267, 0.996679553296417], a call to <tt>nice()</tt> might
+   * extend the domain to [0.2, 1].
+   *
+   * <p>This method must be invoked each time after setting the domain.
+   *
+   * @function
+   * @name pv.Scale.quantitative.prototype.nice
+   * @returns {pv.Scale.quantitative} <tt>this</tt>.
+   */
+  scale.nice = function() {
+    if (d.length != 2) return this; // TODO support non-uniform domains
+    var start = d[0],
+        end = d[d.length - 1],
+        reverse = end < start,
+        min = reverse ? end : start,
+        max = reverse ? start : end,
+        span = max - min;
+
+    /* Special case: empty, invalid or infinite span. */
+    if (!span || !isFinite(span)) return this;
+
+    var step = Math.pow(10, Math.round(Math.log(span) / Math.log(10)) - 1);
+    d = [Math.floor(min / step) * step, Math.ceil(max / step) * step];
+    if (reverse) d.reverse();
+    l = d.map(f);
+    return this;
+  };
+
+  /**
+   * Returns a view of this scale by the specified accessor function <tt>f</tt>.
+   * Given a scale <tt>y</tt>, <tt>y.by(function(d) d.foo)</tt> is equivalent to
+   * <tt>function(d) y(d.foo)</tt>.
+   *
+   * <p>This method is provided for convenience, such that scales can be
+   * succinctly defined inline. For example, given an array of data elements
+   * that have a <tt>score</tt> attribute with the domain [0, 1], the height
+   * property could be specified as:
+   *
+   * <pre>    .height(pv.Scale.linear().range(0, 480).by(function(d) d.score))</pre>
+   *
+   * This is equivalent to:
+   *
+   * <pre>    .height(function(d) d.score * 480)</pre>
+   *
+   * This method should be used judiciously; it is typically more clear to
+   * invoke the scale directly, passing in the value to be scaled.
+   *
+   * @function
+   * @name pv.Scale.quantitative.prototype.by
+   * @param {function} f an accessor function.
+   * @returns {pv.Scale.quantitative} a view of this scale by the specified
+   * accessor function.
+   */
+  scale.by = function(f) {
+    function by() { return scale(f.apply(this, arguments)); }
+    for (var method in scale) by[method] = scale[method];
+    return by;
+  };
+
+  scale.domain.apply(scale, arguments);
+  return scale;
+};
+/**
+ * Returns a linear scale for the specified domain. The arguments to this
+ * constructor are optional, and equivalent to calling {@link #domain}.
+ * The default domain and range are [0,1].
+ *
+ * @class Represents a linear scale; a function that performs a linear
+ * transformation. <style type="text/css">sub{line-height:0}</style> Most
+ * commonly, a linear scale represents a 1-dimensional linear transformation
+ * from a numeric domain of input data [<i>d<sub>0</sub></i>,
+ * <i>d<sub>1</sub></i>] to a numeric range of pixels [<i>r<sub>0</sub></i>,
+ * <i>r<sub>1</sub></i>]. The equation for such a scale is:
+ *
+ * <blockquote><i>f(x) = (x - d<sub>0</sub>) / (d<sub>1</sub> - d<sub>0</sub>) *
+ * (r<sub>1</sub> - r<sub>0</sub>) + r<sub>0</sub></i></blockquote>
+ *
+ * For example, a linear scale from the domain [0, 100] to range [0, 640]:
+ *
+ * <blockquote><i>f(x) = (x - 0) / (100 - 0) * (640 - 0) + 0</i><br>
+ * <i>f(x) = x / 100 * 640</i><br>
+ * <i>f(x) = x * 6.4</i><br>
+ * </blockquote>
+ *
+ * Thus, saying
+ *
+ * <pre>    .height(function(d) d * 6.4)</pre>
+ *
+ * is identical to
+ *
+ * <pre>    .height(pv.Scale.linear(0, 100).range(0, 640))</pre>
+ *
+ * Note that the scale is itself a function, and thus can be used as a property
+ * directly, assuming that the data associated with a mark is a number. While
+ * this is convenient for single-use scales, frequently it is desirable to
+ * define scales globally:
+ *
+ * <pre>var y = pv.Scale.linear(0, 100).range(0, 640);</pre>
+ *
+ * The <tt>y</tt> scale can now be equivalently referenced within a property:
+ *
+ * <pre>    .height(function(d) y(d))</pre>
+ *
+ * Alternatively, if the data are not simple numbers, the appropriate value can
+ * be passed to the <tt>y</tt> scale (e.g., <tt>d.foo</tt>). The {@link #by}
+ * method similarly allows the data to be mapped to a numeric value before
+ * performing the linear transformation.
+ *
+ * @param {number...} domain... optional domain values.
+ * @extends pv.Scale.quantitative
+ */
+pv.Scale.linear = function() {
+  var scale = pv.Scale.quantitative();
+  scale.domain.apply(scale, arguments);
+  return scale;
+};
+/**
+ * Returns a log scale for the specified domain. The arguments to this
+ * constructor are optional, and equivalent to calling {@link #domain}.
+ * The default domain is [1,10] and the default range is [0,1].
+ *
+ * @class Represents a log scale. <style
+ * type="text/css">sub{line-height:0}</style> Most commonly, a log scale
+ * represents a 1-dimensional log transformation from a numeric domain of input
+ * data [<i>d<sub>0</sub></i>, <i>d<sub>1</sub></i>] to a numeric range of
+ * pixels [<i>r<sub>0</sub></i>, <i>r<sub>1</sub></i>]. The equation for such a
+ * scale is:
+ *
+ * <blockquote><i>f(x) = (log(x) - log(d<sub>0</sub>)) / (log(d<sub>1</sub>) -
+ * log(d<sub>0</sub>)) * (r<sub>1</sub> - r<sub>0</sub>) +
+ * r<sub>0</sub></i></blockquote>
+ *
+ * where <i>log(x)</i> represents the zero-symmetric logarthim of <i>x</i> using
+ * the scale's associated base (default: 10, see {@link pv.logSymmetric}). For
+ * example, a log scale from the domain [1, 100] to range [0, 640]:
+ *
+ * <blockquote><i>f(x) = (log(x) - log(1)) / (log(100) - log(1)) * (640 - 0) + 0</i><br>
+ * <i>f(x) = log(x) / 2 * 640</i><br>
+ * <i>f(x) = log(x) * 320</i><br>
+ * </blockquote>
+ *
+ * Thus, saying
+ *
+ * <pre>    .height(function(d) Math.log(d) * 138.974)</pre>
+ *
+ * is equivalent to
+ *
+ * <pre>    .height(pv.Scale.log(1, 100).range(0, 640))</pre>
+ *
+ * Note that the scale is itself a function, and thus can be used as a property
+ * directly, assuming that the data associated with a mark is a number. While
+ * this is convenient for single-use scales, frequently it is desirable to
+ * define scales globally:
+ *
+ * <pre>var y = pv.Scale.log(1, 100).range(0, 640);</pre>
+ *
+ * The <tt>y</tt> scale can now be equivalently referenced within a property:
+ *
+ * <pre>    .height(function(d) y(d))</pre>
+ *
+ * Alternatively, if the data are not simple numbers, the appropriate value can
+ * be passed to the <tt>y</tt> scale (e.g., <tt>d.foo</tt>). The {@link #by}
+ * method similarly allows the data to be mapped to a numeric value before
+ * performing the log transformation.
+ *
+ * @param {number...} domain... optional domain values.
+ * @extends pv.Scale.quantitative
+ */
+pv.Scale.log = function() {
+  var scale = pv.Scale.quantitative(1, 10),
+      b, // logarithm base
+      p, // cached Math.log(b)
+      /** @ignore */ log = function(x) { return Math.log(x) / p; },
+      /** @ignore */ pow = function(y) { return Math.pow(b, y); };
+
+  /**
+   * Returns an array of evenly-spaced, suitably-rounded values in the input
+   * domain. These values are frequently used in conjunction with
+   * {@link pv.Rule} to display tick marks or grid lines.
+   *
+   * @function
+   * @name pv.Scale.log.prototype.ticks
+   * @returns {number[]} an array input domain values to use as ticks.
+   */
+  scale.ticks = function() {
+    // TODO support non-uniform domains
+    var d = scale.domain(),
+        n = d[0] < 0,
+        i = Math.floor(n ? -log(-d[0]) : log(d[0])),
+        j = Math.ceil(n ? -log(-d[1]) : log(d[1])),
+        ticks = [];
+    if (n) {
+      ticks.push(-pow(-i));
+      for (; i++ < j;) for (var k = b - 1; k > 0; k--) ticks.push(-pow(-i) * k);
+    } else {
+      for (; i < j; i++) for (var k = 1; k < b; k++) ticks.push(pow(i) * k);
+      ticks.push(pow(i));
+    }
+    for (i = 0; ticks[i] < d[0]; i++); // strip small values
+    for (j = ticks.length; ticks[j - 1] > d[1]; j--); // strip big values
+    return ticks.slice(i, j);
+  };
+
+  /**
+   * Formats the specified tick value using the appropriate precision, assuming
+   * base 10.
+   *
+   * @function
+   * @name pv.Scale.log.prototype.tickFormat
+   * @param {number} t a tick value.
+   * @returns {string} a formatted tick value.
+   */
+  scale.tickFormat = function(t) {
+    return t.toPrecision(1);
+  };
+
+  /**
+   * "Nices" this scale, extending the bounds of the input domain to
+   * evenly-rounded values. This method uses {@link pv.logFloor} and
+   * {@link pv.logCeil}. Nicing is useful if the domain is computed dynamically
+   * from data, and may be irregular. For example, given a domain of
+   * [0.20147987687960267, 0.996679553296417], a call to <tt>nice()</tt> might
+   * extend the domain to [0.1, 1].
+   *
+   * <p>This method must be invoked each time after setting the domain (and
+   * base).
+   *
+   * @function
+   * @name pv.Scale.log.prototype.nice
+   * @returns {pv.Scale.log} <tt>this</tt>.
+   */
+  scale.nice = function() {
+    // TODO support non-uniform domains
+    var d = scale.domain();
+    return scale.domain(pv.logFloor(d[0], b), pv.logCeil(d[1], b));
+  };
+
+  /**
+   * Sets or gets the logarithm base. Defaults to 10.
+   *
+   * @function
+   * @name pv.Scale.log.prototype.base
+   * @param {number} [v] the new base.
+   * @returns {pv.Scale.log} <tt>this</tt>, or the current base.
+   */
+  scale.base = function(v) {
+    if (arguments.length) {
+      b = Number(v);
+      p = Math.log(b);
+      scale.transform(log, pow); // update transformed domain
+      return this;
+    }
+    return b;
+  };
+
+  scale.domain.apply(scale, arguments);
+  return scale.base(10);
+};
+/**
+ * Returns a root scale for the specified domain. The arguments to this
+ * constructor are optional, and equivalent to calling {@link #domain}.
+ * The default domain and range are [0,1].
+ *
+ * @class Represents a root scale; a function that performs a power
+ * transformation. <style type="text/css">sub{line-height:0}</style> Most
+ * commonly, a root scale represents a 1-dimensional root transformation from a
+ * numeric domain of input data [<i>d<sub>0</sub></i>, <i>d<sub>1</sub></i>] to
+ * a numeric range of pixels [<i>r<sub>0</sub></i>, <i>r<sub>1</sub></i>].
+ *
+ * <p>Note that the scale is itself a function, and thus can be used as a
+ * property directly, assuming that the data associated with a mark is a
+ * number. While this is convenient for single-use scales, frequently it is
+ * desirable to define scales globally:
+ *
+ * <pre>var y = pv.Scale.root(0, 100).range(0, 640);</pre>
+ *
+ * The <tt>y</tt> scale can now be equivalently referenced within a property:
+ *
+ * <pre>    .height(function(d) y(d))</pre>
+ *
+ * Alternatively, if the data are not simple numbers, the appropriate value can
+ * be passed to the <tt>y</tt> scale (e.g., <tt>d.foo</tt>). The {@link #by}
+ * method similarly allows the data to be mapped to a numeric value before
+ * performing the root transformation.
+ *
+ * @param {number...} domain... optional domain values.
+ * @extends pv.Scale.quantitative
+ */
+pv.Scale.root = function() {
+  var scale = pv.Scale.quantitative();
+
+  /**
+   * Sets or gets the exponent; defaults to 2.
+   *
+   * @function
+   * @name pv.Scale.root.prototype.power
+   * @param {number} [v] the new exponent.
+   * @returns {pv.Scale.root} <tt>this</tt>, or the current base.
+   */
+  scale.power = function(v) {
+    if (arguments.length) {
+      var b = Number(v), p = 1 / b;
+      scale.transform(
+        function(x) { return Math.pow(x, p); },
+        function(y) { return Math.pow(y, b); });
+      return this;
+    }
+    return b;
+  };
+
+  scale.domain.apply(scale, arguments);
+  return scale.power(2);
+};
+/**
+ * Returns an ordinal scale for the specified domain. The arguments to this
+ * constructor are optional, and equivalent to calling {@link #domain}.
+ *
+ * @class Represents an ordinal scale. <style
+ * type="text/css">sub{line-height:0}</style> An ordinal scale represents a
+ * pairwise mapping from <i>n</i> discrete values in the input domain to
+ * <i>n</i> discrete values in the output range. For example, an ordinal scale
+ * might map a domain of species ["setosa", "versicolor", "virginica"] to colors
+ * ["red", "green", "blue"]. Thus, saying
+ *
+ * <pre>    .fillStyle(function(d) {
+ *         switch (d.species) {
+ *           case "setosa": return "red";
+ *           case "versicolor": return "green";
+ *           case "virginica": return "blue";
+ *         }
+ *       })</pre>
+ *
+ * is equivalent to
+ *
+ * <pre>    .fillStyle(pv.Scale.ordinal("setosa", "versicolor", "virginica")
+ *         .range("red", "green", "blue")
+ *         .by(function(d) d.species))</pre>
+ *
+ * If the mapping from species to color does not need to be specified
+ * explicitly, the domain can be omitted. In this case it will be inferred
+ * lazily from the data:
+ *
+ * <pre>    .fillStyle(pv.colors("red", "green", "blue")
+ *         .by(function(d) d.species))</pre>
+ *
+ * When the domain is inferred, the first time the scale is invoked, the first
+ * element from the range will be returned. Subsequent calls with unique values
+ * will return subsequent elements from the range. If the inferred domain grows
+ * larger than the range, range values will be reused. However, it is strongly
+ * recommended that the domain and the range contain the same number of
+ * elements.
+ *
+ * <p>A range can be discretized from a continuous interval (e.g., for pixel
+ * positioning) by using {@link #split}, {@link #splitFlush} or
+ * {@link #splitBanded} after the domain has been set. For example, if
+ * <tt>states</tt> is an array of the fifty U.S. state names, the state name can
+ * be encoded in the left position:
+ *
+ * <pre>    .left(pv.Scale.ordinal(states)
+ *         .split(0, 640)
+ *         .by(function(d) d.state))</pre>
+ *
+ * <p>N.B.: ordinal scales are not invertible (at least not yet), since the
+ * domain and range and discontinuous. A workaround is to use a linear scale.
+ *
+ * @param {...} domain... optional domain values.
+ * @extends pv.Scale
+ * @see pv.colors
+ */
+pv.Scale.ordinal = function() {
+  var d = [], i = {}, r = [], band = 0;
+
+  /** @private */
+  function scale(x) {
+    if (!(x in i)) i[x] = d.push(x) - 1;
+    return r[i[x] % r.length];
+  }
+
+  /**
+   * Sets or gets the input domain. This method can be invoked several ways:
+   *
+   * <p>1. <tt>domain(values...)</tt>
+   *
+   * <p>Specifying the domain as a series of values is the most explicit and
+   * recommended approach. However, if the domain values are derived from data,
+   * you may find the second method more appropriate.
+   *
+   * <p>2. <tt>domain(array, f)</tt>
+   *
+   * <p>Rather than enumerating the domain values as explicit arguments to this
+   * method, you can specify a single argument of an array. In addition, you can
+   * specify an optional accessor function to extract the domain values from the
+   * array.
+   *
+   * <p>3. <tt>domain()</tt>
+   *
+   * <p>Invoking the <tt>domain</tt> method with no arguments returns the
+   * current domain as an array.
+   *
+   * @function
+   * @name pv.Scale.ordinal.prototype.domain
+   * @param {...} domain... domain values.
+   * @returns {pv.Scale.ordinal} <tt>this</tt>, or the current domain.
+   */
+  scale.domain = function(array, f) {
+    if (arguments.length) {
+      array = (array instanceof Array)
+          ? ((arguments.length > 1) ? pv.map(array, f) : array)
+          : Array.prototype.slice.call(arguments);
+
+      /* Filter the specified ordinals to their unique values. */
+      d = [];
+      var seen = {};
+      for (var j = 0; j < array.length; j++) {
+        var o = array[j];
+        if (!(o in seen)) {
+          seen[o] = true;
+          d.push(o);
+        }
+      }
+
+      i = pv.numerate(d);
+      return this;
+    }
+    return d;
+  };
+
+  /**
+   * Sets or gets the output range. This method can be invoked several ways:
+   *
+   * <p>1. <tt>range(values...)</tt>
+   *
+   * <p>Specifying the range as a series of values is the most explicit and
+   * recommended approach. However, if the range values are derived from data,
+   * you may find the second method more appropriate.
+   *
+   * <p>2. <tt>range(array, f)</tt>
+   *
+   * <p>Rather than enumerating the range values as explicit arguments to this
+   * method, you can specify a single argument of an array. In addition, you can
+   * specify an optional accessor function to extract the range values from the
+   * array.
+   *
+   * <p>3. <tt>range()</tt>
+   *
+   * <p>Invoking the <tt>range</tt> method with no arguments returns the
+   * current range as an array.
+   *
+   * @function
+   * @name pv.Scale.ordinal.prototype.range
+   * @param {...} range... range values.
+   * @returns {pv.Scale.ordinal} <tt>this</tt>, or the current range.
+   */
+  scale.range = function(array, f) {
+    if (arguments.length) {
+      r = (array instanceof Array)
+          ? ((arguments.length > 1) ? pv.map(array, f) : array)
+          : Array.prototype.slice.call(arguments);
+      if (typeof r[0] == "string") r = r.map(pv.color);
+      return this;
+    }
+    return r;
+  };
+
+  /**
+   * Sets the range from the given continuous interval. The interval
+   * [<i>min</i>, <i>max</i>] is subdivided into <i>n</i> equispaced points,
+   * where <i>n</i> is the number of (unique) values in the domain. The first
+   * and last point are offset from the edge of the range by half the distance
+   * between points.
+   *
+   * <p>This method must be called <i>after</i> the domain is set.
+   *
+   * @function
+   * @name pv.Scale.ordinal.prototype.split
+   * @param {number} min minimum value of the output range.
+   * @param {number} max maximum value of the output range.
+   * @returns {pv.Scale.ordinal} <tt>this</tt>.
+   * @see #splitFlush
+   * @see #splitBanded
+   */
+  scale.split = function(min, max) {
+    var step = (max - min) / this.domain().length;
+    r = pv.range(min + step / 2, max, step);
+    return this;
+  };
+
+  /**
+   * Sets the range from the given continuous interval. The interval
+   * [<i>min</i>, <i>max</i>] is subdivided into <i>n</i> equispaced points,
+   * where <i>n</i> is the number of (unique) values in the domain. The first
+   * and last point are exactly on the edge of the range.
+   *
+   * <p>This method must be called <i>after</i> the domain is set.
+   *
+   * @function
+   * @name pv.Scale.ordinal.prototype.splitFlush
+   * @param {number} min minimum value of the output range.
+   * @param {number} max maximum value of the output range.
+   * @returns {pv.Scale.ordinal} <tt>this</tt>.
+   * @see #split
+   */
+  scale.splitFlush = function(min, max) {
+    var n = this.domain().length, step = (max - min) / (n - 1);
+    r = (n == 1) ? [(min + max) / 2]
+        : pv.range(min, max + step / 2, step);
+    return this;
+  };
+
+  /**
+   * Sets the range from the given continuous interval. The interval
+   * [<i>min</i>, <i>max</i>] is subdivided into <i>n</i> equispaced bands,
+   * where <i>n</i> is the number of (unique) values in the domain. The first
+   * and last band are offset from the edge of the range by the distance between
+   * bands.
+   *
+   * <p>The band width argument, <tt>band</tt>, is typically in the range [0, 1]
+   * and defaults to 1. This fraction corresponds to the amount of space in the
+   * range to allocate to the bands, as opposed to padding. A value of 0.5 means
+   * that the band width will be equal to the padding width. The computed
+   * absolute band width can be retrieved from the range as
+   * <tt>scale.range().band</tt>.
+   *
+   * <p>If the band width argument is negative, this method will allocate bands
+   * of a <i>fixed</i> width <tt>-band</tt>, rather than a relative fraction of
+   * the available space.
+   *
+   * <p>Tip: to inset the bands by a fixed amount <tt>p</tt>, specify a minimum
+   * value of <tt>min + p</tt> (or simply <tt>p</tt>, if <tt>min</tt> is
+   * 0). Then set the mark width to <tt>scale.range().band - p</tt>.
+   *
+   * <p>This method must be called <i>after</i> the domain is set.
+   *
+   * @function
+   * @name pv.Scale.ordinal.prototype.splitBanded
+   * @param {number} min minimum value of the output range.
+   * @param {number} max maximum value of the output range.
+   * @param {number} [band] the fractional band width in [0, 1]; defaults to 1.
+   * @returns {pv.Scale.ordinal} <tt>this</tt>.
+   * @see #split
+   */
+  scale.splitBanded = function(min, max, band) {
+    if (arguments.length < 3) band = 1;
+    if (band < 0) {
+      var n = this.domain().length,
+          total = -band * n,
+          remaining = max - min - total,
+          padding = remaining / (n + 1);
+      r = pv.range(min + padding, max, padding - band);
+      r.band = -band;
+    } else {
+      var step = (max - min) / (this.domain().length + (1 - band));
+      r = pv.range(min + step * (1 - band), max, step);
+      r.band = step * band;
+    }
+    return this;
+  };
+
+  /**
+   * Returns a view of this scale by the specified accessor function <tt>f</tt>.
+   * Given a scale <tt>y</tt>, <tt>y.by(function(d) d.foo)</tt> is equivalent to
+   * <tt>function(d) y(d.foo)</tt>. This method should be used judiciously; it
+   * is typically more clear to invoke the scale directly, passing in the value
+   * to be scaled.
+   *
+   * @function
+   * @name pv.Scale.ordinal.prototype.by
+   * @param {function} f an accessor function.
+   * @returns {pv.Scale.ordinal} a view of this scale by the specified accessor
+   * function.
+   */
+  scale.by = function(f) {
+    function by() { return scale(f.apply(this, arguments)); }
+    for (var method in scale) by[method] = scale[method];
+    return by;
+  };
+
+  scale.domain.apply(scale, arguments);
+  return scale;
+};
+/**
+ * Constructs a default quantile scale. The arguments to this constructor are
+ * optional, and equivalent to calling {@link #domain}. The default domain is
+ * the empty set, and the default range is [0,1].
+ *
+ * @class Represents a quantile scale; a function that maps from a value within
+ * a sortable domain to a quantized numeric range. Typically, the domain is a
+ * set of numbers, but any sortable value (such as strings) can be used as the
+ * domain of a quantile scale. The range defaults to [0,1], with 0 corresponding
+ * to the smallest value in the domain, 1 the largest, .5 the median, etc.
+ *
+ * <p>By default, the number of quantiles in the range corresponds to the number
+ * of values in the domain. The {@link #quantiles} method can be used to specify
+ * an explicit number of quantiles; for example, <tt>quantiles(4)</tt> produces
+ * a standard quartile scale. A quartile scale's range is a set of four discrete
+ * values, such as [0, 1/3, 2/3, 1]. Calling the {@link #range} method will
+ * scale these discrete values accordingly, similar to {@link
+ * pv.Scale.ordinal#splitFlush}.
+ *
+ * <p>For example, given the strings ["c", "a", "b"], a default quantile scale:
+ *
+ * <pre>pv.Scale.quantile("c", "a", "b")</pre>
+ *
+ * will return 0 for "a", .5 for "b", and 1 for "c".
+ *
+ * @extends pv.Scale
+ */
+pv.Scale.quantile = function() {
+  var n = -1, // number of quantiles
+      j = -1, // max quantile index
+      q = [], // quantile boundaries
+      d = [], // domain
+      y = pv.Scale.linear(); // range
+
+  /** @private */
+  function scale(x) {
+    return y(Math.max(0, Math.min(j, pv.search.index(q, x) - 1)) / j);
+  }
+
+  /**
+   * Sets or gets the quantile boundaries. By default, each element in the
+   * domain is in its own quantile. If the argument to this method is a number,
+   * it specifies the number of equal-sized quantiles by which to divide the
+   * domain.
+   *
+   * <p>If no arguments are specified, this method returns the quantile
+   * boundaries; the first element is always the minimum value of the domain,
+   * and the last element is the maximum value of the domain. Thus, the length
+   * of the returned array is always one greater than the number of quantiles.
+   *
+   * @function
+   * @name pv.Scale.quantile.prototype.quantiles
+   * @param {number} x the number of quantiles.
+   */
+  scale.quantiles = function(x) {
+    if (arguments.length) {
+      n = Number(x);
+      if (n < 0) {
+        q = [d[0]].concat(d);
+        j = d.length - 1;
+      } else {
+        q = [];
+        q[0] = d[0];
+        for (var i = 1; i <= n; i++) {
+          q[i] = d[~~(i * (d.length - 1) / n)];
+        }
+        j = n - 1;
+      }
+      return this;
+    }
+    return q;
+  };
+
+  /**
+   * Sets or gets the input domain. This method can be invoked several ways:
+   *
+   * <p>1. <tt>domain(values...)</tt>
+   *
+   * <p>Specifying the domain as a series of values is the most explicit and
+   * recommended approach. However, if the domain values are derived from data,
+   * you may find the second method more appropriate.
+   *
+   * <p>2. <tt>domain(array, f)</tt>
+   *
+   * <p>Rather than enumerating the domain values as explicit arguments to this
+   * method, you can specify a single argument of an array. In addition, you can
+   * specify an optional accessor function to extract the domain values from the
+   * array.
+   *
+   * <p>3. <tt>domain()</tt>
+   *
+   * <p>Invoking the <tt>domain</tt> method with no arguments returns the
+   * current domain as an array.
+   *
+   * @function
+   * @name pv.Scale.quantile.prototype.domain
+   * @param {...} domain... domain values.
+   * @returns {pv.Scale.quantile} <tt>this</tt>, or the current domain.
+   */
+  scale.domain = function(array, f) {
+    if (arguments.length) {
+      d = (array instanceof Array)
+          ? pv.map(array, f)
+          : Array.prototype.slice.call(arguments);
+      d.sort(pv.naturalOrder);
+      scale.quantiles(n); // recompute quantiles
+      return this;
+    }
+    return d;
+  };
+
+  /**
+   * Sets or gets the output range. This method can be invoked several ways:
+   *
+   * <p>1. <tt>range(min, ..., max)</tt>
+   *
+   * <p>The range may be specified as a series of numbers or colors. Most
+   * commonly, two numbers are specified: the minimum and maximum pixel values.
+   * For a color scale, values may be specified as {@link pv.Color}s or
+   * equivalent strings. For a diverging scale, or other subdivided non-uniform
+   * scales, multiple values can be specified. For example:
+   *
+   * <pre>    .range("red", "white", "green")</pre>
+   *
+   * <p>Currently, only numbers and colors are supported as range values. The
+   * number of range values must exactly match the number of domain values, or
+   * the behavior of the scale is undefined.
+   *
+   * <p>2. <tt>range()</tt>
+   *
+   * <p>Invoking the <tt>range</tt> method with no arguments returns the current
+   * range as an array of numbers or colors.
+   *
+   * @function
+   * @name pv.Scale.quantile.prototype.range
+   * @param {...} range... range values.
+   * @returns {pv.Scale.quantile} <tt>this</tt>, or the current range.
+   */
+  scale.range = function() {
+    if (arguments.length) {
+      y.range.apply(y, arguments);
+      return this;
+    }
+    return y.range();
+  };
+
+  /**
+   * Returns a view of this scale by the specified accessor function <tt>f</tt>.
+   * Given a scale <tt>y</tt>, <tt>y.by(function(d) d.foo)</tt> is equivalent to
+   * <tt>function(d) y(d.foo)</tt>.
+   *
+   * <p>This method is provided for convenience, such that scales can be
+   * succinctly defined inline. For example, given an array of data elements
+   * that have a <tt>score</tt> attribute with the domain [0, 1], the height
+   * property could be specified as:
+   *
+   * <pre>.height(pv.Scale.linear().range(0, 480).by(function(d) d.score))</pre>
+   *
+   * This is equivalent to:
+   *
+   * <pre>.height(function(d) d.score * 480)</pre>
+   *
+   * This method should be used judiciously; it is typically more clear to
+   * invoke the scale directly, passing in the value to be scaled.
+   *
+   * @function
+   * @name pv.Scale.quantile.prototype.by
+   * @param {function} f an accessor function.
+   * @returns {pv.Scale.quantile} a view of this scale by the specified
+   * accessor function.
+   */
+  scale.by = function(f) {
+    function by() { return scale(f.apply(this, arguments)); }
+    for (var method in scale) by[method] = scale[method];
+    return by;
+  };
+
+  scale.domain.apply(scale, arguments);
+  return scale;
+};
+/**
+ * Returns a histogram operator for the specified data, with an optional
+ * accessor function. If the data specified is not an array of numbers, an
+ * accessor function must be specified to map the data to numeric values.
+ *
+ * @class Represents a histogram operator.
+ *
+ * @param {array} data an array of numbers or objects.
+ * @param {function} [f] an optional accessor function.
+ */
+pv.histogram = function(data, f) {
+  var frequency = true;
+  return {
+
+    /**
+     * Returns the computed histogram bins. An optional array of numbers,
+     * <tt>ticks</tt>, may be specified as the break points. If the ticks are
+     * not specified, default ticks will be computed using a linear scale on the
+     * data domain.
+     *
+     * <p>The returned array contains {@link pv.histogram.Bin}s. The <tt>x</tt>
+     * attribute corresponds to the bin's start value (inclusive), while the
+     * <tt>dx</tt> attribute stores the bin size (end - start). The <tt>y</tt>
+     * attribute stores either the frequency count or probability, depending on
+     * how the histogram operator has been configured.
+     *
+     * <p>The {@link pv.histogram.Bin} objects are themselves arrays, containing
+     * the data elements present in each bin, i.e., the elements in the
+     * <tt>data</tt> array (prior to invoking the accessor function, if any).
+     * For example, if the data represented countries, and the accessor function
+     * returned the GDP of each country, the returned bins would be arrays of
+     * countries (not GDPs).
+     *
+     * @function
+     * @name pv.histogram.prototype.bins
+     * @param {array} [ticks]
+     * @returns {array}
+     */ /** @private */
+    bins: function(ticks) {
+      var x = pv.map(data, f), bins = [];
+
+      /* Initialize default ticks. */
+      if (!arguments.length) ticks = pv.Scale.linear(x).ticks();
+
+      /* Initialize the bins. */
+      for (var i = 0; i < ticks.length - 1; i++) {
+        var bin = bins[i] = [];
+        bin.x = ticks[i];
+        bin.dx = ticks[i + 1] - ticks[i];
+        bin.y = 0;
+      }
+
+      /* Count the number of samples per bin. */
+      for (var i = 0; i < x.length; i++) {
+        var j = pv.search.index(ticks, x[i]) - 1,
+            bin = bins[Math.max(0, Math.min(bins.length - 1, j))];
+        bin.y++;
+        bin.push(data[i]);
+      }
+
+      /* Convert frequencies to probabilities. */
+      if (!frequency) for (var i = 0; i < bins.length; i++) {
+        bins[i].y /= x.length;
+      }
+
+      return bins;
+    },
+
+    /**
+     * Sets or gets whether this histogram operator returns frequencies or
+     * probabilities.
+     *
+     * @function
+     * @name pv.histogram.prototype.frequency
+     * @param {boolean} [x]
+     * @returns {pv.histogram} this.
+     */ /** @private */
+    frequency: function(x) {
+      if (arguments.length) {
+        frequency = Boolean(x);
+        return this;
+      }
+      return frequency;
+    }
+  };
+};
+
+/**
+ * @class Represents a bin returned by the {@link pv.histogram} operator. Bins
+ * are themselves arrays containing the data elements present in the given bin
+ * (prior to the accessor function being invoked to convert the data object to a
+ * numeric value). These bin arrays have additional attributes with meta
+ * information about the bin.
+ *
+ * @name pv.histogram.Bin
+ * @extends array
+ * @see pv.histogram
+ */
+
+/**
+ * The start value of the bin's range.
+ *
+ * @type number
+ * @name pv.histogram.Bin.prototype.x
+ */
+
+/**
+ * The magnitude value of the bin's range; end - start.
+ *
+ * @type number
+ * @name pv.histogram.Bin.prototype.dx
+ */
+
+/**
+ * The frequency or probability of the bin, depending on how the histogram
+ * operator was configured.
+ *
+ * @type number
+ * @name pv.histogram.Bin.prototype.y
+ */
+/**
+ * Returns the {@link pv.Color} for the specified color format string. Colors
+ * may have an associated opacity, or alpha channel. Color formats are specified
+ * by CSS Color Modular Level 3, using either in RGB or HSL color space. For
+ * example:<ul>
+ *
+ * <li>#f00 // #rgb
+ * <li>#ff0000 // #rrggbb
+ * <li>rgb(255, 0, 0)
+ * <li>rgb(100%, 0%, 0%)
+ * <li>hsl(0, 100%, 50%)
+ * <li>rgba(0, 0, 255, 0.5)
+ * <li>hsla(120, 100%, 50%, 1)
+ *
+ * </ul>The SVG 1.0 color keywords names are also supported, such as "aliceblue"
+ * and "yellowgreen". The "transparent" keyword is supported for fully-
+ * transparent black.
+ *
+ * <p>If the <tt>format</tt> argument is already an instance of <tt>Color</tt>,
+ * the argument is returned with no further processing.
+ *
+ * @param {string} format the color specification string, such as "#f00".
+ * @returns {pv.Color} the corresponding <tt>Color</tt>.
+ * @see <a href="http://www.w3.org/TR/SVG/types.html#ColorKeywords">SVG color
+ * keywords</a>
+ * @see <a href="http://www.w3.org/TR/css3-color/">CSS3 color module</a>
+ */
+pv.color = function(format) {
+  if (format.rgb) return format.rgb();
+
+  /* Handle hsl, rgb. */
+  var m1 = /([a-z]+)\((.*)\)/i.exec(format);
+  if (m1) {
+    var m2 = m1[2].split(","), a = 1;
+    switch (m1[1]) {
+      case "hsla":
+      case "rgba": {
+        a = parseFloat(m2[3]);
+        if (!a) return pv.Color.transparent;
+        break;
+      }
+    }
+    switch (m1[1]) {
+      case "hsla":
+      case "hsl": {
+        var h = parseFloat(m2[0]), // degrees
+            s = parseFloat(m2[1]) / 100, // percentage
+            l = parseFloat(m2[2]) / 100; // percentage
+        return (new pv.Color.Hsl(h, s, l, a)).rgb();
+      }
+      case "rgba":
+      case "rgb": {
+        function parse(c) { // either integer or percentage
+          var f = parseFloat(c);
+          return (c[c.length - 1] == '%') ? Math.round(f * 2.55) : f;
+        }
+        var r = parse(m2[0]), g = parse(m2[1]), b = parse(m2[2]);
+        return pv.rgb(r, g, b, a);
+      }
+    }
+  }
+
+  /* Named colors. */
+  var named = pv.Color.names[format];
+  if (named) return named;
+
+  /* Hexadecimal colors: #rgb and #rrggbb. */
+  if (format.charAt(0) == "#") {
+    var r, g, b;
+    if (format.length == 4) {
+      r = format.charAt(1); r += r;
+      g = format.charAt(2); g += g;
+      b = format.charAt(3); b += b;
+    } else if (format.length == 7) {
+      r = format.substring(1, 3);
+      g = format.substring(3, 5);
+      b = format.substring(5, 7);
+    }
+    return pv.rgb(parseInt(r, 16), parseInt(g, 16), parseInt(b, 16), 1);
+  }
+
+  /* Otherwise, pass-through unsupported colors. */
+  return new pv.Color(format, 1);
+};
+
+/**
+ * Constructs a color with the specified color format string and opacity. This
+ * constructor should not be invoked directly; use {@link pv.color} instead.
+ *
+ * @class Represents an abstract (possibly translucent) color. The color is
+ * divided into two parts: the <tt>color</tt> attribute, an opaque color format
+ * string, and the <tt>opacity</tt> attribute, a float in [0, 1]. The color
+ * space is dependent on the implementing class; all colors support the
+ * {@link #rgb} method to convert to RGB color space for interpolation.
+ *
+ * <p>See also the <a href="../../api/Color.html">Color guide</a>.
+ *
+ * @param {string} color an opaque color format string, such as "#f00".
+ * @param {number} opacity the opacity, in [0,1].
+ * @see pv.color
+ */
+pv.Color = function(color, opacity) {
+  /**
+   * An opaque color format string, such as "#f00".
+   *
+   * @type string
+   * @see <a href="http://www.w3.org/TR/SVG/types.html#ColorKeywords">SVG color
+   * keywords</a>
+   * @see <a href="http://www.w3.org/TR/css3-color/">CSS3 color module</a>
+   */
+  this.color = color;
+
+  /**
+   * The opacity, a float in [0, 1].
+   *
+   * @type number
+   */
+  this.opacity = opacity;
+};
+
+/**
+ * Returns a new color that is a brighter version of this color. The behavior of
+ * this method may vary slightly depending on the underlying color space.
+ * Although brighter and darker are inverse operations, the results of a series
+ * of invocations of these two methods might be inconsistent because of rounding
+ * errors.
+ *
+ * @param [k] {number} an optional scale factor; defaults to 1.
+ * @see #darker
+ * @returns {pv.Color} a brighter color.
+ */
+pv.Color.prototype.brighter = function(k) {
+  return this.rgb().brighter(k);
+};
+
+/**
+ * Returns a new color that is a brighter version of this color. The behavior of
+ * this method may vary slightly depending on the underlying color space.
+ * Although brighter and darker are inverse operations, the results of a series
+ * of invocations of these two methods might be inconsistent because of rounding
+ * errors.
+ *
+ * @param [k] {number} an optional scale factor; defaults to 1.
+ * @see #brighter
+ * @returns {pv.Color} a darker color.
+ */
+pv.Color.prototype.darker = function(k) {
+  return this.rgb().darker(k);
+};
+
+/**
+ * Constructs a new RGB color with the specified channel values.
+ *
+ * @param {number} r the red channel, an integer in [0,255].
+ * @param {number} g the green channel, an integer in [0,255].
+ * @param {number} b the blue channel, an integer in [0,255].
+ * @param {number} [a] the alpha channel, a float in [0,1].
+ * @returns pv.Color.Rgb
+ */
+pv.rgb = function(r, g, b, a) {
+  return new pv.Color.Rgb(r, g, b, (arguments.length == 4) ? a : 1);
+};
+
+/**
+ * Constructs a new RGB color with the specified channel values.
+ *
+ * @class Represents a color in RGB space.
+ *
+ * @param {number} r the red channel, an integer in [0,255].
+ * @param {number} g the green channel, an integer in [0,255].
+ * @param {number} b the blue channel, an integer in [0,255].
+ * @param {number} a the alpha channel, a float in [0,1].
+ * @extends pv.Color
+ */
+pv.Color.Rgb = function(r, g, b, a) {
+  pv.Color.call(this, a ? ("rgb(" + r + "," + g + "," + b + ")") : "none", a);
+
+  /**
+   * The red channel, an integer in [0, 255].
+   *
+   * @type number
+   */
+  this.r = r;
+
+  /**
+   * The green channel, an integer in [0, 255].
+   *
+   * @type number
+   */
+  this.g = g;
+
+  /**
+   * The blue channel, an integer in [0, 255].
+   *
+   * @type number
+   */
+  this.b = b;
+
+  /**
+   * The alpha channel, a float in [0, 1].
+   *
+   * @type number
+   */
+  this.a = a;
+};
+pv.Color.Rgb.prototype = pv.extend(pv.Color);
+
+/**
+ * Constructs a new RGB color with the same green, blue and alpha channels as
+ * this color, with the specified red channel.
+ *
+ * @param {number} r the red channel, an integer in [0,255].
+ */
+pv.Color.Rgb.prototype.red = function(r) {
+  return pv.rgb(r, this.g, this.b, this.a);
+};
+
+/**
+ * Constructs a new RGB color with the same red, blue and alpha channels as this
+ * color, with the specified green channel.
+ *
+ * @param {number} g the green channel, an integer in [0,255].
+ */
+pv.Color.Rgb.prototype.green = function(g) {
+  return pv.rgb(this.r, g, this.b, this.a);
+};
+
+/**
+ * Constructs a new RGB color with the same red, green and alpha channels as
+ * this color, with the specified blue channel.
+ *
+ * @param {number} b the blue channel, an integer in [0,255].
+ */
+pv.Color.Rgb.prototype.blue = function(b) {
+  return pv.rgb(this.r, this.g, b, this.a);
+};
+
+/**
+ * Constructs a new RGB color with the same red, green and blue channels as this
+ * color, with the specified alpha channel.
+ *
+ * @param {number} a the alpha channel, a float in [0,1].
+ */
+pv.Color.Rgb.prototype.alpha = function(a) {
+  return pv.rgb(this.r, this.g, this.b, a);
+};
+
+/**
+ * Returns the RGB color equivalent to this color. This method is abstract and
+ * must be implemented by subclasses.
+ *
+ * @returns {pv.Color.Rgb} an RGB color.
+ * @function
+ * @name pv.Color.prototype.rgb
+ */
+
+/**
+ * Returns this.
+ *
+ * @returns {pv.Color.Rgb} this.
+ */
+pv.Color.Rgb.prototype.rgb = function() { return this; };
+
+/**
+ * Returns a new color that is a brighter version of this color. This method
+ * applies an arbitrary scale factor to each of the three RGB components of this
+ * color to create a brighter version of this color. Although brighter and
+ * darker are inverse operations, the results of a series of invocations of
+ * these two methods might be inconsistent because of rounding errors.
+ *
+ * @param [k] {number} an optional scale factor; defaults to 1.
+ * @see #darker
+ * @returns {pv.Color.Rgb} a brighter color.
+ */
+pv.Color.Rgb.prototype.brighter = function(k) {
+  k = Math.pow(0.7, arguments.length ? k : 1);
+  var r = this.r, g = this.g, b = this.b, i = 30;
+  if (!r && !g && !b) return pv.rgb(i, i, i, this.a);
+  if (r && (r < i)) r = i;
+  if (g && (g < i)) g = i;
+  if (b && (b < i)) b = i;
+  return pv.rgb(
+      Math.min(255, Math.floor(r / k)),
+      Math.min(255, Math.floor(g / k)),
+      Math.min(255, Math.floor(b / k)),
+      this.a);
+};
+
+/**
+ * Returns a new color that is a darker version of this color. This method
+ * applies an arbitrary scale factor to each of the three RGB components of this
+ * color to create a darker version of this color. Although brighter and darker
+ * are inverse operations, the results of a series of invocations of these two
+ * methods might be inconsistent because of rounding errors.
+ *
+ * @param [k] {number} an optional scale factor; defaults to 1.
+ * @see #brighter
+ * @returns {pv.Color.Rgb} a darker color.
+ */
+pv.Color.Rgb.prototype.darker = function(k) {
+  k = Math.pow(0.7, arguments.length ? k : 1);
+  return pv.rgb(
+      Math.max(0, Math.floor(k * this.r)),
+      Math.max(0, Math.floor(k * this.g)),
+      Math.max(0, Math.floor(k * this.b)),
+      this.a);
+};
+
+/**
+ * Constructs a new HSL color with the specified values.
+ *
+ * @param {number} h the hue, an integer in [0, 360].
+ * @param {number} s the saturation, a float in [0, 1].
+ * @param {number} l the lightness, a float in [0, 1].
+ * @param {number} [a] the opacity, a float in [0, 1].
+ * @returns pv.Color.Hsl
+ */
+pv.hsl = function(h, s, l, a) {
+  return new pv.Color.Hsl(h, s, l,  (arguments.length == 4) ? a : 1);
+};
+
+/**
+ * Constructs a new HSL color with the specified values.
+ *
+ * @class Represents a color in HSL space.
+ *
+ * @param {number} h the hue, an integer in [0, 360].
+ * @param {number} s the saturation, a float in [0, 1].
+ * @param {number} l the lightness, a float in [0, 1].
+ * @param {number} a the opacity, a float in [0, 1].
+ * @extends pv.Color
+ */
+pv.Color.Hsl = function(h, s, l, a) {
+  pv.Color.call(this, "hsl(" + h + "," + (s * 100) + "%," + (l * 100) + "%)", a);
+
+  /**
+   * The hue, an integer in [0, 360].
+   *
+   * @type number
+   */
+  this.h = h;
+
+  /**
+   * The saturation, a float in [0, 1].
+   *
+   * @type number
+   */
+  this.s = s;
+
+  /**
+   * The lightness, a float in [0, 1].
+   *
+   * @type number
+   */
+  this.l = l;
+
+  /**
+   * The opacity, a float in [0, 1].
+   *
+   * @type number
+   */
+  this.a = a;
+};
+pv.Color.Hsl.prototype = pv.extend(pv.Color);
+
+/**
+ * Constructs a new HSL color with the same saturation, lightness and alpha as
+ * this color, and the specified hue.
+ *
+ * @param {number} h the hue, an integer in [0, 360].
+ */
+pv.Color.Hsl.prototype.hue = function(h) {
+  return pv.hsl(h, this.s, this.l, this.a);
+};
+
+/**
+ * Constructs a new HSL color with the same hue, lightness and alpha as this
+ * color, and the specified saturation.
+ *
+ * @param {number} s the saturation, a float in [0, 1].
+ */
+pv.Color.Hsl.prototype.saturation = function(s) {
+  return pv.hsl(this.h, s, this.l, this.a);
+};
+
+/**
+ * Constructs a new HSL color with the same hue, saturation and alpha as this
+ * color, and the specified lightness.
+ *
+ * @param {number} l the lightness, a float in [0, 1].
+ */
+pv.Color.Hsl.prototype.lightness = function(l) {
+  return pv.hsl(this.h, this.s, l, this.a);
+};
+
+/**
+ * Constructs a new HSL color with the same hue, saturation and lightness as
+ * this color, and the specified alpha.
+ *
+ * @param {number} a the opacity, a float in [0, 1].
+ */
+pv.Color.Hsl.prototype.alpha = function(a) {
+  return pv.hsl(this.h, this.s, this.l, a);
+};
+
+/**
+ * Returns the RGB color equivalent to this HSL color.
+ *
+ * @returns {pv.Color.Rgb} an RGB color.
+ */
+pv.Color.Hsl.prototype.rgb = function() {
+  var h = this.h, s = this.s, l = this.l;
+
+  /* Some simple corrections for h, s and l. */
+  h = h % 360; if (h < 0) h += 360;
+  s = Math.max(0, Math.min(s, 1));
+  l = Math.max(0, Math.min(l, 1));
+
+  /* From FvD 13.37, CSS Color Module Level 3 */
+  var m2 = (l <= .5) ? (l * (1 + s)) : (l + s - l * s);
+  var m1 = 2 * l - m2;
+  function v(h) {
+    if (h > 360) h -= 360;
+    else if (h < 0) h += 360;
+    if (h < 60) return m1 + (m2 - m1) * h / 60;
+    if (h < 180) return m2;
+    if (h < 240) return m1 + (m2 - m1) * (240 - h) / 60;
+    return m1;
+  }
+  function vv(h) {
+    return Math.round(v(h) * 255);
+  }
+
+  return pv.rgb(vv(h + 120), vv(h), vv(h - 120), this.a);
+};
+
+/**
+ * @private SVG color keywords, per CSS Color Module Level 3.
+ *
+ * @see <a href="http://www.w3.org/TR/SVG/types.html#ColorKeywords">SVG color
+ * keywords</a>
+ */
+pv.Color.names = {
+  aliceblue: "#f0f8ff",
+  antiquewhite: "#faebd7",
+  aqua: "#00ffff",
+  aquamarine: "#7fffd4",
+  azure: "#f0ffff",
+  beige: "#f5f5dc",
+  bisque: "#ffe4c4",
+  black: "#000000",
+  blanchedalmond: "#ffebcd",
+  blue: "#0000ff",
+  blueviolet: "#8a2be2",
+  brown: "#a52a2a",
+  burlywood: "#deb887",
+  cadetblue: "#5f9ea0",
+  chartreuse: "#7fff00",
+  chocolate: "#d2691e",
+  coral: "#ff7f50",
+  cornflowerblue: "#6495ed",
+  cornsilk: "#fff8dc",
+  crimson: "#dc143c",
+  cyan: "#00ffff",
+  darkblue: "#00008b",
+  darkcyan: "#008b8b",
+  darkgoldenrod: "#b8860b",
+  darkgray: "#a9a9a9",
+  darkgreen: "#006400",
+  darkgrey: "#a9a9a9",
+  darkkhaki: "#bdb76b",
+  darkmagenta: "#8b008b",
+  darkolivegreen: "#556b2f",
+  darkorange: "#ff8c00",
+  darkorchid: "#9932cc",
+  darkred: "#8b0000",
+  darksalmon: "#e9967a",
+  darkseagreen: "#8fbc8f",
+  darkslateblue: "#483d8b",
+  darkslategray: "#2f4f4f",
+  darkslategrey: "#2f4f4f",
+  darkturquoise: "#00ced1",
+  darkviolet: "#9400d3",
+  deeppink: "#ff1493",
+  deepskyblue: "#00bfff",
+  dimgray: "#696969",
+  dimgrey: "#696969",
+  dodgerblue: "#1e90ff",
+  firebrick: "#b22222",
+  floralwhite: "#fffaf0",
+  forestgreen: "#228b22",
+  fuchsia: "#ff00ff",
+  gainsboro: "#dcdcdc",
+  ghostwhite: "#f8f8ff",
+  gold: "#ffd700",
+  goldenrod: "#daa520",
+  gray: "#808080",
+  green: "#008000",
+  greenyellow: "#adff2f",
+  grey: "#808080",
+  honeydew: "#f0fff0",
+  hotpink: "#ff69b4",
+  indianred: "#cd5c5c",
+  indigo: "#4b0082",
+  ivory: "#fffff0",
+  khaki: "#f0e68c",
+  lavender: "#e6e6fa",
+  lavenderblush: "#fff0f5",
+  lawngreen: "#7cfc00",
+  lemonchiffon: "#fffacd",
+  lightblue: "#add8e6",
+  lightcoral: "#f08080",
+  lightcyan: "#e0ffff",
+  lightgoldenrodyellow: "#fafad2",
+  lightgray: "#d3d3d3",
+  lightgreen: "#90ee90",
+  lightgrey: "#d3d3d3",
+  lightpink: "#ffb6c1",
+  lightsalmon: "#ffa07a",
+  lightseagreen: "#20b2aa",
+  lightskyblue: "#87cefa",
+  lightslategray: "#778899",
+  lightslategrey: "#778899",
+  lightsteelblue: "#b0c4de",
+  lightyellow: "#ffffe0",
+  lime: "#00ff00",
+  limegreen: "#32cd32",
+  linen: "#faf0e6",
+  magenta: "#ff00ff",
+  maroon: "#800000",
+  mediumaquamarine: "#66cdaa",
+  mediumblue: "#0000cd",
+  mediumorchid: "#ba55d3",
+  mediumpurple: "#9370db",
+  mediumseagreen: "#3cb371",
+  mediumslateblue: "#7b68ee",
+  mediumspringgreen: "#00fa9a",
+  mediumturquoise: "#48d1cc",
+  mediumvioletred: "#c71585",
+  midnightblue: "#191970",
+  mintcream: "#f5fffa",
+  mistyrose: "#ffe4e1",
+  moccasin: "#ffe4b5",
+  navajowhite: "#ffdead",
+  navy: "#000080",
+  oldlace: "#fdf5e6",
+  olive: "#808000",
+  olivedrab: "#6b8e23",
+  orange: "#ffa500",
+  orangered: "#ff4500",
+  orchid: "#da70d6",
+  palegoldenrod: "#eee8aa",
+  palegreen: "#98fb98",
+  paleturquoise: "#afeeee",
+  palevioletred: "#db7093",
+  papayawhip: "#ffefd5",
+  peachpuff: "#ffdab9",
+  peru: "#cd853f",
+  pink: "#ffc0cb",
+  plum: "#dda0dd",
+  powderblue: "#b0e0e6",
+  purple: "#800080",
+  red: "#ff0000",
+  rosybrown: "#bc8f8f",
+  royalblue: "#4169e1",
+  saddlebrown: "#8b4513",
+  salmon: "#fa8072",
+  sandybrown: "#f4a460",
+  seagreen: "#2e8b57",
+  seashell: "#fff5ee",
+  sienna: "#a0522d",
+  silver: "#c0c0c0",
+  skyblue: "#87ceeb",
+  slateblue: "#6a5acd",
+  slategray: "#708090",
+  slategrey: "#708090",
+  snow: "#fffafa",
+  springgreen: "#00ff7f",
+  steelblue: "#4682b4",
+  tan: "#d2b48c",
+  teal: "#008080",
+  thistle: "#d8bfd8",
+  tomato: "#ff6347",
+  turquoise: "#40e0d0",
+  violet: "#ee82ee",
+  wheat: "#f5deb3",
+  white: "#ffffff",
+  whitesmoke: "#f5f5f5",
+  yellow: "#ffff00",
+  yellowgreen: "#9acd32",
+  transparent: pv.Color.transparent = pv.rgb(0, 0, 0, 0)
+};
+
+/* Initialized named colors. */
+(function() {
+  var names = pv.Color.names;
+  for (var name in names) names[name] = pv.color(names[name]);
+})();
+/**
+ * Returns a new categorical color encoding using the specified colors.  The
+ * arguments to this method are an array of colors; see {@link pv.color}. For
+ * example, to create a categorical color encoding using the <tt>species</tt>
+ * attribute:
+ *
+ * <pre>pv.colors("red", "green", "blue").by(function(d) d.species)</pre>
+ *
+ * The result of this expression can be used as a fill- or stroke-style
+ * property. This assumes that the data's <tt>species</tt> attribute is a
+ * string.
+ *
+ * @param {string} colors... categorical colors.
+ * @see pv.Scale.ordinal
+ * @returns {pv.Scale.ordinal} an ordinal color scale.
+ */
+pv.colors = function() {
+  var scale = pv.Scale.ordinal();
+  scale.range.apply(scale, arguments);
+  return scale;
+};
+
+/**
+ * A collection of standard color palettes for categorical encoding.
+ *
+ * @namespace A collection of standard color palettes for categorical encoding.
+ */
+pv.Colors = {};
+
+/**
+ * Returns a new 10-color scheme. The arguments to this constructor are
+ * optional, and equivalent to calling {@link pv.Scale.OrdinalScale#domain}. The
+ * following colors are used:
+ *
+ * <div style="background:#1f77b4;">#1f77b4</div>
+ * <div style="background:#ff7f0e;">#ff7f0e</div>
+ * <div style="background:#2ca02c;">#2ca02c</div>
+ * <div style="background:#d62728;">#d62728</div>
+ * <div style="background:#9467bd;">#9467bd</div>
+ * <div style="background:#8c564b;">#8c564b</div>
+ * <div style="background:#e377c2;">#e377c2</div>
+ * <div style="background:#7f7f7f;">#7f7f7f</div>
+ * <div style="background:#bcbd22;">#bcbd22</div>
+ * <div style="background:#17becf;">#17becf</div>
+ *
+ * @param {number...} domain... domain values.
+ * @returns {pv.Scale.ordinal} a new ordinal color scale.
+ * @see pv.color
+ */
+pv.Colors.category10 = function() {
+  var scale = pv.colors(
+      "#1f77b4", "#ff7f0e", "#2ca02c", "#d62728", "#9467bd",
+      "#8c564b", "#e377c2", "#7f7f7f", "#bcbd22", "#17becf");
+  scale.domain.apply(scale, arguments);
+  return scale;
+};
+
+/**
+ * Returns a new 20-color scheme. The arguments to this constructor are
+ * optional, and equivalent to calling {@link pv.Scale.OrdinalScale#domain}. The
+ * following colors are used:
+ *
+ * <div style="background:#1f77b4;">#1f77b4</div>
+ * <div style="background:#aec7e8;">#aec7e8</div>
+ * <div style="background:#ff7f0e;">#ff7f0e</div>
+ * <div style="background:#ffbb78;">#ffbb78</div>
+ * <div style="background:#2ca02c;">#2ca02c</div>
+ * <div style="background:#98df8a;">#98df8a</div>
+ * <div style="background:#d62728;">#d62728</div>
+ * <div style="background:#ff9896;">#ff9896</div>
+ * <div style="background:#9467bd;">#9467bd</div>
+ * <div style="background:#c5b0d5;">#c5b0d5</div>
+ * <div style="background:#8c564b;">#8c564b</div>
+ * <div style="background:#c49c94;">#c49c94</div>
+ * <div style="background:#e377c2;">#e377c2</div>
+ * <div style="background:#f7b6d2;">#f7b6d2</div>
+ * <div style="background:#7f7f7f;">#7f7f7f</div>
+ * <div style="background:#c7c7c7;">#c7c7c7</div>
+ * <div style="background:#bcbd22;">#bcbd22</div>
+ * <div style="background:#dbdb8d;">#dbdb8d</div>
+ * <div style="background:#17becf;">#17becf</div>
+ * <div style="background:#9edae5;">#9edae5</div>
+ *
+ * @param {number...} domain... domain values.
+ * @returns {pv.Scale.ordinal} a new ordinal color scale.
+ * @see pv.color
+*/
+pv.Colors.category20 = function() {
+  var scale = pv.colors(
+      "#1f77b4", "#aec7e8", "#ff7f0e", "#ffbb78", "#2ca02c",
+      "#98df8a", "#d62728", "#ff9896", "#9467bd", "#c5b0d5",
+      "#8c564b", "#c49c94", "#e377c2", "#f7b6d2", "#7f7f7f",
+      "#c7c7c7", "#bcbd22", "#dbdb8d", "#17becf", "#9edae5");
+  scale.domain.apply(scale, arguments);
+  return scale;
+};
+
+/**
+ * Returns a new alternative 19-color scheme. The arguments to this constructor
+ * are optional, and equivalent to calling
+ * {@link pv.Scale.OrdinalScale#domain}. The following colors are used:
+ *
+ * <div style="background:#9c9ede;">#9c9ede</div>
+ * <div style="background:#7375b5;">#7375b5</div>
+ * <div style="background:#4a5584;">#4a5584</div>
+ * <div style="background:#cedb9c;">#cedb9c</div>
+ * <div style="background:#b5cf6b;">#b5cf6b</div>
+ * <div style="background:#8ca252;">#8ca252</div>
+ * <div style="background:#637939;">#637939</div>
+ * <div style="background:#e7cb94;">#e7cb94</div>
+ * <div style="background:#e7ba52;">#e7ba52</div>
+ * <div style="background:#bd9e39;">#bd9e39</div>
+ * <div style="background:#8c6d31;">#8c6d31</div>
+ * <div style="background:#e7969c;">#e7969c</div>
+ * <div style="background:#d6616b;">#d6616b</div>
+ * <div style="background:#ad494a;">#ad494a</div>
+ * <div style="background:#843c39;">#843c39</div>
+ * <div style="background:#de9ed6;">#de9ed6</div>
+ * <div style="background:#ce6dbd;">#ce6dbd</div>
+ * <div style="background:#a55194;">#a55194</div>
+ * <div style="background:#7b4173;">#7b4173</div>
+ *
+ * @param {number...} domain... domain values.
+ * @returns {pv.Scale.ordinal} a new ordinal color scale.
+ * @see pv.color
+ */
+pv.Colors.category19 = function() {
+  var scale = pv.colors(
+      "#9c9ede", "#7375b5", "#4a5584", "#cedb9c", "#b5cf6b",
+      "#8ca252", "#637939", "#e7cb94", "#e7ba52", "#bd9e39",
+      "#8c6d31", "#e7969c", "#d6616b", "#ad494a", "#843c39",
+      "#de9ed6", "#ce6dbd", "#a55194", "#7b4173");
+  scale.domain.apply(scale, arguments);
+  return scale;
+};
+/**
+ * Returns a linear color ramp from the specified <tt>start</tt> color to the
+ * specified <tt>end</tt> color. The color arguments may be specified either as
+ * <tt>string</tt>s or as {@link pv.Color}s. This is equivalent to:
+ *
+ * <pre>    pv.Scale.linear().domain(0, 1).range(...)</pre>
+ *
+ * @param {string} start the start color; may be a <tt>pv.Color</tt>.
+ * @param {string} end the end color; may be a <tt>pv.Color</tt>.
+ * @returns {Function} a color ramp from <tt>start</tt> to <tt>end</tt>.
+ * @see pv.Scale.linear
+ */
+pv.ramp = function(start, end) {
+  var scale = pv.Scale.linear();
+  scale.range.apply(scale, arguments);
+  return scale;
+};
+/**
+ * @private
+ * @namespace
+ */
+pv.Scene = pv.SvgScene = {
+  /* Various namespaces. */
+  svg: "http://www.w3.org/2000/svg",
+  xmlns: "http://www.w3.org/2000/xmlns",
+  xlink: "http://www.w3.org/1999/xlink",
+  xhtml: "http://www.w3.org/1999/xhtml",
+
+  /** The pre-multipled scale, based on any enclosing transforms. */
+  scale: 1,
+
+  /** The set of supported events. */
+  events: [
+    "DOMMouseScroll", // for Firefox
+    "mousewheel",
+    "mousedown",
+    "mouseup",
+    "mouseover",
+    "mouseout",
+    "mousemove",
+    "click",
+    "dblclick"
+  ],
+
+  /** Implicit values for SVG and CSS properties. */
+  implicit: {
+    svg: {
+      "shape-rendering": "auto",
+      "pointer-events": "painted",
+      "x": 0,
+      "y": 0,
+      "dy": 0,
+      "text-anchor": "start",
+      "transform": "translate(0,0)",
+      "fill": "none",
+      "fill-opacity": 1,
+      "stroke": "none",
+      "stroke-opacity": 1,
+      "stroke-width": 1.5,
+      "stroke-linejoin": "miter"
+    },
+    css: {
+      "font": "10px sans-serif"
+    }
+  }
+};
+
+/**
+ * Updates the display for the specified array of scene nodes.
+ *
+ * @param scenes {array} an array of scene nodes.
+ */
+pv.SvgScene.updateAll = function(scenes) {
+  if (scenes.length
+      && scenes[0].reverse
+      && (scenes.type != "line")
+      && (scenes.type != "area")) {
+    var reversed = pv.extend(scenes);
+    for (var i = 0, j = scenes.length - 1; j >= 0; i++, j--) {
+      reversed[i] = scenes[j];
+    }
+    scenes = reversed;
+  }
+  this.removeSiblings(this[scenes.type](scenes));
+};
+
+/**
+ * Creates a new SVG element of the specified type.
+ *
+ * @param type {string} an SVG element type, such as "rect".
+ * @returns a new SVG element.
+ */
+pv.SvgScene.create = function(type) {
+  return document.createElementNS(this.svg, type);
+};
+
+/**
+ * Expects the element <i>e</i> to be the specified type. If the element does
+ * not exist, a new one is created. If the element does exist but is the wrong
+ * type, it is replaced with the specified element.
+ *
+ * @param e the current SVG element.
+ * @param type {string} an SVG element type, such as "rect".
+ * @param attributes an optional attribute map.
+ * @param style an optional style map.
+ * @returns a new SVG element.
+ */
+pv.SvgScene.expect = function(e, type, attributes, style) {
+  if (e) {
+    if (e.tagName == "a") e = e.firstChild;
+    if (e.tagName != type) {
+      var n = this.create(type);
+      e.parentNode.replaceChild(n, e);
+      e = n;
+    }
+  } else {
+    e = this.create(type);
+  }
+  for (var name in attributes) {
+    var value = attributes[name];
+    if (value == this.implicit.svg[name]) value = null;
+    if (value == null) e.removeAttribute(name);
+    else e.setAttribute(name, value);
+  }
+  for (var name in style) {
+    var value = style[name];
+    if (value == this.implicit.css[name]) value = null;
+    if (value == null) e.style.removeProperty(name);
+    else e.style[name] = value;
+  }
+  return e;
+};
+
+/** TODO */
+pv.SvgScene.append = function(e, scenes, index) {
+  e.$scene = {scenes:scenes, index:index};
+  e = this.title(e, scenes[index]);
+  if (!e.parentNode) scenes.$g.appendChild(e);
+  return e.nextSibling;
+};
+
+/**
+ * Applies a title tooltip to the specified element <tt>e</tt>, using the
+ * <tt>title</tt> property of the specified scene node <tt>s</tt>. Note that
+ * this implementation does not create an SVG <tt>title</tt> element as a child
+ * of <tt>e</tt>; although this is the recommended standard, it is only
+ * supported in Opera. Instead, an anchor element is created around the element
+ * <tt>e</tt>, and the <tt>xlink:title</tt> attribute is set accordingly.
+ *
+ * @param e an SVG element.
+ * @param s a scene node.
+ */
+pv.SvgScene.title = function(e, s) {
+  var a = e.parentNode;
+  if (a && (a.tagName != "a")) a = null;
+  if (s.title) {
+    if (!a) {
+      a = this.create("a");
+      if (e.parentNode) e.parentNode.replaceChild(a, e);
+      a.appendChild(e);
+    }
+    a.setAttributeNS(this.xlink, "title", s.title);
+    return a;
+  }
+  if (a) a.parentNode.replaceChild(e, a);
+  return e;
+};
+
+/** TODO */
+pv.SvgScene.dispatch = pv.listener(function(e) {
+  var t = e.target.$scene;
+  if (t) {
+    var type = e.type;
+
+    /* Fixes for mousewheel support on Firefox & Opera. */
+    switch (type) {
+      case "DOMMouseScroll": {
+        type = "mousewheel";
+        e.wheel = -480 * e.detail;
+        break;
+      }
+      case "mousewheel": {
+        e.wheel = (window.opera ? 12 : 1) * e.wheelDelta;
+        break;
+      }
+    }
+
+    if (pv.Mark.dispatch(type, t.scenes, t.index)) e.preventDefault();
+  }
+});
+
+/** @private Remove siblings following element <i>e</i>. */
+pv.SvgScene.removeSiblings = function(e) {
+  while (e) {
+    var n = e.nextSibling;
+    e.parentNode.removeChild(e);
+    e = n;
+  }
+};
+
+/** @private Do nothing when rendering undefined mark types. */
+pv.SvgScene.undefined = function() {};
+/**
+ * @private Converts the specified b-spline curve segment to a bezier curve
+ * compatible with SVG "C".
+ *
+ * @param p0 the first control point.
+ * @param p1 the second control point.
+ * @param p2 the third control point.
+ * @param p3 the fourth control point.
+ */
+pv.SvgScene.pathBasis = (function() {
+
+  /**
+   * Matrix to transform basis (b-spline) control points to bezier control
+   * points. Derived from FvD 11.2.8.
+   */
+  var basis = [
+    [ 1/6, 2/3, 1/6,   0 ],
+    [   0, 2/3, 1/3,   0 ],
+    [   0, 1/3, 2/3,   0 ],
+    [   0, 1/6, 2/3, 1/6 ]
+  ];
+
+  /**
+   * Returns the point that is the weighted sum of the specified control points,
+   * using the specified weights. This method requires that there are four
+   * weights and four control points.
+   */
+  function weight(w, p0, p1, p2, p3) {
+    return {
+      x: w[0] * p0.left + w[1] * p1.left + w[2] * p2.left + w[3] * p3.left,
+      y: w[0] * p0.top  + w[1] * p1.top  + w[2] * p2.top  + w[3] * p3.top
+    };
+  }
+
+  var convert = function(p0, p1, p2, p3) {
+    var b1 = weight(basis[1], p0, p1, p2, p3),
+        b2 = weight(basis[2], p0, p1, p2, p3),
+        b3 = weight(basis[3], p0, p1, p2, p3);
+    return "C" + b1.x + "," + b1.y
+         + "," + b2.x + "," + b2.y
+         + "," + b3.x + "," + b3.y;
+  };
+
+  convert.segment = function(p0, p1, p2, p3) {
+    var b0 = weight(basis[0], p0, p1, p2, p3),
+        b1 = weight(basis[1], p0, p1, p2, p3),
+        b2 = weight(basis[2], p0, p1, p2, p3),
+        b3 = weight(basis[3], p0, p1, p2, p3);
+    return "M" + b0.x + "," + b0.y
+         + "C" + b1.x + "," + b1.y
+         + "," + b2.x + "," + b2.y
+         + "," + b3.x + "," + b3.y;
+  };
+
+  return convert;
+})();
+
+/**
+ * @private Interpolates the given points using the basis spline interpolation.
+ * Returns an SVG path without the leading M instruction to allow path
+ * appending.
+ *
+ * @param points the array of points.
+ */
+pv.SvgScene.curveBasis = function(points) {
+  if (points.length <= 2) return "";
+  var path = "",
+      p0 = points[0],
+      p1 = p0,
+      p2 = p0,
+      p3 = points[1];
+  path += this.pathBasis(p0, p1, p2, p3);
+  for (var i = 2; i < points.length; i++) {
+    p0 = p1;
+    p1 = p2;
+    p2 = p3;
+    p3 = points[i];
+    path += this.pathBasis(p0, p1, p2, p3);
+  }
+  /* Cycle through to get the last point. */
+  path += this.pathBasis(p1, p2, p3, p3);
+  path += this.pathBasis(p2, p3, p3, p3);
+  return path;
+};
+
+/**
+ * @private Interpolates the given points using the basis spline interpolation.
+ * If points.length == tangents.length then a regular Hermite interpolation is
+ * performed, if points.length == tangents.length + 2 then the first and last
+ * segments are filled in with cubic bazier segments.  Returns an array of path
+ * strings.
+ *
+ * @param points the array of points.
+ */
+pv.SvgScene.curveBasisSegments = function(points) {
+  if (points.length <= 2) return "";
+  var paths = [],
+      p0 = points[0],
+      p1 = p0,
+      p2 = p0,
+      p3 = points[1],
+      firstPath = this.pathBasis.segment(p0, p1, p2, p3);
+
+  p0 = p1;
+  p1 = p2;
+  p2 = p3;
+  p3 = points[2];
+  paths.push(firstPath + this.pathBasis(p0, p1, p2, p3)); // merge first & second path
+  for (var i = 3; i < points.length; i++) {
+    p0 = p1;
+    p1 = p2;
+    p2 = p3;
+    p3 = points[i];
+    paths.push(this.pathBasis.segment(p0, p1, p2, p3));
+  }
+
+  // merge last & second-to-last path
+  paths.push(this.pathBasis.segment(p1, p2, p3, p3) + this.pathBasis(p2, p3, p3, p3));
+  return paths;
+};
+
+/**
+ * @private Interpolates the given points with respective tangents using the cubic
+ * Hermite spline interpolation. If points.length == tangents.length then a regular
+ * Hermite interpolation is performed, if points.length == tangents.length + 2 then
+ * the first and last segments are filled in with cubic bazier segments.
+ * Returns an SVG path without the leading M instruction to allow path appending.
+ *
+ * @param points the array of points.
+ * @param tangents the array of tangent vectors.
+ */
+pv.SvgScene.curveHermite = function(points, tangents) {
+  if (tangents.length < 1
+      || (points.length != tangents.length
+      && points.length != tangents.length + 2)) return "";
+  var quad = points.length != tangents.length,
+      path = "",
+      p0 = points[0],
+      p = points[1],
+      t0 = tangents[0],
+      t = t0,
+      pi = 1;
+
+  if (quad) {
+    path += "Q" + (p.left - t0.x * 2 / 3) + ","  + (p.top - t0.y * 2 / 3)
+        + "," + p.left + "," + p.top;
+    p0 = points[1];
+    pi = 2;
+  }
+
+  if (tangents.length > 1) {
+    t = tangents[1];
+    p = points[pi];
+    pi++;
+    path += "C" + (p0.left + t0.x) + "," + (p0.top + t0.y)
+        + "," + (p.left - t.x) + "," + (p.top - t.y)
+        + "," + p.left + "," + p.top;
+    for (var i = 2; i < tangents.length; i++, pi++) {
+      p = points[pi];
+      t = tangents[i];
+      path += "S" + (p.left - t.x) + "," + (p.top - t.y)
+          + "," + p.left + "," + p.top;
+    }
+  }
+
+  if (quad) {
+    var lp = points[pi];
+    path += "Q" + (p.left + t.x * 2 / 3) + ","  + (p.top + t.y * 2 / 3) + ","
+        + lp.left + "," + lp.top;
+  }
+
+  return path;
+};
+
+/**
+ * @private Interpolates the given points with respective tangents using the
+ * cubic Hermite spline interpolation. Returns an array of path strings.
+ *
+ * @param points the array of points.
+ * @param tangents the array of tangent vectors.
+ */
+pv.SvgScene.curveHermiteSegments = function(points, tangents) {
+  if (tangents.length < 1
+      || (points.length != tangents.length
+      && points.length != tangents.length + 2)) return [];
+  var quad = points.length != tangents.length,
+      paths = [],
+      p0 = points[0],
+      p = p0,
+      t0 = tangents[0],
+      t = t0,
+      pi = 1;
+
+  if (quad) {
+    p = points[1];
+    paths.push("M" + p0.left + "," + p0.top
+        + "Q" + (p.left - t.x * 2 / 3) + "," + (p.top - t.y * 2 / 3)
+        + "," + p.left + "," + p.top);
+    pi = 2;
+  }
+
+  for (var i = 1; i < tangents.length; i++, pi++) {
+    p0 = p;
+    t0 = t;
+    p = points[pi];
+    t = tangents[i];
+    paths.push("M" + p0.left + "," + p0.top
+        + "C" + (p0.left + t0.x) + "," + (p0.top + t0.y)
+        + "," + (p.left - t.x) + "," + (p.top - t.y)
+        + "," + p.left + "," + p.top);
+  }
+
+  if (quad) {
+    var lp = points[pi];
+    paths.push("M" + p.left + "," + p.top
+        + "Q" + (p.left + t.x * 2 / 3) + ","  + (p.top + t.y * 2 / 3) + ","
+        + lp.left + "," + lp.top);
+  }
+
+  return paths;
+};
+
+/**
+ * @private Computes the tangents for the given points needed for cardinal
+ * spline interpolation. Returns an array of tangent vectors. Note: that for n
+ * points only the n-2 well defined tangents are returned.
+ *
+ * @param points the array of points.
+ * @param tension the tension of hte cardinal spline.
+ */
+pv.SvgScene.cardinalTangents = function(points, tension) {
+  var tangents = [],
+      a = (1 - tension) / 2,
+      p0 = points[0],
+      p1 = points[1],
+      p2 = points[2];
+
+  for (var i = 3; i < points.length; i++) {
+    tangents.push({x: a * (p2.left - p0.left), y: a * (p2.top - p0.top)});
+    p0 = p1;
+    p1 = p2;
+    p2 = points[i];
+  }
+
+  tangents.push({x: a * (p2.left - p0.left), y: a * (p2.top - p0.top)});
+  return tangents;
+};
+
+/**
+ * @private Interpolates the given points using cardinal spline interpolation.
+ * Returns an SVG path without the leading M instruction to allow path
+ * appending.
+ *
+ * @param points the array of points.
+ * @param tension the tension of hte cardinal spline.
+ */
+pv.SvgScene.curveCardinal = function(points, tension) {
+  if (points.length <= 2) return "";
+  return this.curveHermite(points, this.cardinalTangents(points, tension));
+};
+
+/**
+ * @private Interpolates the given points using cardinal spline interpolation.
+ * Returns an array of path strings.
+ *
+ * @param points the array of points.
+ * @param tension the tension of hte cardinal spline.
+ */
+pv.SvgScene.curveCardinalSegments = function(points, tension) {
+  if (points.length <= 2) return "";
+  return this.curveHermiteSegments(points, this.cardinalTangents(points, tension));
+};
+
+/**
+ * @private Interpolates the given points using Fritsch-Carlson Monotone cubic
+ * Hermite interpolation. Returns an array of tangent vectors.
+ *
+ * @param points the array of points.
+ */
+pv.SvgScene.monotoneTangents = function(points) {
+  var tangents = [],
+      d = [],
+      m = [],
+      dx = [],
+      k = 0;
+
+  /* Compute the slopes of the secant lines between successive points. */
+  for (k = 0; k < points.length-1; k++) {
+    d[k] = (points[k+1].top - points[k].top)/(points[k+1].left - points[k].left);
+  }
+
+  /* Initialize the tangents at every point as the average of the secants. */
+  m[0] = d[0];
+  dx[0] = points[1].left - points[0].left;
+  for (k = 1; k < points.length - 1; k++) {
+    m[k] = (d[k-1]+d[k])/2;
+    dx[k] = (points[k+1].left - points[k-1].left)/2;
+  }
+  m[k] = d[k-1];
+  dx[k] = (points[k].left - points[k-1].left);
+
+  /* Step 3. Very important, step 3. Yep. Wouldn't miss it. */
+  for (k = 0; k < points.length - 1; k++) {
+    if (d[k] == 0) {
+      m[ k ] = 0;
+      m[k+1] = 0;
+    }
+  }
+
+  /* Step 4 + 5. Out of 5 or more steps. */
+  for (k = 0; k < points.length - 1; k++) {
+    if ((Math.abs(m[k]) < 1e-5) || (Math.abs(m[k+1]) < 1e-5)) continue;
+    var ak = m[k] / d[k],
+        bk = m[k + 1] / d[k],
+        s = ak * ak + bk * bk; // monotone constant (?)
+    if (s > 9) {
+      var tk = 3 / Math.sqrt(s);
+      m[k] = tk * ak * d[k];
+      m[k + 1] = tk * bk * d[k];
+    }
+  }
+
+  var len;
+  for (var i = 0; i < points.length; i++) {
+    len = 1 + m[i] * m[i]; // pv.vector(1, m[i]).norm().times(dx[i]/3)
+    tangents.push({x: dx[i] / 3 / len, y: m[i] * dx[i] / 3 / len});
+  }
+
+  return tangents;
+};
+
+/**
+ * @private Interpolates the given points using Fritsch-Carlson Monotone cubic
+ * Hermite interpolation. Returns an SVG path without the leading M instruction
+ * to allow path appending.
+ *
+ * @param points the array of points.
+ */
+pv.SvgScene.curveMonotone = function(points) {
+  if (points.length <= 2) return "";
+  return this.curveHermite(points, this.monotoneTangents(points));
+}
+
+/**
+ * @private Interpolates the given points using Fritsch-Carlson Monotone cubic
+ * Hermite interpolation.
+ * Returns an array of path strings.
+ *
+ * @param points the array of points.
+ */
+pv.SvgScene.curveMonotoneSegments = function(points) {
+  if (points.length <= 2) return "";
+  return this.curveHermiteSegments(points, this.monotoneTangents(points));
+};
+pv.SvgScene.area = function(scenes) {
+  var e = scenes.$g.firstChild;
+  if (!scenes.length) return e;
+  var s = scenes[0];
+
+  /* segmented */
+  if (s.segmented) return this.areaSegment(scenes);
+
+  /* visible */
+  if (!s.visible) return e;
+  var fill = s.fillStyle, stroke = s.strokeStyle;
+  if (!fill.opacity && !stroke.opacity) return e;
+
+  /** @private Computes the straight path for the range [i, j]. */
+  function path(i, j) {
+    var p1 = [], p2 = [];
+    for (var k = j; i <= k; i++, j--) {
+      var si = scenes[i],
+          sj = scenes[j],
+          pi = si.left + "," + si.top,
+          pj = (sj.left + sj.width) + "," + (sj.top + sj.height);
+
+      /* interpolate */
+      if (i < k) {
+        var sk = scenes[i + 1], sl = scenes[j - 1];
+        switch (s.interpolate) {
+          case "step-before": {
+            pi += "V" + sk.top;
+            pj += "H" + (sl.left + sl.width);
+            break;
+          }
+          case "step-after": {
+            pi += "H" + sk.left;
+            pj += "V" + (sl.top + sl.height);
+            break;
+          }
+        }
+      }
+
+      p1.push(pi);
+      p2.push(pj);
+    }
+    return p1.concat(p2).join("L");
+  }
+
+  /** @private Computes the curved path for the range [i, j]. */
+  function pathCurve(i, j) {
+    var pointsT = [], pointsB = [], pathT, pathB;
+
+    for (var k = j; i <= k; i++, j--) {
+      var sj = scenes[j];
+      pointsT.push(scenes[i]);
+      pointsB.push({left: sj.left + sj.width, top: sj.top + sj.height});
+    }
+
+    if (s.interpolate == "basis") {
+      pathT = pv.SvgScene.curveBasis(pointsT);
+      pathB = pv.SvgScene.curveBasis(pointsB);
+    } else if (s.interpolate == "cardinal") {
+      pathT = pv.SvgScene.curveCardinal(pointsT, s.tension);
+      pathB = pv.SvgScene.curveCardinal(pointsB, s.tension);
+    } else { // monotone
+      pathT = pv.SvgScene.curveMonotone(pointsT);
+      pathB = pv.SvgScene.curveMonotone(pointsB);
+    }
+
+    return pointsT[0].left + "," + pointsT[0].top + pathT
+         + "L" + pointsB[0].left + "," + pointsB[0].top + pathB;
+  }
+
+  /* points */
+  var d = [], si, sj;
+  for (var i = 0; i < scenes.length; i++) {
+    si = scenes[i]; if (!si.width && !si.height) continue;
+    for (var j = i + 1; j < scenes.length; j++) {
+      sj = scenes[j]; if (!sj.width && !sj.height) break;
+    }
+    if (i && (s.interpolate != "step-after")) i--;
+    if ((j < scenes.length) && (s.interpolate != "step-before")) j++;
+    d.push(((j - i > 2
+        && (s.interpolate == "basis"
+        || s.interpolate == "cardinal"
+        || s.interpolate == "monotone"))
+        ? pathCurve : path)(i, j - 1));
+    i = j - 1;
+  }
+  if (!d.length) return e;
+
+  e = this.expect(e, "path", {
+      "shape-rendering": s.antialias ? null : "crispEdges",
+      "pointer-events": s.events,
+      "cursor": s.cursor,
+      "d": "M" + d.join("ZM") + "Z",
+      "fill": fill.color,
+      "fill-opacity": fill.opacity || null,
+      "stroke": stroke.color,
+      "stroke-opacity": stroke.opacity || null,
+      "stroke-width": stroke.opacity ? s.lineWidth / this.scale : null
+    });
+  return this.append(e, scenes, 0);
+};
+
+pv.SvgScene.areaSegment = function(scenes) {
+  var e = scenes.$g.firstChild, s = scenes[0], pathsT, pathsB;
+  if (s.interpolate == "basis"
+      || s.interpolate == "cardinal"
+      || s.interpolate == "monotone") {
+    var pointsT = [], pointsB = [];
+
+    for (var i = 0, n = scenes.length; i < n; i++) {
+      var sj = scenes[n - i - 1];
+      pointsT.push(scenes[i]);
+      pointsB.push({left: sj.left + sj.width, top: sj.top + sj.height});
+    }
+
+    if (s.interpolate == "basis") {
+      pathsT = this.curveBasisSegments(pointsT);
+      pathsB = this.curveBasisSegments(pointsB);
+    } else if (s.interpolate == "cardinal") {
+      pathsT = this.curveCardinalSegments(pointsT, s.tension);
+      pathsB = this.curveCardinalSegments(pointsB, s.tension);
+    } else { // monotone
+      pathsT = this.curveMonotoneSegments(pointsT);
+      pathsB = this.curveMonotoneSegments(pointsB);
+    }
+  }
+
+  for (var i = 0, n = scenes.length - 1; i < n; i++) {
+    var s1 = scenes[i], s2 = scenes[i + 1];
+
+    /* visible */
+    if (!s1.visible || !s2.visible) continue;
+    var fill = s1.fillStyle, stroke = s1.strokeStyle;
+    if (!fill.opacity && !stroke.opacity) continue;
+
+    var d;
+    if (pathsT) {
+      var pathT = pathsT[i],
+          pathB = "L" + pathsB[n - i - 1].substr(1);
+
+      d = pathT + pathB + "Z";
+    } else {
+      /* interpolate */
+      var si = s1, sj = s2;
+      switch (s1.interpolate) {
+        case "step-before": si = s2; break;
+        case "step-after": sj = s1; break;
+      }
+
+      /* path */
+      d = "M" + s1.left + "," + si.top
+        + "L" + s2.left + "," + sj.top
+        + "L" + (s2.left + s2.width) + "," + (sj.top + sj.height)
+        + "L" + (s1.left + s1.width) + "," + (si.top + si.height)
+        + "Z";
+    }
+
+    e = this.expect(e, "path", {
+        "shape-rendering": s1.antialias ? null : "crispEdges",
+        "pointer-events": s1.events,
+        "cursor": s1.cursor,
+        "d": d,
+        "fill": fill.color,
+        "fill-opacity": fill.opacity || null,
+        "stroke": stroke.color,
+        "stroke-opacity": stroke.opacity || null,
+        "stroke-width": stroke.opacity ? s1.lineWidth / this.scale : null
+      });
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+pv.SvgScene.bar = function(scenes) {
+  var e = scenes.$g.firstChild;
+  for (var i = 0; i < scenes.length; i++) {
+    var s = scenes[i];
+
+    /* visible */
+    if (!s.visible) continue;
+    var fill = s.fillStyle, stroke = s.strokeStyle;
+    if (!fill.opacity && !stroke.opacity) continue;
+
+    e = this.expect(e, "rect", {
+        "shape-rendering": s.antialias ? null : "crispEdges",
+        "pointer-events": s.events,
+        "cursor": s.cursor,
+        "x": s.left,
+        "y": s.top,
+        "width": Math.max(1E-10, s.width),
+        "height": Math.max(1E-10, s.height),
+        "fill": fill.color,
+        "fill-opacity": fill.opacity || null,
+        "stroke": stroke.color,
+        "stroke-opacity": stroke.opacity || null,
+        "stroke-width": stroke.opacity ? s.lineWidth / this.scale : null
+      });
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+pv.SvgScene.dot = function(scenes) {
+  var e = scenes.$g.firstChild;
+  for (var i = 0; i < scenes.length; i++) {
+    var s = scenes[i];
+
+    /* visible */
+    if (!s.visible) continue;
+    var fill = s.fillStyle, stroke = s.strokeStyle;
+    if (!fill.opacity && !stroke.opacity) continue;
+
+    /* points */
+    var radius = s.radius, path = null;
+    switch (s.shape) {
+      case "cross": {
+        path = "M" + -radius + "," + -radius
+            + "L" + radius + "," + radius
+            + "M" + radius + "," + -radius
+            + "L" + -radius + "," + radius;
+        break;
+      }
+      case "triangle": {
+        var h = radius, w = radius * 1.1547; // 2 / Math.sqrt(3)
+        path = "M0," + h
+            + "L" + w +"," + -h
+            + " " + -w + "," + -h
+            + "Z";
+        break;
+      }
+      case "diamond": {
+        radius *= Math.SQRT2;
+        path = "M0," + -radius
+            + "L" + radius + ",0"
+            + " 0," + radius
+            + " " + -radius + ",0"
+            + "Z";
+        break;
+      }
+      case "square": {
+        path = "M" + -radius + "," + -radius
+            + "L" + radius + "," + -radius
+            + " " + radius + "," + radius
+            + " " + -radius + "," + radius
+            + "Z";
+        break;
+      }
+      case "tick": {
+        path = "M0,0L0," + -s.size;
+        break;
+      }
+      case "bar": {
+        path = "M0," + (s.size / 2) + "L0," + -(s.size / 2);
+        break;
+      }
+    }
+
+    /* Use <circle> for circles, <path> for everything else. */
+    var svg = {
+      "shape-rendering": s.antialias ? null : "crispEdges",
+      "pointer-events": s.events,
+      "cursor": s.cursor,
+      "fill": fill.color,
+      "fill-opacity": fill.opacity || null,
+      "stroke": stroke.color,
+      "stroke-opacity": stroke.opacity || null,
+      "stroke-width": stroke.opacity ? s.lineWidth / this.scale : null
+    };
+    if (path) {
+      svg.transform = "translate(" + s.left + "," + s.top + ")";
+      if (s.angle) svg.transform += " rotate(" + 180 * s.angle / Math.PI + ")";
+      svg.d = path;
+      e = this.expect(e, "path", svg);
+    } else {
+      svg.cx = s.left;
+      svg.cy = s.top;
+      svg.r = radius;
+      e = this.expect(e, "circle", svg);
+    }
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+pv.SvgScene.image = function(scenes) {
+  var e = scenes.$g.firstChild;
+  for (var i = 0; i < scenes.length; i++) {
+    var s = scenes[i];
+
+    /* visible */
+    if (!s.visible) continue;
+
+    /* fill */
+    e = this.fill(e, scenes, i);
+
+    /* image */
+    if (s.image) {
+      e = this.expect(e, "foreignObject", {
+          "cursor": s.cursor,
+          "x": s.left,
+          "y": s.top,
+          "width": s.width,
+          "height": s.height
+        });
+      var c = e.firstChild || e.appendChild(document.createElementNS(this.xhtml, "canvas"));
+      c.$scene = {scenes:scenes, index:i};
+      c.style.width = s.width;
+      c.style.height = s.height;
+      c.width = s.imageWidth;
+      c.height = s.imageHeight;
+      c.getContext("2d").putImageData(s.image, 0, 0);
+    } else {
+      e = this.expect(e, "image", {
+          "preserveAspectRatio": "none",
+          "cursor": s.cursor,
+          "x": s.left,
+          "y": s.top,
+          "width": s.width,
+          "height": s.height
+        });
+      e.setAttributeNS(this.xlink, "href", s.url);
+    }
+    e = this.append(e, scenes, i);
+
+    /* stroke */
+    e = this.stroke(e, scenes, i);
+  }
+  return e;
+};
+pv.SvgScene.label = function(scenes) {
+  var e = scenes.$g.firstChild;
+  for (var i = 0; i < scenes.length; i++) {
+    var s = scenes[i];
+
+    /* visible */
+    if (!s.visible) continue;
+    var fill = s.textStyle;
+    if (!fill.opacity || !s.text) continue;
+
+    /* text-baseline, text-align */
+    var x = 0, y = 0, dy = 0, anchor = "start";
+    switch (s.textBaseline) {
+      case "middle": dy = ".35em"; break;
+      case "top": dy = ".71em"; y = s.textMargin; break;
+      case "bottom": y = "-" + s.textMargin; break;
+    }
+    switch (s.textAlign) {
+      case "right": anchor = "end"; x = "-" + s.textMargin; break;
+      case "center": anchor = "middle"; break;
+      case "left": x = s.textMargin; break;
+    }
+
+    e = this.expect(e, "text", {
+        "pointer-events": s.events,
+        "cursor": s.cursor,
+        "x": x,
+        "y": y,
+        "dy": dy,
+        "transform": "translate(" + s.left + "," + s.top + ")"
+            + (s.textAngle ? " rotate(" + 180 * s.textAngle / Math.PI + ")" : "")
+            + (this.scale != 1 ? " scale(" + 1 / this.scale + ")" : ""),
+        "fill": fill.color,
+        "fill-opacity": fill.opacity || null,
+        "text-anchor": anchor
+      }, {
+        "font": s.font,
+        "text-shadow": s.textShadow,
+        "text-decoration": s.textDecoration
+      });
+    if (e.firstChild) e.firstChild.nodeValue = s.text;
+    else e.appendChild(document.createTextNode(s.text));
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+pv.SvgScene.line = function(scenes) {
+  var e = scenes.$g.firstChild;
+  if (scenes.length < 2) return e;
+  var s = scenes[0];
+
+  /* segmented */
+  if (s.segmented) return this.lineSegment(scenes);
+
+  /* visible */
+  if (!s.visible) return e;
+  var fill = s.fillStyle, stroke = s.strokeStyle;
+  if (!fill.opacity && !stroke.opacity) return e;
+
+  /* points */
+  var d = "M" + s.left + "," + s.top;
+
+  if (scenes.length > 2 && (s.interpolate == "basis" || s.interpolate == "cardinal" || s.interpolate == "monotone")) {
+    switch (s.interpolate) {
+      case "basis": d += this.curveBasis(scenes); break;
+      case "cardinal": d += this.curveCardinal(scenes, s.tension); break;
+      case "monotone": d += this.curveMonotone(scenes); break;
+    }
+  } else {
+    for (var i = 1; i < scenes.length; i++) {
+      d += this.pathSegment(scenes[i - 1], scenes[i]);
+    }
+  }
+
+  e = this.expect(e, "path", {
+      "shape-rendering": s.antialias ? null : "crispEdges",
+      "pointer-events": s.events,
+      "cursor": s.cursor,
+      "d": d,
+      "fill": fill.color,
+      "fill-opacity": fill.opacity || null,
+      "stroke": stroke.color,
+      "stroke-opacity": stroke.opacity || null,
+      "stroke-width": stroke.opacity ? s.lineWidth / this.scale : null,
+      "stroke-linejoin": s.lineJoin
+    });
+  return this.append(e, scenes, 0);
+};
+
+pv.SvgScene.lineSegment = function(scenes) {
+  var e = scenes.$g.firstChild;
+
+  var s = scenes[0];
+  var paths;
+  switch (s.interpolate) {
+    case "basis": paths = this.curveBasisSegments(scenes); break;
+    case "cardinal": paths = this.curveCardinalSegments(scenes, s.tension); break;
+    case "monotone": paths = this.curveMonotoneSegments(scenes); break;
+  }
+
+  for (var i = 0, n = scenes.length - 1; i < n; i++) {
+    var s1 = scenes[i], s2 = scenes[i + 1];
+
+    /* visible */
+    if (!s1.visible || !s2.visible) continue;
+    var stroke = s1.strokeStyle, fill = pv.Color.transparent;
+    if (!stroke.opacity) continue;
+
+    /* interpolate */
+    var d;
+    if ((s1.interpolate == "linear") && (s1.lineJoin == "miter")) {
+      fill = stroke;
+      stroke = pv.Color.transparent;
+      d = this.pathJoin(scenes[i - 1], s1, s2, scenes[i + 2]);
+    } else if(paths) {
+      d = paths[i];
+    } else {
+      d = "M" + s1.left + "," + s1.top + this.pathSegment(s1, s2);
+    }
+
+    e = this.expect(e, "path", {
+        "shape-rendering": s1.antialias ? null : "crispEdges",
+        "pointer-events": s1.events,
+        "cursor": s1.cursor,
+        "d": d,
+        "fill": fill.color,
+        "fill-opacity": fill.opacity || null,
+        "stroke": stroke.color,
+        "stroke-opacity": stroke.opacity || null,
+        "stroke-width": stroke.opacity ? s1.lineWidth / this.scale : null,
+        "stroke-linejoin": s1.lineJoin
+      });
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+
+/** @private Returns the path segment for the specified points. */
+pv.SvgScene.pathSegment = function(s1, s2) {
+  var l = 1; // sweep-flag
+  switch (s1.interpolate) {
+    case "polar-reverse":
+      l = 0;
+    case "polar": {
+      var dx = s2.left - s1.left,
+          dy = s2.top - s1.top,
+          e = 1 - s1.eccentricity,
+          r = Math.sqrt(dx * dx + dy * dy) / (2 * e);
+      if ((e <= 0) || (e > 1)) break; // draw a straight line
+      return "A" + r + "," + r + " 0 0," + l + " " + s2.left + "," + s2.top;
+    }
+    case "step-before": return "V" + s2.top + "H" + s2.left;
+    case "step-after": return "H" + s2.left + "V" + s2.top;
+  }
+  return "L" + s2.left + "," + s2.top;
+};
+
+/** @private Line-line intersection, per Akenine-Moller 16.16.1. */
+pv.SvgScene.lineIntersect = function(o1, d1, o2, d2) {
+  return o1.plus(d1.times(o2.minus(o1).dot(d2.perp()) / d1.dot(d2.perp())));
+}
+
+/** @private Returns the miter join path for the specified points. */
+pv.SvgScene.pathJoin = function(s0, s1, s2, s3) {
+  /*
+   * P1-P2 is the current line segment. V is a vector that is perpendicular to
+   * the line segment, and has length lineWidth / 2. ABCD forms the initial
+   * bounding box of the line segment (i.e., the line segment if we were to do
+   * no joins).
+   */
+  var p1 = pv.vector(s1.left, s1.top),
+      p2 = pv.vector(s2.left, s2.top),
+      p = p2.minus(p1),
+      v = p.perp().norm(),
+      w = v.times(s1.lineWidth / (2 * this.scale)),
+      a = p1.plus(w),
+      b = p2.plus(w),
+      c = p2.minus(w),
+      d = p1.minus(w);
+
+  /*
+   * Start join. P0 is the previous line segment's start point. We define the
+   * cutting plane as the average of the vector perpendicular to P0-P1, and
+   * the vector perpendicular to P1-P2. This insures that the cross-section of
+   * the line on the cutting plane is equal if the line-width is unchanged.
+   * Note that we don't implement miter limits, so these can get wild.
+   */
+  if (s0 && s0.visible) {
+    var v1 = p1.minus(s0.left, s0.top).perp().norm().plus(v);
+    d = this.lineIntersect(p1, v1, d, p);
+    a = this.lineIntersect(p1, v1, a, p);
+  }
+
+  /* Similarly, for end join. */
+  if (s3 && s3.visible) {
+    var v2 = pv.vector(s3.left, s3.top).minus(p2).perp().norm().plus(v);
+    c = this.lineIntersect(p2, v2, c, p);
+    b = this.lineIntersect(p2, v2, b, p);
+  }
+
+  return "M" + a.x + "," + a.y
+       + "L" + b.x + "," + b.y
+       + " " + c.x + "," + c.y
+       + " " + d.x + "," + d.y;
+};
+pv.SvgScene.panel = function(scenes) {
+  var g = scenes.$g, e = g && g.firstChild;
+  for (var i = 0; i < scenes.length; i++) {
+    var s = scenes[i];
+
+    /* visible */
+    if (!s.visible) continue;
+
+    /* svg */
+    if (!scenes.parent) {
+      s.canvas.style.display = "inline-block";
+      if (g && (g.parentNode != s.canvas)) {
+        g = s.canvas.firstChild;
+        e = g && g.firstChild;
+      }
+      if (!g) {
+        g = s.canvas.appendChild(this.create("svg"));
+        g.setAttribute("font-size", "10px");
+        g.setAttribute("font-family", "sans-serif");
+        g.setAttribute("fill", "none");
+        g.setAttribute("stroke", "none");
+        g.setAttribute("stroke-width", 1.5);
+        for (var j = 0; j < this.events.length; j++) {
+          g.addEventListener(this.events[j], this.dispatch, false);
+        }
+        e = g.firstChild;
+      }
+      scenes.$g = g;
+      g.setAttribute("width", s.width + s.left + s.right);
+      g.setAttribute("height", s.height + s.top + s.bottom);
+    }
+
+    /* clip (nest children) */
+    if (s.overflow == "hidden") {
+      var id = pv.id().toString(36),
+          c = this.expect(e, "g", {"clip-path": "url(#" + id + ")"});
+      if (!c.parentNode) g.appendChild(c);
+      scenes.$g = g = c;
+      e = c.firstChild;
+
+      e = this.expect(e, "clipPath", {"id": id});
+      var r = e.firstChild || e.appendChild(this.create("rect"));
+      r.setAttribute("x", s.left);
+      r.setAttribute("y", s.top);
+      r.setAttribute("width", s.width);
+      r.setAttribute("height", s.height);
+      if (!e.parentNode) g.appendChild(e);
+      e = e.nextSibling;
+    }
+
+    /* fill */
+    e = this.fill(e, scenes, i);
+
+    /* transform (push) */
+    var k = this.scale,
+        t = s.transform,
+        x = s.left + t.x,
+        y = s.top + t.y;
+    this.scale *= t.k;
+
+    /* children */
+    for (var j = 0; j < s.children.length; j++) {
+      s.children[j].$g = e = this.expect(e, "g", {
+          "transform": "translate(" + x + "," + y + ")"
+              + (t.k != 1 ? " scale(" + t.k + ")" : "")
+        });
+      this.updateAll(s.children[j]);
+      if (!e.parentNode) g.appendChild(e);
+      e = e.nextSibling;
+    }
+
+    /* transform (pop) */
+    this.scale = k;
+
+    /* stroke */
+    e = this.stroke(e, scenes, i);
+
+    /* clip (restore group) */
+    if (s.overflow == "hidden") {
+      scenes.$g = g = c.parentNode;
+      e = c.nextSibling;
+    }
+  }
+  return e;
+};
+
+pv.SvgScene.fill = function(e, scenes, i) {
+  var s = scenes[i], fill = s.fillStyle;
+  if (fill.opacity || s.events == "all") {
+    e = this.expect(e, "rect", {
+        "shape-rendering": s.antialias ? null : "crispEdges",
+        "pointer-events": s.events,
+        "cursor": s.cursor,
+        "x": s.left,
+        "y": s.top,
+        "width": s.width,
+        "height": s.height,
+        "fill": fill.color,
+        "fill-opacity": fill.opacity,
+        "stroke": null
+      });
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+
+pv.SvgScene.stroke = function(e, scenes, i) {
+  var s = scenes[i], stroke = s.strokeStyle;
+  if (stroke.opacity || s.events == "all") {
+    e = this.expect(e, "rect", {
+        "shape-rendering": s.antialias ? null : "crispEdges",
+        "pointer-events": s.events == "all" ? "stroke" : s.events,
+        "cursor": s.cursor,
+        "x": s.left,
+        "y": s.top,
+        "width": Math.max(1E-10, s.width),
+        "height": Math.max(1E-10, s.height),
+        "fill": null,
+        "stroke": stroke.color,
+        "stroke-opacity": stroke.opacity,
+        "stroke-width": s.lineWidth / this.scale
+      });
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+pv.SvgScene.rule = function(scenes) {
+  var e = scenes.$g.firstChild;
+  for (var i = 0; i < scenes.length; i++) {
+    var s = scenes[i];
+
+    /* visible */
+    if (!s.visible) continue;
+    var stroke = s.strokeStyle;
+    if (!stroke.opacity) continue;
+
+    e = this.expect(e, "line", {
+        "shape-rendering": s.antialias ? null : "crispEdges",
+        "pointer-events": s.events,
+        "cursor": s.cursor,
+        "x1": s.left,
+        "y1": s.top,
+        "x2": s.left + s.width,
+        "y2": s.top + s.height,
+        "stroke": stroke.color,
+        "stroke-opacity": stroke.opacity,
+        "stroke-width": s.lineWidth / this.scale
+      });
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+pv.SvgScene.wedge = function(scenes) {
+  var e = scenes.$g.firstChild;
+  for (var i = 0; i < scenes.length; i++) {
+    var s = scenes[i];
+
+    /* visible */
+    if (!s.visible) continue;
+    var fill = s.fillStyle, stroke = s.strokeStyle;
+    if (!fill.opacity && !stroke.opacity) continue;
+
+    /* points */
+    var r1 = s.innerRadius, r2 = s.outerRadius, a = Math.abs(s.angle), p;
+    if (a >= 2 * Math.PI) {
+      if (r1) {
+        p = "M0," + r2
+            + "A" + r2 + "," + r2 + " 0 1,1 0," + (-r2)
+            + "A" + r2 + "," + r2 + " 0 1,1 0," + r2
+            + "M0," + r1
+            + "A" + r1 + "," + r1 + " 0 1,1 0," + (-r1)
+            + "A" + r1 + "," + r1 + " 0 1,1 0," + r1
+            + "Z";
+      } else {
+        p = "M0," + r2
+            + "A" + r2 + "," + r2 + " 0 1,1 0," + (-r2)
+            + "A" + r2 + "," + r2 + " 0 1,1 0," + r2
+            + "Z";
+      }
+    } else {
+      var sa = Math.min(s.startAngle, s.endAngle),
+          ea = Math.max(s.startAngle, s.endAngle),
+          c1 = Math.cos(sa), c2 = Math.cos(ea),
+          s1 = Math.sin(sa), s2 = Math.sin(ea);
+      if (r1) {
+        p = "M" + r2 * c1 + "," + r2 * s1
+            + "A" + r2 + "," + r2 + " 0 "
+            + ((a < Math.PI) ? "0" : "1") + ",1 "
+            + r2 * c2 + "," + r2 * s2
+            + "L" + r1 * c2 + "," + r1 * s2
+            + "A" + r1 + "," + r1 + " 0 "
+            + ((a < Math.PI) ? "0" : "1") + ",0 "
+            + r1 * c1 + "," + r1 * s1 + "Z";
+      } else {
+        p = "M" + r2 * c1 + "," + r2 * s1
+            + "A" + r2 + "," + r2 + " 0 "
+            + ((a < Math.PI) ? "0" : "1") + ",1 "
+            + r2 * c2 + "," + r2 * s2 + "L0,0Z";
+      }
+    }
+
+    e = this.expect(e, "path", {
+        "shape-rendering": s.antialias ? null : "crispEdges",
+        "pointer-events": s.events,
+        "cursor": s.cursor,
+        "transform": "translate(" + s.left + "," + s.top + ")",
+        "d": p,
+        "fill": fill.color,
+        "fill-rule": "evenodd",
+        "fill-opacity": fill.opacity || null,
+        "stroke": stroke.color,
+        "stroke-opacity": stroke.opacity || null,
+        "stroke-width": stroke.opacity ? s.lineWidth / this.scale : null
+      });
+    e = this.append(e, scenes, i);
+  }
+  return e;
+};
+/**
+ * Constructs a new mark with default properties. Marks, with the exception of
+ * the root panel, are not typically constructed directly; instead, they are
+ * added to a panel or an existing mark via {@link pv.Mark#add}.
+ *
+ * @class Represents a data-driven graphical mark. The <tt>Mark</tt> class is
+ * the base class for all graphical marks in Protovis; it does not provide any
+ * specific rendering functionality, but together with {@link Panel} establishes
+ * the core framework.
+ *
+ * <p>Concrete mark types include familiar visual elements such as bars, lines
+ * and labels. Although a bar mark may be used to construct a bar chart, marks
+ * know nothing about charts; it is only through their specification and
+ * composition that charts are produced. These building blocks permit many
+ * combinatorial possibilities.
+ *
+ * <p>Marks are associated with <b>data</b>: a mark is generated once per
+ * associated datum, mapping the datum to visual <b>properties</b> such as
+ * position and color. Thus, a single mark specification represents a set of
+ * visual elements that share the same data and visual encoding. The type of
+ * mark defines the names of properties and their meaning. A property may be
+ * static, ignoring the associated datum and returning a constant; or, it may be
+ * dynamic, derived from the associated datum or index. Such dynamic encodings
+ * can be specified succinctly using anonymous functions. Special properties
+ * called event handlers can be registered to add interactivity.
+ *
+ * <p>Protovis uses <b>inheritance</b> to simplify the specification of related
+ * marks: a new mark can be derived from an existing mark, inheriting its
+ * properties. The new mark can then override properties to specify new
+ * behavior, potentially in terms of the old behavior. In this way, the old mark
+ * serves as the <b>prototype</b> for the new mark. Most mark types share the
+ * same basic properties for consistency and to facilitate inheritance.
+ *
+ * <p>The prioritization of redundant properties is as follows:<ol>
+ *
+ * <li>If the <tt>width</tt> property is not specified (i.e., null), its value
+ * is the width of the parent panel, minus this mark's left and right margins;
+ * the left and right margins are zero if not specified.
+ *
+ * <li>Otherwise, if the <tt>right</tt> margin is not specified, its value is
+ * the width of the parent panel, minus this mark's width and left margin; the
+ * left margin is zero if not specified.
+ *
+ * <li>Otherwise, if the <tt>left</tt> property is not specified, its value is
+ * the width of the parent panel, minus this mark's width and the right margin.
+ *
+ * </ol>This prioritization is then duplicated for the <tt>height</tt>,
+ * <tt>bottom</tt> and <tt>top</tt> properties, respectively.
+ *
+ * <p>While most properties are <i>variable</i>, some mark types, such as lines
+ * and areas, generate a single visual element rather than a distinct visual
+ * element per datum. With these marks, some properties may be <b>fixed</b>.
+ * Fixed properties can vary per mark, but not <i>per datum</i>! These
+ * properties are evaluated solely for the first (0-index) datum, and typically
+ * are specified as a constant. However, it is valid to use a function if the
+ * property varies between panels or is dynamically generated.
+ *
+ * <p>See also the <a href="../../api/">Protovis guide</a>.
+ */
+pv.Mark = function() {
+  /*
+   * TYPE 0 constant defs
+   * TYPE 1 function defs
+   * TYPE 2 constant properties
+   * TYPE 3 function properties
+   * in order of evaluation!
+   */
+  this.$properties = [];
+  this.$handlers = {};
+};
+
+/** @private Records which properties are defined on this mark type. */
+pv.Mark.prototype.properties = {};
+
+/** @private Records the cast function for each property. */
+pv.Mark.cast = {};
+
+/**
+ * @private Defines and registers a property method for the property with the
+ * given name.  This method should be called on a mark class prototype to define
+ * each exposed property. (Note this refers to the JavaScript
+ * <tt>prototype</tt>, not the Protovis mark prototype, which is the {@link
+ * #proto} field.)
+ *
+ * <p>The created property method supports several modes of invocation: <ol>
+ *
+ * <li>If invoked with a <tt>Function</tt> argument, this function is evaluated
+ * for each associated datum. The return value of the function is used as the
+ * computed property value. The context of the function (<tt>this</tt>) is this
+ * mark. The arguments to the function are the associated data of this mark and
+ * any enclosing panels. For example, a linear encoding of numerical data to
+ * height is specified as
+ *
+ * <pre>m.height(function(d) d * 100);</pre>
+ *
+ * The expression <tt>d * 100</tt> will be evaluated for the height property of
+ * each mark instance. The return value of the property method (e.g.,
+ * <tt>m.height</tt>) is this mark (<tt>m</tt>)).<p>
+ *
+ * <li>If invoked with a non-function argument, the property is treated as a
+ * constant. The return value of the property method (e.g., <tt>m.height</tt>)
+ * is this mark.<p>
+ *
+ * <li>If invoked with no arguments, the computed property value for the current
+ * mark instance in the scene graph is returned. This facilitates <i>property
+ * chaining</i>, where one mark's properties are defined in terms of another's.
+ * For example, to offset a mark's location from its prototype, you might say
+ *
+ * <pre>m.top(function() this.proto.top() + 10);</pre>
+ *
+ * Note that the index of the mark being evaluated (in the above example,
+ * <tt>this.proto</tt>) is inherited from the <tt>Mark</tt> class and set by
+ * this mark. So, if the fifth element's top property is being evaluated, the
+ * fifth instance of <tt>this.proto</tt> will similarly be queried for the value
+ * of its top property. If the mark being evaluated has a different number of
+ * instances, or its data is unrelated, the behavior of this method is
+ * undefined. In these cases it may be better to index the <tt>scene</tt>
+ * explicitly to specify the exact instance.
+ *
+ * </ol><p>Property names should follow standard JavaScript method naming
+ * conventions, using lowerCamel-style capitalization.
+ *
+ * <p>In addition to creating the property method, every property is registered
+ * in the {@link #properties} map on the <tt>prototype</tt>. Although this is an
+ * instance field, it is considered immutable and shared by all instances of a
+ * given mark type. The <tt>properties</tt> map can be queried to see if a mark
+ * type defines a particular property, such as width or height.
+ *
+ * @param {string} name the property name.
+ * @param {function} [cast] the cast function for this property.
+ */
+pv.Mark.prototype.property = function(name, cast) {
+  if (!this.hasOwnProperty("properties")) {
+    this.properties = pv.extend(this.properties);
+  }
+  this.properties[name] = true;
+
+  /*
+   * Define the setter-getter globally, since the default behavior should be the
+   * same for all properties, and since the Protovis inheritance chain is
+   * independent of the JavaScript inheritance chain. For example, anchors
+   * define a "name" property that is evaluated on derived marks, even though
+   * those marks don't normally have a name.
+   */
+  pv.Mark.prototype.propertyMethod(name, false, pv.Mark.cast[name] = cast);
+  return this;
+};
+
+/**
+ * @private Defines a setter-getter for the specified property.
+ *
+ * <p>If a cast function has been assigned to the specified property name, the
+ * property function is wrapped by the cast function, or, if a constant is
+ * specified, the constant is immediately cast. Note, however, that if the
+ * property value is null, the cast function is not invoked.
+ *
+ * @param {string} name the property name.
+ * @param {boolean} [def] whether is a property or a def.
+ * @param {function} [cast] the cast function for this property.
+ */
+pv.Mark.prototype.propertyMethod = function(name, def, cast) {
+  if (!cast) cast = pv.Mark.cast[name];
+  this[name] = function(v) {
+
+      /* If this is a def, use it rather than property. */
+      if (def && this.scene) {
+        var defs = this.scene.defs;
+        if (arguments.length) {
+          defs[name] = {
+            id: (v == null) ? 0 : pv.id(),
+            value: ((v != null) && cast) ? cast(v) : v
+          };
+          return this;
+        }
+        return defs[name] ? defs[name].value : null;
+      }
+
+      /* If arguments are specified, set the property value. */
+      if (arguments.length) {
+        var type = !def << 1 | (typeof v == "function");
+        this.propertyValue(name, (type & 1 && cast) ? function() {
+            var x = v.apply(this, arguments);
+            return (x != null) ? cast(x) : null;
+          } : (((v != null) && cast) ? cast(v) : v)).type = type;
+        return this;
+      }
+
+      return this.instance()[name];
+    };
+};
+
+/** @private Sets the value of the property <i>name</i> to <i>v</i>. */
+pv.Mark.prototype.propertyValue = function(name, v) {
+  var properties = this.$properties, p = {name: name, id: pv.id(), value: v};
+  for (var i = 0; i < properties.length; i++) {
+    if (properties[i].name == name) {
+      properties.splice(i, 1);
+      break;
+    }
+  }
+  properties.push(p);
+  return p;
+};
+
+/* Define all global properties. */
+pv.Mark.prototype
+    .property("data")
+    .property("visible", Boolean)
+    .property("left", Number)
+    .property("right", Number)
+    .property("top", Number)
+    .property("bottom", Number)
+    .property("cursor", String)
+    .property("title", String)
+    .property("reverse", Boolean)
+    .property("antialias", Boolean)
+    .property("events", String);
+
+/**
+ * The mark type; a lower camelCase name. The type name controls rendering
+ * behavior, and unless the rendering engine is extended, must be one of the
+ * built-in concrete mark types: area, bar, dot, image, label, line, panel,
+ * rule, or wedge.
+ *
+ * @type string
+ * @name pv.Mark.prototype.type
+ */
+
+/**
+ * The mark prototype, possibly undefined, from which to inherit property
+ * functions. The mark prototype is not necessarily of the same type as this
+ * mark. Any properties defined on this mark will override properties inherited
+ * either from the prototype or from the type-specific defaults.
+ *
+ * @type pv.Mark
+ * @name pv.Mark.prototype.proto
+ */
+
+/**
+ * The mark anchor target, possibly undefined.
+ *
+ * @type pv.Mark
+ * @name pv.Mark.prototype.target
+ */
+
+/**
+ * The enclosing parent panel. The parent panel is generally undefined only for
+ * the root panel; however, it is possible to create "offscreen" marks that are
+ * used only for inheritance purposes.
+ *
+ * @type pv.Panel
+ * @name pv.Mark.prototype.parent
+ */
+
+/**
+ * The child index. -1 if the enclosing parent panel is null; otherwise, the
+ * zero-based index of this mark into the parent panel's <tt>children</tt> array.
+ *
+ * @type number
+ */
+pv.Mark.prototype.childIndex = -1;
+
+/**
+ * The mark index. The value of this field depends on which instance (i.e.,
+ * which element of the data array) is currently being evaluated. During the
+ * build phase, the index is incremented over each datum; when handling events,
+ * the index is set to the instance that triggered the event.
+ *
+ * @type number
+ */
+pv.Mark.prototype.index = -1;
+
+/**
+ * The current scale factor, based on any enclosing transforms. The current
+ * scale can be used to create scale-independent graphics. For example, to
+ * define a dot that has a radius of 10 irrespective of any zooming, say:
+ *
+ * <pre>dot.radius(function() 10 / this.scale)</pre>
+ *
+ * Note that the stroke width and font size are defined irrespective of scale
+ * (i.e., in screen space) already. Also note that when a transform is applied
+ * to a panel, the scale affects only the child marks, not the panel itself.
+ *
+ * @type number
+ * @see pv.Panel#transform
+ */
+pv.Mark.prototype.scale = 1;
+
+/**
+ * @private The scene graph. The scene graph is an array of objects; each object
+ * (or "node") corresponds to an instance of this mark and an element in the
+ * data array. The scene graph can be traversed to lookup previously-evaluated
+ * properties.
+ *
+ * @name pv.Mark.prototype.scene
+ */
+
+/**
+ * The root parent panel. This may be undefined for "offscreen" marks that are
+ * created for inheritance purposes only.
+ *
+ * @type pv.Panel
+ * @name pv.Mark.prototype.root
+ */
+
+/**
+ * The data property; an array of objects. The size of the array determines the
+ * number of marks that will be instantiated; each element in the array will be
+ * passed to property functions to compute the property values. Typically, the
+ * data property is specified as a constant array, such as
+ *
+ * <pre>m.data([1, 2, 3, 4, 5]);</pre>
+ *
+ * However, it is perfectly acceptable to define the data property as a
+ * function. This function might compute the data dynamically, allowing
+ * different data to be used per enclosing panel. For instance, in the stacked
+ * area graph example (see {@link #scene}), the data function on the area mark
+ * dereferences each series.
+ *
+ * @type array
+ * @name pv.Mark.prototype.data
+ */
+
+/**
+ * The visible property; a boolean determining whether or not the mark instance
+ * is visible. If a mark instance is not visible, its other properties will not
+ * be evaluated. Similarly, for panels no child marks will be rendered.
+ *
+ * @type boolean
+ * @name pv.Mark.prototype.visible
+ */
+
+/**
+ * The left margin; the distance, in pixels, between the left edge of the
+ * enclosing panel and the left edge of this mark. Note that in some cases this
+ * property may be redundant with the right property, or with the conjunction of
+ * right and width.
+ *
+ * @type number
+ * @name pv.Mark.prototype.left
+ */
+
+/**
+ * The right margin; the distance, in pixels, between the right edge of the
+ * enclosing panel and the right edge of this mark. Note that in some cases this
+ * property may be redundant with the left property, or with the conjunction of
+ * left and width.
+ *
+ * @type number
+ * @name pv.Mark.prototype.right
+ */
+
+/**
+ * The top margin; the distance, in pixels, between the top edge of the
+ * enclosing panel and the top edge of this mark. Note that in some cases this
+ * property may be redundant with the bottom property, or with the conjunction
+ * of bottom and height.
+ *
+ * @type number
+ * @name pv.Mark.prototype.top
+ */
+
+/**
+ * The bottom margin; the distance, in pixels, between the bottom edge of the
+ * enclosing panel and the bottom edge of this mark. Note that in some cases
+ * this property may be redundant with the top property, or with the conjunction
+ * of top and height.
+ *
+ * @type number
+ * @name pv.Mark.prototype.bottom
+ */
+
+/**
+ * The cursor property; corresponds to the CSS cursor property. This is
+ * typically used in conjunction with event handlers to indicate interactivity.
+ *
+ * @type string
+ * @name pv.Mark.prototype.cursor
+ * @see <a href="http://www.w3.org/TR/CSS2/ui.html#propdef-cursor">CSS2 cursor</a>
+ */
+
+/**
+ * The title property; corresponds to the HTML/SVG title property, allowing the
+ * general of simple plain text tooltips.
+ *
+ * @type string
+ * @name pv.Mark.prototype.title
+ */
+
+/**
+ * The events property; corresponds to the SVG pointer-events property,
+ * specifying how the mark should participate in mouse events. The default value
+ * is "painted". Supported values are:
+ *
+ * <p>"painted": The given mark may receive events when the mouse is over a
+ * "painted" area. The painted areas are the interior (i.e., fill) of the mark
+ * if a 'fillStyle' is specified, and the perimeter (i.e., stroke) of the mark
+ * if a 'strokeStyle' is specified.
+ *
+ * <p>"all": The given mark may receive events when the mouse is over either the
+ * interior (i.e., fill) or the perimeter (i.e., stroke) of the mark, regardless
+ * of the specified fillStyle and strokeStyle.
+ *
+ * <p>"none": The given mark may not receive events.
+ *
+ * @type string
+ * @name pv.Mark.prototype.events
+ */
+
+/**
+ * The reverse property; a boolean determining whether marks are ordered from
+ * front-to-back or back-to-front. SVG does not support explicit z-ordering;
+ * shapes are rendered in the order they appear. Thus, by default, marks are
+ * rendered in data order. Setting the reverse property to false reverses the
+ * order in which they are rendered; however, the properties are still evaluated
+ * (i.e., built) in forward order.
+ *
+ * @type boolean
+ * @name pv.Mark.prototype.reverse
+ */
+
+/**
+ * Default properties for all mark types. By default, the data array is the
+ * parent data as a single-element array; if the data property is not specified,
+ * this causes each mark to be instantiated as a singleton with the parents
+ * datum. The visible property is true by default, and the reverse property is
+ * false.
+ *
+ * @type pv.Mark
+ */
+pv.Mark.prototype.defaults = new pv.Mark()
+    .data(function(d) { return [d]; })
+    .visible(true)
+    .antialias(true)
+    .events("painted");
+
+/**
+ * Sets the prototype of this mark to the specified mark. Any properties not
+ * defined on this mark may be inherited from the specified prototype mark, or
+ * its prototype, and so on. The prototype mark need not be the same type of
+ * mark as this mark. (Note that for inheritance to be useful, properties with
+ * the same name on different mark types should have equivalent meaning.)
+ *
+ * @param {pv.Mark} proto the new prototype.
+ * @returns {pv.Mark} this mark.
+ * @see #add
+ */
+pv.Mark.prototype.extend = function(proto) {
+  this.proto = proto;
+  this.target = proto.target;
+  return this;
+};
+
+/**
+ * Adds a new mark of the specified type to the enclosing parent panel, whilst
+ * simultaneously setting the prototype of the new mark to be this mark.
+ *
+ * @param {function} type the type of mark to add; a constructor, such as
+ * <tt>pv.Bar</tt>.
+ * @returns {pv.Mark} the new mark.
+ * @see #extend
+ */
+pv.Mark.prototype.add = function(type) {
+  return this.parent.add(type).extend(this);
+};
+
+/**
+ * Defines a custom property on this mark. Custom properties are currently
+ * fixed, in that they are initialized once per mark set (i.e., per parent panel
+ * instance). Custom properties can be used to store local state for the mark,
+ * such as data needed by other properties (e.g., a custom scale) or interaction
+ * state.
+ *
+ * <p>WARNING We plan on changing this feature in a future release to define
+ * standard properties, as opposed to <i>fixed</i> properties that behave
+ * idiosyncratically within event handlers. Furthermore, we recommend storing
+ * state in an external data structure, rather than tying it to the
+ * visualization specification as with defs.
+ *
+ * @param {string} name the name of the local variable.
+ * @param {function} [v] an optional initializer; may be a constant or a
+ * function.
+ */
+pv.Mark.prototype.def = function(name, v) {
+  this.propertyMethod(name, true);
+  return this[name](arguments.length > 1 ? v : null);
+};
+
+/**
+ * Returns an anchor with the specified name. All marks support the five
+ * standard anchor names:<ul>
+ *
+ * <li>top
+ * <li>left
+ * <li>center
+ * <li>bottom
+ * <li>right
+ *
+ * </ul>In addition to positioning properties (left, right, top bottom), the
+ * anchors support text rendering properties (text-align, text-baseline). Text is
+ * rendered to appear inside the mark by default.
+ *
+ * <p>To facilitate stacking, anchors are defined in terms of their opposite
+ * edge. For example, the top anchor defines the bottom property, such that the
+ * mark extends upwards; the bottom anchor instead defines the top property,
+ * such that the mark extends downwards. See also {@link pv.Layout.Stack}.
+ *
+ * <p>While anchor names are typically constants, the anchor name is a true
+ * property, which means you can specify a function to compute the anchor name
+ * dynamically. See the {@link pv.Anchor#name} property for details.
+ *
+ * @param {string} name the anchor name; either a string or a property function.
+ * @returns {pv.Anchor} the new anchor.
+ */
+pv.Mark.prototype.anchor = function(name) {
+  if (!name) name = "center"; // default anchor name
+  return new pv.Anchor(this)
+    .name(name)
+    .data(function() {
+        return this.scene.target.map(function(s) { return s.data; });
+      })
+    .visible(function() {
+        return this.scene.target[this.index].visible;
+      })
+    .left(function() {
+        var s = this.scene.target[this.index], w = s.width || 0;
+        switch (this.name()) {
+          case "bottom":
+          case "top":
+          case "center": return s.left + w / 2;
+          case "left": return null;
+        }
+        return s.left + w;
+      })
+    .top(function() {
+        var s = this.scene.target[this.index], h = s.height || 0;
+        switch (this.name()) {
+          case "left":
+          case "right":
+          case "center": return s.top + h / 2;
+          case "top": return null;
+        }
+        return s.top + h;
+      })
+    .right(function() {
+        var s = this.scene.target[this.index];
+        return this.name() == "left" ? s.right + (s.width || 0) : null;
+      })
+    .bottom(function() {
+        var s = this.scene.target[this.index];
+        return this.name() == "top" ? s.bottom + (s.height || 0) : null;
+      })
+    .textAlign(function() {
+        switch (this.name()) {
+          case "bottom":
+          case "top":
+          case "center": return "center";
+          case "right": return "right";
+        }
+        return "left";
+      })
+    .textBaseline(function() {
+        switch (this.name()) {
+          case "right":
+          case "left":
+          case "center": return "middle";
+          case "top": return "top";
+        }
+        return "bottom";
+      });
+};
+
+/** @deprecated Replaced by {@link #target}. */
+pv.Mark.prototype.anchorTarget = function() {
+  return this.target;
+};
+
+/**
+ * Alias for setting the left, right, top and bottom properties simultaneously.
+ *
+ * @see #left
+ * @see #right
+ * @see #top
+ * @see #bottom
+ * @returns {pv.Mark} this.
+ */
+pv.Mark.prototype.margin = function(n) {
+  return this.left(n).right(n).top(n).bottom(n);
+};
+
+/**
+ * @private Returns the current instance of this mark in the scene graph. This
+ * is typically equivalent to <tt>this.scene[this.index]</tt>, however if the
+ * scene or index is unset, the default instance of the mark is returned. If no
+ * default is set, the default is the last instance. Similarly, if the scene or
+ * index of the parent panel is unset, the default instance of this mark in the
+ * last instance of the enclosing panel is returned, and so on.
+ *
+ * @returns a node in the scene graph.
+ */
+pv.Mark.prototype.instance = function(defaultIndex) {
+  var scene = this.scene || this.parent.instance(-1).children[this.childIndex],
+      index = !arguments.length || this.hasOwnProperty("index") ? this.index : defaultIndex;
+  return scene[index < 0 ? scene.length - 1 : index];
+};
+
+/**
+ * @private Find the instances of this mark that match source.
+ *
+ * @see pv.Anchor
+ */
+pv.Mark.prototype.instances = function(source) {
+  var mark = this, index = [], scene;
+
+  /* Mirrored descent. */
+  while (!(scene = mark.scene)) {
+    source = source.parent;
+    index.push({index: source.index, childIndex: mark.childIndex});
+    mark = mark.parent;
+  }
+  while (index.length) {
+    var i = index.pop();
+    scene = scene[i.index].children[i.childIndex];
+  }
+
+  /*
+   * When the anchor target is also an ancestor, as in the case of adding
+   * to a panel anchor, only generate one instance per panel. Also, set
+   * the margins to zero, since they are offset by the enclosing panel.
+   */
+  if (this.hasOwnProperty("index")) {
+    var s = pv.extend(scene[this.index]);
+    s.right = s.top = s.left = s.bottom = 0;
+    return [s];
+  }
+  return scene;
+};
+
+/**
+ * @private Returns the first instance of this mark in the scene graph. This
+ * method can only be called when the mark is bound to the scene graph (for
+ * example, from an event handler, or within a property function).
+ *
+ * @returns a node in the scene graph.
+ */
+pv.Mark.prototype.first = function() {
+  return this.scene[0];
+};
+
+/**
+ * @private Returns the last instance of this mark in the scene graph. This
+ * method can only be called when the mark is bound to the scene graph (for
+ * example, from an event handler, or within a property function). In addition,
+ * note that mark instances are built sequentially, so the last instance of this
+ * mark may not yet be constructed.
+ *
+ * @returns a node in the scene graph.
+ */
+pv.Mark.prototype.last = function() {
+  return this.scene[this.scene.length - 1];
+};
+
+/**
+ * @private Returns the previous instance of this mark in the scene graph, or
+ * null if this is the first instance.
+ *
+ * @returns a node in the scene graph, or null.
+ */
+pv.Mark.prototype.sibling = function() {
+  return (this.index == 0) ? null : this.scene[this.index - 1];
+};
+
+/**
+ * @private Returns the current instance in the scene graph of this mark, in the
+ * previous instance of the enclosing parent panel. May return null if this
+ * instance could not be found.
+ *
+ * @returns a node in the scene graph, or null.
+ */
+pv.Mark.prototype.cousin = function() {
+  var p = this.parent, s = p && p.sibling();
+  return (s && s.children) ? s.children[this.childIndex][this.index] : null;
+};
+
+/**
+ * Renders this mark, including recursively rendering all child marks if this is
+ * a panel. This method finds all instances of this mark and renders them. This
+ * method descends recursively to the level of the mark to be rendered, finding
+ * all visible instances of the mark. After the marks are rendered, the scene
+ * and index attributes are removed from the mark to restore them to a clean
+ * state.
+ *
+ * <p>If an enclosing panel has an index property set (as is the case inside in
+ * an event handler), then only instances of this mark inside the given instance
+ * of the panel will be rendered; otherwise, all visible instances of the mark
+ * will be rendered.
+ */
+pv.Mark.prototype.render = function() {
+  var parent = this.parent,
+      stack = pv.Mark.stack;
+
+  /* For the first render, take it from the top. */
+  if (parent && !this.root.scene) {
+    this.root.render();
+    return;
+  }
+
+  /* Record the path to this mark. */
+  var indexes = [];
+  for (var mark = this; mark.parent; mark = mark.parent) {
+    indexes.unshift(mark.childIndex);
+  }
+
+  /** @private */
+  function render(mark, depth, scale) {
+    mark.scale = scale;
+    if (depth < indexes.length) {
+      stack.unshift(null);
+      if (mark.hasOwnProperty("index")) {
+        renderInstance(mark, depth, scale);
+      } else {
+        for (var i = 0, n = mark.scene.length; i < n; i++) {
+          mark.index = i;
+          renderInstance(mark, depth, scale);
+        }
+        delete mark.index;
+      }
+      stack.shift();
+    } else {
+      mark.build();
+
+      /*
+       * In the update phase, the scene is rendered by creating and updating
+       * elements and attributes in the SVG image. No properties are evaluated
+       * during the update phase; instead the values computed previously in the
+       * build phase are simply translated into SVG. The update phase is
+       * decoupled (see pv.Scene) to allow different rendering engines.
+       */
+      pv.Scene.scale = scale;
+      pv.Scene.updateAll(mark.scene);
+    }
+    delete mark.scale;
+  }
+
+  /**
+   * @private Recursively renders the current instance of the specified mark.
+   * This is slightly tricky because `index` and `scene` properties may or may
+   * not already be set; if they are set, it means we are rendering only a
+   * specific instance; if they are unset, we are rendering all instances.
+   * Furthermore, we must preserve the original context of these properties when
+   * rendering completes.
+   *
+   * <p>Another tricky aspect is that the `scene` attribute should be set for
+   * any preceding children, so as to allow property chaining. This is
+   * consistent with first-pass rendering.
+   */
+  function renderInstance(mark, depth, scale) {
+    var s = mark.scene[mark.index], i;
+    if (s.visible) {
+      var childIndex = indexes[depth],
+          child = mark.children[childIndex];
+
+      /* Set preceding child scenes. */
+      for (i = 0; i < childIndex; i++) {
+        mark.children[i].scene = s.children[i];
+      }
+
+      /* Set current child scene, if necessary. */
+      stack[0] = s.data;
+      if (child.scene) {
+        render(child, depth + 1, scale * s.transform.k);
+      } else {
+        child.scene = s.children[childIndex];
+        render(child, depth + 1, scale * s.transform.k);
+        delete child.scene;
+      }
+
+      /* Clear preceding child scenes. */
+      for (i = 0; i < childIndex; i++) {
+        delete mark.children[i].scene;
+      }
+    }
+  }
+
+  /* Bind this mark's property definitions. */
+  this.bind();
+
+  /* The render context is the first ancestor with an explicit index. */
+  while (parent && !parent.hasOwnProperty("index")) parent = parent.parent;
+
+  /* Recursively render all instances of this mark. */
+  this.context(
+      parent ? parent.scene : undefined,
+      parent ? parent.index : -1,
+      function() { render(this.root, 0, 1); });
+};
+
+/** @private Stores the current data stack. */
+pv.Mark.stack = [];
+
+/**
+ * @private In the bind phase, inherited property definitions are cached so they
+ * do not need to be queried during build.
+ */
+pv.Mark.prototype.bind = function() {
+  var seen = {}, types = [[], [], [], []], data, visible;
+
+  /** Scans the proto chain for the specified mark. */
+  function bind(mark) {
+    do {
+      var properties = mark.$properties;
+      for (var i = properties.length - 1; i >= 0 ; i--) {
+        var p = properties[i];
+        if (!(p.name in seen)) {
+          seen[p.name] = p;
+          switch (p.name) {
+            case "data": data = p; break;
+            case "visible": visible = p; break;
+            default: types[p.type].push(p); break;
+          }
+        }
+      }
+    } while (mark = mark.proto);
+  }
+
+  /* Scan the proto chain for all defined properties. */
+  bind(this);
+  bind(this.defaults);
+  types[1].reverse();
+  types[3].reverse();
+
+  /* Any undefined properties are null. */
+  var mark = this;
+  do for (var name in mark.properties) {
+    if (!(name in seen)) {
+      types[2].push(seen[name] = {name: name, type: 2, value: null});
+    }
+  } while (mark = mark.proto);
+
+  /* Define setter-getter for inherited defs. */
+  var defs = types[0].concat(types[1]);
+  for (var i = 0; i < defs.length; i++) {
+    this.propertyMethod(defs[i].name, true);
+  }
+
+  /* Setup binds to evaluate constants before functions. */
+  this.binds = {
+    properties: seen,
+    data: data,
+    defs: defs,
+    required: [visible],
+    optional: pv.blend(types)
+  };
+};
+
+/**
+ * @private Evaluates properties and computes implied properties. Properties are
+ * stored in the {@link #scene} array for each instance of this mark.
+ *
+ * <p>As marks are built recursively, the {@link #index} property is updated to
+ * match the current index into the data array for each mark. Note that the
+ * index property is only set for the mark currently being built and its
+ * enclosing parent panels. The index property for other marks is unset, but is
+ * inherited from the global <tt>Mark</tt> class prototype. This allows mark
+ * properties to refer to properties on other marks <i>in the same panel</i>
+ * conveniently; however, in general it is better to reference mark instances
+ * specifically through the scene graph rather than depending on the magical
+ * behavior of {@link #index}.
+ *
+ * <p>The root scene array has a special property, <tt>data</tt>, which stores
+ * the current data stack. The first element in this stack is the current datum,
+ * followed by the datum of the enclosing parent panel, and so on. The data
+ * stack should not be accessed directly; instead, property functions are passed
+ * the current data stack as arguments.
+ *
+ * <p>The evaluation of the <tt>data</tt> and <tt>visible</tt> properties is
+ * special. The <tt>data</tt> property is evaluated first; unlike the other
+ * properties, the data stack is from the parent panel, rather than the current
+ * mark, since the data is not defined until the data property is evaluated.
+ * The <tt>visisble</tt> property is subsequently evaluated for each instance;
+ * only if true will the {@link #buildInstance} method be called, evaluating
+ * other properties and recursively building the scene graph.
+ *
+ * <p>If this mark is being re-built, any old instances of this mark that no
+ * longer exist (because the new data array contains fewer elements) will be
+ * cleared using {@link #clearInstance}.
+ *
+ * @param parent the instance of the parent panel from the scene graph.
+ */
+pv.Mark.prototype.build = function() {
+  var scene = this.scene, stack = pv.Mark.stack;
+  if (!scene) {
+    scene = this.scene = [];
+    scene.mark = this;
+    scene.type = this.type;
+    scene.childIndex = this.childIndex;
+    if (this.parent) {
+      scene.parent = this.parent.scene;
+      scene.parentIndex = this.parent.index;
+    }
+  }
+
+  /* Resolve anchor target. */
+  if (this.target) scene.target = this.target.instances(scene);
+
+  /* Evaluate defs. */
+  if (this.binds.defs.length) {
+    var defs = scene.defs;
+    if (!defs) scene.defs = defs = {};
+    for (var i = 0; i < this.binds.defs.length; i++) {
+      var p = this.binds.defs[i], d = defs[p.name];
+      if (!d || (p.id > d.id)) {
+        defs[p.name] = {
+          id: 0, // this def will be re-evaluated on next build
+          value: (p.type & 1) ? p.value.apply(this, stack) : p.value
+        };
+      }
+    }
+  }
+
+  /* Evaluate special data property. */
+  var data = this.binds.data;
+  data = data.type & 1 ? data.value.apply(this, stack) : data.value;
+
+  /* Create, update and delete scene nodes. */
+  stack.unshift(null);
+  scene.length = data.length;
+  for (var i = 0; i < data.length; i++) {
+    pv.Mark.prototype.index = this.index = i;
+    var s = scene[i];
+    if (!s) scene[i] = s = {};
+    s.data = stack[0] = data[i];
+    this.buildInstance(s);
+  }
+  pv.Mark.prototype.index = -1;
+  delete this.index;
+  stack.shift();
+
+  return this;
+};
+
+/**
+ * @private Evaluates the specified array of properties for the specified
+ * instance <tt>s</tt> in the scene graph.
+ *
+ * @param s a node in the scene graph; the instance of the mark to build.
+ * @param properties an array of properties.
+ */
+pv.Mark.prototype.buildProperties = function(s, properties) {
+  for (var i = 0, n = properties.length; i < n; i++) {
+    var p = properties[i], v = p.value; // assume case 2 (constant)
+    switch (p.type) {
+      case 0:
+      case 1: v = this.scene.defs[p.name].value; break;
+      case 3: v = v.apply(this, pv.Mark.stack); break;
+    }
+    s[p.name] = v;
+  }
+};
+
+/**
+ * @private Evaluates all of the properties for this mark for the specified
+ * instance <tt>s</tt> in the scene graph. The set of properties to evaluate is
+ * retrieved from the {@link #properties} array for this mark type (see {@link
+ * #type}).  After these properties are evaluated, any <b>implied</b> properties
+ * may be computed by the mark and set on the scene graph; see
+ * {@link #buildImplied}.
+ *
+ * <p>For panels, this method recursively builds the scene graph for all child
+ * marks as well. In general, this method should not need to be overridden by
+ * concrete mark types.
+ *
+ * @param s a node in the scene graph; the instance of the mark to build.
+ */
+pv.Mark.prototype.buildInstance = function(s) {
+  this.buildProperties(s, this.binds.required);
+  if (s.visible) {
+    this.buildProperties(s, this.binds.optional);
+    this.buildImplied(s);
+  }
+};
+
+/**
+ * @private Computes the implied properties for this mark for the specified
+ * instance <tt>s</tt> in the scene graph. Implied properties are those with
+ * dependencies on multiple other properties; for example, the width property
+ * may be implied if the left and right properties are set. This method can be
+ * overridden by concrete mark types to define new implied properties, if
+ * necessary.
+ *
+ * @param s a node in the scene graph; the instance of the mark to build.
+ */
+pv.Mark.prototype.buildImplied = function(s) {
+  var l = s.left;
+  var r = s.right;
+  var t = s.top;
+  var b = s.bottom;
+
+  /* Assume width and height are zero if not supported by this mark type. */
+  var p = this.properties;
+  var w = p.width ? s.width : 0;
+  var h = p.height ? s.height : 0;
+
+  /* Compute implied width, right and left. */
+  var width = this.parent ? this.parent.width() : (w + l + r);
+  if (w == null) {
+    w = width - (r = r || 0) - (l = l || 0);
+  } else if (r == null) {
+    if (l == null) {
+      l = r = (width - w) / 2;
+    } else {
+      r = width - w - (l = l || 0);
+    }
+  } else if (l == null) {
+    l = width - w - r;
+  }
+
+  /* Compute implied height, bottom and top. */
+  var height = this.parent ? this.parent.height() : (h + t + b);
+  if (h == null) {
+    h = height - (t = t || 0) - (b = b || 0);
+  } else if (b == null) {
+    if (t == null) {
+      b = t = (height - h) / 2;
+    } else {
+      b = height - h - (t = t || 0);
+    }
+  } else if (t == null) {
+    t = height - h - b;
+  }
+
+  s.left = l;
+  s.right = r;
+  s.top = t;
+  s.bottom = b;
+
+  /* Only set width and height if they are supported by this mark type. */
+  if (p.width) s.width = w;
+  if (p.height) s.height = h;
+
+  /* Set any null colors to pv.Color.transparent. */
+  if (p.textStyle && !s.textStyle) s.textStyle = pv.Color.transparent;
+  if (p.fillStyle && !s.fillStyle) s.fillStyle = pv.Color.transparent;
+  if (p.strokeStyle && !s.strokeStyle) s.strokeStyle = pv.Color.transparent;
+};
+
+/**
+ * Returns the current location of the mouse (cursor) relative to this mark's
+ * parent. The <i>x</i> coordinate corresponds to the left margin, while the
+ * <i>y</i> coordinate corresponds to the top margin.
+ *
+ * @returns {pv.Vector} the mouse location.
+ */
+pv.Mark.prototype.mouse = function() {
+
+  /* Compute xy-coordinates relative to the panel. */
+  var x = pv.event.pageX || 0,
+      y = pv.event.pageY || 0,
+      n = this.root.canvas();
+  do {
+    x -= n.offsetLeft;
+    y -= n.offsetTop;
+  } while (n = n.offsetParent);
+
+  /* Compute the inverse transform of all enclosing panels. */
+  var t = pv.Transform.identity,
+      p = this.properties.transform ? this : this.parent,
+      pz = [];
+  do { pz.push(p); } while (p = p.parent);
+  while (p = pz.pop()) t = t.translate(p.left(), p.top()).times(p.transform());
+  t = t.invert();
+
+  return pv.vector(x * t.k + t.x, y * t.k + t.y);
+};
+
+/**
+ * Registers an event handler for the specified event type with this mark. When
+ * an event of the specified type is triggered, the specified handler will be
+ * invoked. The handler is invoked in a similar method to property functions:
+ * the context is <tt>this</tt> mark instance, and the arguments are the full
+ * data stack. Event handlers can use property methods to manipulate the display
+ * properties of the mark:
+ *
+ * <pre>m.event("click", function() this.fillStyle("red"));</pre>
+ *
+ * Alternatively, the external data can be manipulated and the visualization
+ * redrawn:
+ *
+ * <pre>m.event("click", function(d) {
+ *     data = all.filter(function(k) k.name == d);
+ *     vis.render();
+ *   });</pre>
+ *
+ * The return value of the event handler determines which mark gets re-rendered.
+ * Use defs ({@link #def}) to set temporary state from event handlers.
+ *
+ * <p>The complete set of event types is defined by SVG; see the reference
+ * below. The set of supported event types is:<ul>
+ *
+ * <li>click
+ * <li>mousedown
+ * <li>mouseup
+ * <li>mouseover
+ * <li>mousemove
+ * <li>mouseout
+ *
+ * </ul>Since Protovis does not specify any concept of focus, it does not
+ * support key events; these should be handled outside the visualization using
+ * standard JavaScript. In the future, support for interaction may be extended
+ * to support additional event types, particularly those most relevant to
+ * interactive visualization, such as selection.
+ *
+ * <p>TODO In the current implementation, event handlers are not inherited from
+ * prototype marks. They must be defined explicitly on each interactive mark. In
+ * addition, only one event handler for a given event type can be defined; when
+ * specifying multiple event handlers for the same type, only the last one will
+ * be used.
+ *
+ * @see <a href="http://www.w3.org/TR/SVGTiny12/interact.html#SVGEvents">SVG events</a>
+ * @param {string} type the event type.
+ * @param {function} handler the event handler.
+ * @returns {pv.Mark} this.
+ */
+pv.Mark.prototype.event = function(type, handler) {
+  this.$handlers[type] = pv.functor(handler);
+  return this;
+};
+
+/** @private Evaluates the function <i>f</i> with the specified context. */
+pv.Mark.prototype.context = function(scene, index, f) {
+  var proto = pv.Mark.prototype,
+      stack = pv.Mark.stack,
+      oscene = pv.Mark.scene,
+      oindex = proto.index;
+
+  /** @private Sets the context. */
+  function apply(scene, index) {
+    pv.Mark.scene = scene;
+    proto.index = index;
+    if (!scene) return;
+
+    var that = scene.mark,
+        mark = that,
+        ancestors = [];
+
+    /* Set ancestors' scene and index; populate data stack. */
+    do {
+      ancestors.push(mark);
+      stack.push(scene[index].data);
+      mark.index = index;
+      mark.scene = scene;
+      index = scene.parentIndex;
+      scene = scene.parent;
+    } while (mark = mark.parent);
+
+    /* Set ancestors' scale; requires top-down. */
+    for (var i = ancestors.length - 1, k = 1; i > 0; i--) {
+      mark = ancestors[i];
+      mark.scale = k;
+      k *= mark.scene[mark.index].transform.k;
+    }
+
+    /* Set children's scene and scale. */
+    if (that.children) for (var i = 0, n = that.children.length; i < n; i++) {
+      mark = that.children[i];
+      mark.scene = that.scene[that.index].children[i];
+      mark.scale = k;
+    }
+  }
+
+  /** @private Clears the context. */
+  function clear(scene, index) {
+    if (!scene) return;
+    var that = scene.mark,
+        mark;
+
+    /* Reset children. */
+    if (that.children) for (var i = 0, n = that.children.length; i < n; i++) {
+      mark = that.children[i];
+      delete mark.scene;
+      delete mark.scale;
+    }
+
+    /* Reset ancestors. */
+    mark = that;
+    do {
+      stack.pop();
+      if (mark.parent) {
+        delete mark.scene;
+        delete mark.scale;
+      }
+      delete mark.index;
+    } while (mark = mark.parent);
+  }
+
+  /* Context switch, invoke the function, then switch back. */
+  clear(oscene, oindex);
+  apply(scene, index);
+  try {
+    f.apply(this, stack);
+  } finally {
+    clear(scene, index);
+    apply(oscene, oindex);
+  }
+};
+
+/** @private Execute the event listener, then re-render. */
+pv.Mark.dispatch = function(type, scene, index) {
+  var m = scene.mark, p = scene.parent, l = m.$handlers[type];
+  if (!l) return p && pv.Mark.dispatch(type, p, scene.parentIndex);
+  m.context(scene, index, function() {
+      m = l.apply(m, pv.Mark.stack);
+      if (m && m.render) m.render();
+    });
+  return true;
+};
+/**
+ * Constructs a new mark anchor with default properties.
+ *
+ * @class Represents an anchor on a given mark. An anchor is itself a mark, but
+ * without a visual representation. It serves only to provide useful default
+ * properties that can be inherited by other marks. Each type of mark can define
+ * any number of named anchors for convenience. If the concrete mark type does
+ * not define an anchor implementation specifically, one will be inherited from
+ * the mark's parent class.
+ *
+ * <p>For example, the bar mark provides anchors for its four sides: left,
+ * right, top and bottom. Adding a label to the top anchor of a bar,
+ *
+ * <pre>bar.anchor("top").add(pv.Label);</pre>
+ *
+ * will render a text label on the top edge of the bar; the top anchor defines
+ * the appropriate position properties (top and left), as well as text-rendering
+ * properties for convenience (textAlign and textBaseline).
+ *
+ * <p>Note that anchors do not <i>inherit</i> from their targets; the positional
+ * properties are copied from the scene graph, which guarantees that the anchors
+ * are positioned correctly, even if the positional properties are not defined
+ * deterministically. (In addition, it also improves performance by avoiding
+ * re-evaluating expensive properties.) If you want the anchor to inherit from
+ * the target, use {@link pv.Mark#extend} before adding. For example:
+ *
+ * <pre>bar.anchor("top").extend(bar).add(pv.Label);</pre>
+ *
+ * The anchor defines it's own positional properties, but other properties (such
+ * as the title property, say) can be inherited using the above idiom. Also note
+ * that you can override positional properties in the anchor for custom
+ * behavior.
+ *
+ * @extends pv.Mark
+ * @param {pv.Mark} target the anchor target.
+ */
+pv.Anchor = function(target) {
+  pv.Mark.call(this);
+  this.target = target;
+  this.parent = target.parent;
+};
+
+pv.Anchor.prototype = pv.extend(pv.Mark)
+    .property("name", String);
+
+/**
+ * The anchor name. The set of supported anchor names is dependent on the
+ * concrete mark type; see the mark type for details. For example, bars support
+ * left, right, top and bottom anchors.
+ *
+ * <p>While anchor names are typically constants, the anchor name is a true
+ * property, which means you can specify a function to compute the anchor name
+ * dynamically. For instance, if you wanted to alternate top and bottom anchors,
+ * saying
+ *
+ * <pre>m.anchor(function() (this.index % 2) ? "top" : "bottom").add(pv.Dot);</pre>
+ *
+ * would have the desired effect.
+ *
+ * @type string
+ * @name pv.Anchor.prototype.name
+ */
+
+/**
+ * Sets the prototype of this anchor to the specified mark. Any properties not
+ * defined on this mark may be inherited from the specified prototype mark, or
+ * its prototype, and so on. The prototype mark need not be the same type of
+ * mark as this mark. (Note that for inheritance to be useful, properties with
+ * the same name on different mark types should have equivalent meaning.)
+ *
+ * <p>This method differs slightly from the normal mark behavior in that the
+ * anchor's target is preserved.
+ *
+ * @param {pv.Mark} proto the new prototype.
+ * @returns {pv.Anchor} this anchor.
+ * @see pv.Mark#add
+ */
+pv.Anchor.prototype.extend = function(proto) {
+  this.proto = proto;
+  return this;
+};
+/**
+ * Constructs a new area mark with default properties. Areas are not typically
+ * constructed directly, but by adding to a panel or an existing mark via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents an area mark: the solid area between two series of
+ * connected line segments. Unsurprisingly, areas are used most frequently for
+ * area charts.
+ *
+ * <p>Just as a line represents a polyline, the <tt>Area</tt> mark type
+ * represents a <i>polygon</i>. However, an area is not an arbitrary polygon;
+ * vertices are paired either horizontally or vertically into parallel
+ * <i>spans</i>, and each span corresponds to an associated datum. Either the
+ * width or the height must be specified, but not both; this determines whether
+ * the area is horizontally-oriented or vertically-oriented.  Like lines, areas
+ * can be stroked and filled with arbitrary colors.
+ *
+ * <p>See also the <a href="../../api/Area.html">Area guide</a>.
+ *
+ * @extends pv.Mark
+ */
+pv.Area = function() {
+  pv.Mark.call(this);
+};
+
+pv.Area.prototype = pv.extend(pv.Mark)
+    .property("width", Number)
+    .property("height", Number)
+    .property("lineWidth", Number)
+    .property("strokeStyle", pv.color)
+    .property("fillStyle", pv.color)
+    .property("segmented", Boolean)
+    .property("interpolate", String)
+    .property("tension", Number);
+
+pv.Area.prototype.type = "area";
+
+/**
+ * The width of a given span, in pixels; used for horizontal spans. If the width
+ * is specified, the height property should be 0 (the default). Either the top
+ * or bottom property should be used to space the spans vertically, typically as
+ * a multiple of the index.
+ *
+ * @type number
+ * @name pv.Area.prototype.width
+ */
+
+/**
+ * The height of a given span, in pixels; used for vertical spans. If the height
+ * is specified, the width property should be 0 (the default). Either the left
+ * or right property should be used to space the spans horizontally, typically
+ * as a multiple of the index.
+ *
+ * @type number
+ * @name pv.Area.prototype.height
+ */
+
+/**
+ * The width of stroked lines, in pixels; used in conjunction with
+ * <tt>strokeStyle</tt> to stroke the perimeter of the area. Unlike the
+ * {@link Line} mark type, the entire perimeter is stroked, rather than just one
+ * edge. The default value of this property is 1.5, but since the default stroke
+ * style is null, area marks are not stroked by default.
+ *
+ * <p>This property is <i>fixed</i> for non-segmented areas. See
+ * {@link pv.Mark}.
+ *
+ * @type number
+ * @name pv.Area.prototype.lineWidth
+ */
+
+/**
+ * The style of stroked lines; used in conjunction with <tt>lineWidth</tt> to
+ * stroke the perimeter of the area. Unlike the {@link Line} mark type, the
+ * entire perimeter is stroked, rather than just one edge. The default value of
+ * this property is null, meaning areas are not stroked by default.
+ *
+ * <p>This property is <i>fixed</i> for non-segmented areas. See
+ * {@link pv.Mark}.
+ *
+ * @type string
+ * @name pv.Area.prototype.strokeStyle
+ * @see pv.color
+ */
+
+/**
+ * The area fill style; if non-null, the interior of the polygon forming the
+ * area is filled with the specified color. The default value of this property
+ * is a categorical color.
+ *
+ * <p>This property is <i>fixed</i> for non-segmented areas. See
+ * {@link pv.Mark}.
+ *
+ * @type string
+ * @name pv.Area.prototype.fillStyle
+ * @see pv.color
+ */
+
+/**
+ * Whether the area is segmented; whether variations in fill style, stroke
+ * style, and the other properties are treated as fixed. Rendering segmented
+ * areas is noticeably slower than non-segmented areas.
+ *
+ * <p>This property is <i>fixed</i>. See {@link pv.Mark}.
+ *
+ * @type boolean
+ * @name pv.Area.prototype.segmented
+ */
+
+/**
+ * How to interpolate between values. Linear interpolation ("linear") is the
+ * default, producing a straight line between points. For piecewise constant
+ * functions (i.e., step functions), either "step-before" or "step-after" can be
+ * specified. To draw open uniform b-splines, specify "basis". To draw cardinal
+ * splines, specify "cardinal"; see also {@link #tension}.
+ *
+ * <p>This property is <i>fixed</i>. See {@link pv.Mark}.
+ *
+ * @type string
+ * @name pv.Area.prototype.interpolate
+ */
+
+/**
+ * The tension of cardinal splines; used in conjunction with
+ * interpolate("cardinal"). A value between 0 and 1 draws cardinal splines with
+ * the given tension. In some sense, the tension can be interpreted as the
+ * "length" of the tangent; a tension of 1 will yield all zero tangents (i.e.,
+ * linear interpolation), and a tension of 0 yields a Catmull-Rom spline. The
+ * default value is 0.7.
+ *
+ * <p>This property is <i>fixed</i>. See {@link pv.Mark}.
+ *
+ * @type number
+ * @name pv.Area.prototype.tension
+ */
+
+/**
+ * Default properties for areas. By default, there is no stroke and the fill
+ * style is a categorical color.
+ *
+ * @type pv.Area
+ */
+pv.Area.prototype.defaults = new pv.Area()
+    .extend(pv.Mark.prototype.defaults)
+    .lineWidth(1.5)
+    .fillStyle(pv.Colors.category20().by(pv.parent))
+    .interpolate("linear")
+    .tension(.7);
+
+/** @private Sets width and height to zero if null. */
+pv.Area.prototype.buildImplied = function(s) {
+  if (s.height == null) s.height = 0;
+  if (s.width == null) s.width = 0;
+  pv.Mark.prototype.buildImplied.call(this, s);
+};
+
+/** @private Records which properties may be fixed. */
+pv.Area.fixed = {
+  lineWidth: 1,
+  lineJoin: 1,
+  strokeStyle: 1,
+  fillStyle: 1,
+  segmented: 1,
+  interpolate: 1,
+  tension: 1
+};
+
+/**
+ * @private Make segmented required, such that this fixed property is always
+ * evaluated, even if the first segment is not visible. Also cache which
+ * properties are normally fixed.
+ */
+pv.Area.prototype.bind = function() {
+  pv.Mark.prototype.bind.call(this);
+  var binds = this.binds,
+      required = binds.required,
+      optional = binds.optional;
+  for (var i = 0, n = optional.length; i < n; i++) {
+    var p = optional[i];
+    p.fixed = p.name in pv.Area.fixed;
+    if (p.name == "segmented") {
+      required.push(p);
+      optional.splice(i, 1);
+      i--;
+      n--;
+    }
+  }
+
+  /* Cache the original arrays so they can be restored on build. */
+  this.binds.$required = required;
+  this.binds.$optional = optional;
+};
+
+/**
+ * @private Override the default build behavior such that fixed properties are
+ * determined dynamically, based on the value of the (always) fixed segmented
+ * property. Any fixed properties are only evaluated on the first instance,
+ * although their values are propagated to subsequent instances, so that they
+ * are available for property chaining and the like.
+ */
+pv.Area.prototype.buildInstance = function(s) {
+  var binds = this.binds;
+
+  /* Handle fixed properties on secondary instances. */
+  if (this.index) {
+    var fixed = binds.fixed;
+
+    /* Determine which properties are fixed. */
+    if (!fixed) {
+      fixed = binds.fixed = [];
+      function f(p) { return !p.fixed || (fixed.push(p), false); }
+      binds.required = binds.required.filter(f);
+      if (!this.scene[0].segmented) binds.optional = binds.optional.filter(f);
+    }
+
+    /* Copy fixed property values from the first instance. */
+    for (var i = 0, n = fixed.length; i < n; i++) {
+      var p = fixed[i].name;
+      s[p] = this.scene[0][p];
+    }
+  }
+
+  /* Evaluate all properties on the first instance. */
+  else {
+    binds.required = binds.$required;
+    binds.optional = binds.$optional;
+    binds.fixed = null;
+  }
+
+  pv.Mark.prototype.buildInstance.call(this, s);
+};
+
+/**
+ * Constructs a new area anchor with default properties. Areas support five
+ * different anchors:<ul>
+ *
+ * <li>top
+ * <li>left
+ * <li>center
+ * <li>bottom
+ * <li>right
+ *
+ * </ul>In addition to positioning properties (left, right, top bottom), the
+ * anchors support text rendering properties (text-align, text-baseline). Text
+ * is rendered to appear inside the area. The area anchor also propagates the
+ * interpolate, eccentricity, and tension properties such that an anchored area
+ * or line will match positions between control points.
+ *
+ * <p>For consistency with the other mark types, the anchor positions are
+ * defined in terms of their opposite edge. For example, the top anchor defines
+ * the bottom property, such that an area added to the top anchor grows upward.
+ *
+ * @param {string} name the anchor name; either a string or a property function.
+ * @returns {pv.Anchor}
+ */
+pv.Area.prototype.anchor = function(name) {
+  return pv.Mark.prototype.anchor.call(this, name)
+    .interpolate(function() {
+       return this.scene.target[this.index].interpolate;
+      })
+    .eccentricity(function() {
+       return this.scene.target[this.index].eccentricity;
+      })
+    .tension(function() {
+        return this.scene.target[this.index].tension;
+      });
+};
+/**
+ * Constructs a new bar mark with default properties. Bars are not typically
+ * constructed directly, but by adding to a panel or an existing mark via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents a bar: an axis-aligned rectangle that can be stroked and
+ * filled. Bars are used for many chart types, including bar charts, histograms
+ * and Gantt charts. Bars can also be used as decorations, for example to draw a
+ * frame border around a panel; in fact, a panel is a special type (a subclass)
+ * of bar.
+ *
+ * <p>Bars can be positioned in several ways. Most commonly, one of the four
+ * corners is fixed using two margins, and then the width and height properties
+ * determine the extent of the bar relative to this fixed location. For example,
+ * using the bottom and left properties fixes the bottom-left corner; the width
+ * then extends to the right, while the height extends to the top. As an
+ * alternative to the four corners, a bar can be positioned exclusively using
+ * margins; this is convenient as an inset from the containing panel, for
+ * example. See {@link pv.Mark} for details on the prioritization of redundant
+ * positioning properties.
+ *
+ * <p>See also the <a href="../../api/Bar.html">Bar guide</a>.
+ *
+ * @extends pv.Mark
+ */
+pv.Bar = function() {
+  pv.Mark.call(this);
+};
+
+pv.Bar.prototype = pv.extend(pv.Mark)
+    .property("width", Number)
+    .property("height", Number)
+    .property("lineWidth", Number)
+    .property("strokeStyle", pv.color)
+    .property("fillStyle", pv.color);
+
+pv.Bar.prototype.type = "bar";
+
+/**
+ * The width of the bar, in pixels. If the left position is specified, the bar
+ * extends rightward from the left edge; if the right position is specified, the
+ * bar extends leftward from the right edge.
+ *
+ * @type number
+ * @name pv.Bar.prototype.width
+ */
+
+/**
+ * The height of the bar, in pixels. If the bottom position is specified, the
+ * bar extends upward from the bottom edge; if the top position is specified,
+ * the bar extends downward from the top edge.
+ *
+ * @type number
+ * @name pv.Bar.prototype.height
+ */
+
+/**
+ * The width of stroked lines, in pixels; used in conjunction with
+ * <tt>strokeStyle</tt> to stroke the bar's border.
+ *
+ * @type number
+ * @name pv.Bar.prototype.lineWidth
+ */
+
+/**
+ * The style of stroked lines; used in conjunction with <tt>lineWidth</tt> to
+ * stroke the bar's border. The default value of this property is null, meaning
+ * bars are not stroked by default.
+ *
+ * @type string
+ * @name pv.Bar.prototype.strokeStyle
+ * @see pv.color
+ */
+
+/**
+ * The bar fill style; if non-null, the interior of the bar is filled with the
+ * specified color. The default value of this property is a categorical color.
+ *
+ * @type string
+ * @name pv.Bar.prototype.fillStyle
+ * @see pv.color
+ */
+
+/**
+ * Default properties for bars. By default, there is no stroke and the fill
+ * style is a categorical color.
+ *
+ * @type pv.Bar
+ */
+pv.Bar.prototype.defaults = new pv.Bar()
+    .extend(pv.Mark.prototype.defaults)
+    .lineWidth(1.5)
+    .fillStyle(pv.Colors.category20().by(pv.parent));
+/**
+ * Constructs a new dot mark with default properties. Dots are not typically
+ * constructed directly, but by adding to a panel or an existing mark via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents a dot; a dot is simply a sized glyph centered at a given
+ * point that can also be stroked and filled. The <tt>size</tt> property is
+ * proportional to the area of the rendered glyph to encourage meaningful visual
+ * encodings. Dots can visually encode up to eight dimensions of data, though
+ * this may be unwise due to integrality. See {@link pv.Mark} for details on the
+ * prioritization of redundant positioning properties.
+ *
+ * <p>See also the <a href="../../api/Dot.html">Dot guide</a>.
+ *
+ * @extends pv.Mark
+ */
+pv.Dot = function() {
+  pv.Mark.call(this);
+};
+
+pv.Dot.prototype = pv.extend(pv.Mark)
+    .property("size", Number)
+    .property("radius", Number)
+    .property("shape", String)
+    .property("angle", Number)
+    .property("lineWidth", Number)
+    .property("strokeStyle", pv.color)
+    .property("fillStyle", pv.color);
+
+pv.Dot.prototype.type = "dot";
+
+/**
+ * The size of the dot, in square pixels. Square pixels are used such that the
+ * area of the dot is linearly proportional to the value of the size property,
+ * facilitating representative encodings.
+ *
+ * @see #radius
+ * @type number
+ * @name pv.Dot.prototype.size
+ */
+
+/**
+ * The radius of the dot, in pixels. This is an alternative to using
+ * {@link #size}.
+ *
+ * @see #size
+ * @type number
+ * @name pv.Dot.prototype.radius
+ */
+
+/**
+ * The shape name. Several shapes are supported:<ul>
+ *
+ * <li>cross
+ * <li>triangle
+ * <li>diamond
+ * <li>square
+ * <li>circle
+ * <li>tick
+ * <li>bar
+ *
+ * </ul>These shapes can be further changed using the {@link #angle} property;
+ * for instance, a cross can be turned into a plus by rotating. Similarly, the
+ * tick, which is vertical by default, can be rotated horizontally. Note that
+ * some shapes (cross and tick) do not have interior areas, and thus do not
+ * support fill style meaningfully.
+ *
+ * <p>Note: it may be more natural to use the {@link pv.Rule} mark for
+ * horizontal and vertical ticks. The tick shape is only necessary if angled
+ * ticks are needed.
+ *
+ * @type string
+ * @name pv.Dot.prototype.shape
+ */
+
+/**
+ * The rotation angle, in radians. Used to rotate shapes, such as to turn a
+ * cross into a plus.
+ *
+ * @type number
+ * @name pv.Dot.prototype.angle
+ */
+
+/**
+ * The width of stroked lines, in pixels; used in conjunction with
+ * <tt>strokeStyle</tt> to stroke the dot's shape.
+ *
+ * @type number
+ * @name pv.Dot.prototype.lineWidth
+ */
+
+/**
+ * The style of stroked lines; used in conjunction with <tt>lineWidth</tt> to
+ * stroke the dot's shape. The default value of this property is a categorical
+ * color.
+ *
+ * @type string
+ * @name pv.Dot.prototype.strokeStyle
+ * @see pv.color
+ */
+
+/**
+ * The fill style; if non-null, the interior of the dot is filled with the
+ * specified color. The default value of this property is null, meaning dots are
+ * not filled by default.
+ *
+ * @type string
+ * @name pv.Dot.prototype.fillStyle
+ * @see pv.color
+ */
+
+/**
+ * Default properties for dots. By default, there is no fill and the stroke
+ * style is a categorical color. The default shape is "circle" with size 20.
+ *
+ * @type pv.Dot
+ */
+pv.Dot.prototype.defaults = new pv.Dot()
+    .extend(pv.Mark.prototype.defaults)
+    .size(20)
+    .shape("circle")
+    .lineWidth(1.5)
+    .strokeStyle(pv.Colors.category10().by(pv.parent));
+
+/**
+ * Constructs a new dot anchor with default properties. Dots support five
+ * different anchors:<ul>
+ *
+ * <li>top
+ * <li>left
+ * <li>center
+ * <li>bottom
+ * <li>right
+ *
+ * </ul>In addition to positioning properties (left, right, top bottom), the
+ * anchors support text rendering properties (text-align, text-baseline). Text is
+ * rendered to appear outside the dot. Note that this behavior is different from
+ * other mark anchors, which default to rendering text <i>inside</i> the mark.
+ *
+ * <p>For consistency with the other mark types, the anchor positions are
+ * defined in terms of their opposite edge. For example, the top anchor defines
+ * the bottom property, such that a bar added to the top anchor grows upward.
+ *
+ * @param {string} name the anchor name; either a string or a property function.
+ * @returns {pv.Anchor}
+ */
+pv.Dot.prototype.anchor = function(name) {
+  return pv.Mark.prototype.anchor.call(this, name)
+    .left(function() {
+        var s = this.scene.target[this.index];
+        switch (this.name()) {
+          case "bottom":
+          case "top":
+          case "center": return s.left;
+          case "left": return null;
+        }
+        return s.left + s.radius;
+      })
+    .right(function() {
+        var s = this.scene.target[this.index];
+        return this.name() == "left" ? s.right + s.radius : null;
+      })
+    .top(function() {
+        var s = this.scene.target[this.index];
+        switch (this.name()) {
+          case "left":
+          case "right":
+          case "center": return s.top;
+          case "top": return null;
+        }
+        return s.top + s.radius;
+      })
+    .bottom(function() {
+        var s = this.scene.target[this.index];
+        return this.name() == "top" ? s.bottom + s.radius : null;
+      })
+    .textAlign(function() {
+        switch (this.name()) {
+          case "left": return "right";
+          case "bottom":
+          case "top":
+          case "center": return "center";
+        }
+        return "left";
+      })
+    .textBaseline(function() {
+        switch (this.name()) {
+          case "right":
+          case "left":
+          case "center": return "middle";
+          case "bottom": return "top";
+        }
+        return "bottom";
+      });
+};
+
+/** @private Sets radius based on size or vice versa. */
+pv.Dot.prototype.buildImplied = function(s) {
+  if (s.radius == null) s.radius = Math.sqrt(s.size);
+  else if (s.size == null) s.size = s.radius * s.radius;
+  pv.Mark.prototype.buildImplied.call(this, s);
+};
+/**
+ * Constructs a new label mark with default properties. Labels are not typically
+ * constructed directly, but by adding to a panel or an existing mark via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents a text label, allowing textual annotation of other marks or
+ * arbitrary text within the visualization. The character data must be plain
+ * text (unicode), though the text can be styled using the {@link #font}
+ * property. If rich text is needed, external HTML elements can be overlaid on
+ * the canvas by hand.
+ *
+ * <p>Labels are positioned using the box model, similarly to {@link Dot}. Thus,
+ * a label has no width or height, but merely a text anchor location. The text
+ * is positioned relative to this anchor location based on the
+ * {@link #textAlign}, {@link #textBaseline} and {@link #textMargin} properties.
+ * Furthermore, the text may be rotated using {@link #textAngle}.
+ *
+ * <p>Labels ignore events, so as to not interfere with event handlers on
+ * underlying marks, such as bars. In the future, we may support event handlers
+ * on labels.
+ *
+ * <p>See also the <a href="../../api/Label.html">Label guide</a>.
+ *
+ * @extends pv.Mark
+ */
+pv.Label = function() {
+  pv.Mark.call(this);
+};
+
+pv.Label.prototype = pv.extend(pv.Mark)
+    .property("text", String)
+    .property("font", String)
+    .property("textAngle", Number)
+    .property("textStyle", pv.color)
+    .property("textAlign", String)
+    .property("textBaseline", String)
+    .property("textMargin", Number)
+    .property("textDecoration", String)
+    .property("textShadow", String);
+
+pv.Label.prototype.type = "label";
+
+/**
+ * The character data to render; a string. The default value of the text
+ * property is the identity function, meaning the label's associated datum will
+ * be rendered using its <tt>toString</tt>.
+ *
+ * @type string
+ * @name pv.Label.prototype.text
+ */
+
+/**
+ * The font format, per the CSS Level 2 specification. The default font is "10px
+ * sans-serif", for consistency with the HTML 5 canvas element specification.
+ * Note that since text is not wrapped, any line-height property will be
+ * ignored. The other font-style, font-variant, font-weight, font-size and
+ * font-family properties are supported.
+ *
+ * @see <a href="http://www.w3.org/TR/CSS2/fonts.html#font-shorthand">CSS2 fonts</a>
+ * @type string
+ * @name pv.Label.prototype.font
+ */
+
+/**
+ * The rotation angle, in radians. Text is rotated clockwise relative to the
+ * anchor location. For example, with the default left alignment, an angle of
+ * Math.PI / 2 causes text to proceed downwards. The default angle is zero.
+ *
+ * @type number
+ * @name pv.Label.prototype.textAngle
+ */
+
+/**
+ * The text color. The name "textStyle" is used for consistency with "fillStyle"
+ * and "strokeStyle", although it might be better to rename this property (and
+ * perhaps use the same name as "strokeStyle"). The default color is black.
+ *
+ * @type string
+ * @name pv.Label.prototype.textStyle
+ * @see pv.color
+ */
+
+/**
+ * The horizontal text alignment. One of:<ul>
+ *
+ * <li>left
+ * <li>center
+ * <li>right
+ *
+ * </ul>The default horizontal alignment is left.
+ *
+ * @type string
+ * @name pv.Label.prototype.textAlign
+ */
+
+/**
+ * The vertical text alignment. One of:<ul>
+ *
+ * <li>top
+ * <li>middle
+ * <li>bottom
+ *
+ * </ul>The default vertical alignment is bottom.
+ *
+ * @type string
+ * @name pv.Label.prototype.textBaseline
+ */
+
+/**
+ * The text margin; may be specified in pixels, or in font-dependent units (such
+ * as ".1ex"). The margin can be used to pad text away from its anchor location,
+ * in a direction dependent on the horizontal and vertical alignment
+ * properties. For example, if the text is left- and middle-aligned, the margin
+ * shifts the text to the right. The default margin is 3 pixels.
+ *
+ * @type number
+ * @name pv.Label.prototype.textMargin
+ */
+
+/**
+ * A list of shadow effects to be applied to text, per the CSS Text Level 3
+ * text-shadow property. An example specification is "0.1em 0.1em 0.1em
+ * rgba(0,0,0,.5)"; the first length is the horizontal offset, the second the
+ * vertical offset, and the third the blur radius.
+ *
+ * @see <a href="http://www.w3.org/TR/css3-text/#text-shadow">CSS3 text</a>
+ * @type string
+ * @name pv.Label.prototype.textShadow
+ */
+
+/**
+ * A list of decoration to be applied to text, per the CSS Text Level 3
+ * text-decoration property. An example specification is "underline".
+ *
+ * @see <a href="http://www.w3.org/TR/css3-text/#text-decoration">CSS3 text</a>
+ * @type string
+ * @name pv.Label.prototype.textDecoration
+ */
+
+/**
+ * Default properties for labels. See the individual properties for the default
+ * values.
+ *
+ * @type pv.Label
+ */
+pv.Label.prototype.defaults = new pv.Label()
+    .extend(pv.Mark.prototype.defaults)
+    .events("none")
+    .text(pv.identity)
+    .font("10px sans-serif")
+    .textAngle(0)
+    .textStyle("black")
+    .textAlign("left")
+    .textBaseline("bottom")
+    .textMargin(3);
+/**
+ * Constructs a new line mark with default properties. Lines are not typically
+ * constructed directly, but by adding to a panel or an existing mark via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents a series of connected line segments, or <i>polyline</i>,
+ * that can be stroked with a configurable color and thickness. Each
+ * articulation point in the line corresponds to a datum; for <i>n</i> points,
+ * <i>n</i>-1 connected line segments are drawn. The point is positioned using
+ * the box model. Arbitrary paths are also possible, allowing radar plots and
+ * other custom visualizations.
+ *
+ * <p>Like areas, lines can be stroked and filled with arbitrary colors. In most
+ * cases, lines are only stroked, but the fill style can be used to construct
+ * arbitrary polygons.
+ *
+ * <p>See also the <a href="../../api/Line.html">Line guide</a>.
+ *
+ * @extends pv.Mark
+ */
+pv.Line = function() {
+  pv.Mark.call(this);
+};
+
+pv.Line.prototype = pv.extend(pv.Mark)
+    .property("lineWidth", Number)
+    .property("lineJoin", String)
+    .property("strokeStyle", pv.color)
+    .property("fillStyle", pv.color)
+    .property("segmented", Boolean)
+    .property("interpolate", String)
+    .property("eccentricity", Number)
+    .property("tension", Number);
+
+pv.Line.prototype.type = "line";
+
+/**
+ * The width of stroked lines, in pixels; used in conjunction with
+ * <tt>strokeStyle</tt> to stroke the line.
+ *
+ * @type number
+ * @name pv.Line.prototype.lineWidth
+ */
+
+/**
+ * The style of stroked lines; used in conjunction with <tt>lineWidth</tt> to
+ * stroke the line. The default value of this property is a categorical color.
+ *
+ * @type string
+ * @name pv.Line.prototype.strokeStyle
+ * @see pv.color
+ */
+
+/**
+ * The type of corners where two lines meet. Accepted values are "bevel",
+ * "round" and "miter". The default value is "miter".
+ *
+ * <p>For segmented lines, only "miter" joins and "linear" interpolation are
+ * currently supported. Any other value, including null, will disable joins,
+ * producing disjoint line segments. Note that the miter joins must be computed
+ * manually (at least in the current SVG renderer); since this calculation may
+ * be expensive and unnecessary for small lines, specifying null can improve
+ * performance significantly.
+ *
+ * <p>This property is <i>fixed</i>. See {@link pv.Mark}.
+ *
+ * @type string
+ * @name pv.Line.prototype.lineJoin
+ */
+
+/**
+ * The line fill style; if non-null, the interior of the line is closed and
+ * filled with the specified color. The default value of this property is a
+ * null, meaning that lines are not filled by default.
+ *
+ * <p>This property is <i>fixed</i>. See {@link pv.Mark}.
+ *
+ * @type string
+ * @name pv.Line.prototype.fillStyle
+ * @see pv.color
+ */
+
+/**
+ * Whether the line is segmented; whether variations in stroke style, line width
+ * and the other properties are treated as fixed. Rendering segmented lines is
+ * noticeably slower than non-segmented lines.
+ *
+ * <p>This property is <i>fixed</i>. See {@link pv.Mark}.
+ *
+ * @type boolean
+ * @name pv.Line.prototype.segmented
+ */
+
+/**
+ * How to interpolate between values. Linear interpolation ("linear") is the
+ * default, producing a straight line between points. For piecewise constant
+ * functions (i.e., step functions), either "step-before" or "step-after" can be
+ * specified. To draw a clockwise circular arc between points, specify "polar";
+ * to draw a counterclockwise circular arc between points, specify
+ * "polar-reverse". To draw open uniform b-splines, specify "basis". To draw
+ * cardinal splines, specify "cardinal"; see also {@link #tension}.
+ *
+ * <p>This property is <i>fixed</i>. See {@link pv.Mark}.
+ *
+ * @type string
+ * @name pv.Line.prototype.interpolate
+ */
+
+/**
+ * The eccentricity of polar line segments; used in conjunction with
+ * interpolate("polar"). The default value of 0 means that line segments are
+ * drawn as circular arcs. A value of 1 draws a straight line. A value between 0
+ * and 1 draws an elliptical arc with the given eccentricity.
+ *
+ * @type number
+ * @name pv.Line.prototype.eccentricity
+ */
+
+/**
+ * The tension of cardinal splines; used in conjunction with
+ * interpolate("cardinal"). A value between 0 and 1 draws cardinal splines with
+ * the given tension. In some sense, the tension can be interpreted as the
+ * "length" of the tangent; a tension of 1 will yield all zero tangents (i.e.,
+ * linear interpolation), and a tension of 0 yields a Catmull-Rom spline. The
+ * default value is 0.7.
+ *
+ * <p>This property is <i>fixed</i>. See {@link pv.Mark}.
+ *
+ * @type number
+ * @name pv.Line.prototype.tension
+ */
+
+/**
+ * Default properties for lines. By default, there is no fill and the stroke
+ * style is a categorical color. The default interpolation is linear.
+ *
+ * @type pv.Line
+ */
+pv.Line.prototype.defaults = new pv.Line()
+    .extend(pv.Mark.prototype.defaults)
+    .lineJoin("miter")
+    .lineWidth(1.5)
+    .strokeStyle(pv.Colors.category10().by(pv.parent))
+    .interpolate("linear")
+    .eccentricity(0)
+    .tension(.7);
+
+/** @private Reuse Area's implementation for segmented bind & build. */
+pv.Line.prototype.bind = pv.Area.prototype.bind;
+pv.Line.prototype.buildInstance = pv.Area.prototype.buildInstance;
+
+/**
+ * Constructs a new line anchor with default properties. Lines support five
+ * different anchors:<ul>
+ *
+ * <li>top
+ * <li>left
+ * <li>center
+ * <li>bottom
+ * <li>right
+ *
+ * </ul>In addition to positioning properties (left, right, top bottom), the
+ * anchors support text rendering properties (text-align, text-baseline). Text is
+ * rendered to appear outside the line. Note that this behavior is different
+ * from other mark anchors, which default to rendering text <i>inside</i> the
+ * mark.
+ *
+ * <p>For consistency with the other mark types, the anchor positions are
+ * defined in terms of their opposite edge. For example, the top anchor defines
+ * the bottom property, such that a bar added to the top anchor grows upward.
+ *
+ * @param {string} name the anchor name; either a string or a property function.
+ * @returns {pv.Anchor}
+ */
+pv.Line.prototype.anchor = function(name) {
+  return pv.Area.prototype.anchor.call(this, name)
+    .textAlign(function(d) {
+        switch (this.name()) {
+          case "left": return "right";
+          case "bottom":
+          case "top":
+          case "center": return "center";
+          case "right": return "left";
+        }
+      })
+    .textBaseline(function(d) {
+        switch (this.name()) {
+          case "right":
+          case "left":
+          case "center": return "middle";
+          case "top": return "bottom";
+          case "bottom": return "top";
+        }
+      });
+};
+/**
+ * Constructs a new rule with default properties. Rules are not typically
+ * constructed directly, but by adding to a panel or an existing mark via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents a horizontal or vertical rule. Rules are frequently used
+ * for axes and grid lines. For example, specifying only the bottom property
+ * draws horizontal rules, while specifying only the left draws vertical
+ * rules. Rules can also be used as thin bars. The visual style is controlled in
+ * the same manner as lines.
+ *
+ * <p>Rules are positioned exclusively the standard box model properties. The
+ * following combinations of properties are supported:
+ *
+ * <table>
+ * <thead><th style="width:12em;">Properties</th><th>Orientation</th></thead>
+ * <tbody>
+ * <tr><td>left</td><td>vertical</td></tr>
+ * <tr><td>right</td><td>vertical</td></tr>
+ * <tr><td>left, bottom, top</td><td>vertical</td></tr>
+ * <tr><td>right, bottom, top</td><td>vertical</td></tr>
+ * <tr><td>top</td><td>horizontal</td></tr>
+ * <tr><td>bottom</td><td>horizontal</td></tr>
+ * <tr><td>top, left, right</td><td>horizontal</td></tr>
+ * <tr><td>bottom, left, right</td><td>horizontal</td></tr>
+ * <tr><td>left, top, height</td><td>vertical</td></tr>
+ * <tr><td>left, bottom, height</td><td>vertical</td></tr>
+ * <tr><td>right, top, height</td><td>vertical</td></tr>
+ * <tr><td>right, bottom, height</td><td>vertical</td></tr>
+ * <tr><td>left, top, width</td><td>horizontal</td></tr>
+ * <tr><td>left, bottom, width</td><td>horizontal</td></tr>
+ * <tr><td>right, top, width</td><td>horizontal</td></tr>
+ * <tr><td>right, bottom, width</td><td>horizontal</td></tr>
+ * </tbody>
+ * </table>
+ *
+ * <p>Small rules can be used as tick marks; alternatively, a {@link Dot} with
+ * the "tick" shape can be used.
+ *
+ * <p>See also the <a href="../../api/Rule.html">Rule guide</a>.
+ *
+ * @see pv.Line
+ * @extends pv.Mark
+ */
+pv.Rule = function() {
+  pv.Mark.call(this);
+};
+
+pv.Rule.prototype = pv.extend(pv.Mark)
+    .property("width", Number)
+    .property("height", Number)
+    .property("lineWidth", Number)
+    .property("strokeStyle", pv.color);
+
+pv.Rule.prototype.type = "rule";
+
+/**
+ * The width of the rule, in pixels. If the left position is specified, the rule
+ * extends rightward from the left edge; if the right position is specified, the
+ * rule extends leftward from the right edge.
+ *
+ * @type number
+ * @name pv.Rule.prototype.width
+ */
+
+/**
+ * The height of the rule, in pixels. If the bottom position is specified, the
+ * rule extends upward from the bottom edge; if the top position is specified,
+ * the rule extends downward from the top edge.
+ *
+ * @type number
+ * @name pv.Rule.prototype.height
+ */
+
+/**
+ * The width of stroked lines, in pixels; used in conjunction with
+ * <tt>strokeStyle</tt> to stroke the rule. The default value is 1 pixel.
+ *
+ * @type number
+ * @name pv.Rule.prototype.lineWidth
+ */
+
+/**
+ * The style of stroked lines; used in conjunction with <tt>lineWidth</tt> to
+ * stroke the rule. The default value of this property is black.
+ *
+ * @type string
+ * @name pv.Rule.prototype.strokeStyle
+ * @see pv.color
+ */
+
+/**
+ * Default properties for rules. By default, a single-pixel black line is
+ * stroked.
+ *
+ * @type pv.Rule
+ */
+pv.Rule.prototype.defaults = new pv.Rule()
+    .extend(pv.Mark.prototype.defaults)
+    .lineWidth(1)
+    .strokeStyle("black")
+    .antialias(false);
+
+/**
+ * Constructs a new rule anchor with default properties. Rules support five
+ * different anchors:<ul>
+ *
+ * <li>top
+ * <li>left
+ * <li>center
+ * <li>bottom
+ * <li>right
+ *
+ * </ul>In addition to positioning properties (left, right, top bottom), the
+ * anchors support text rendering properties (text-align, text-baseline). Text is
+ * rendered to appear outside the rule. Note that this behavior is different
+ * from other mark anchors, which default to rendering text <i>inside</i> the
+ * mark.
+ *
+ * <p>For consistency with the other mark types, the anchor positions are
+ * defined in terms of their opposite edge. For example, the top anchor defines
+ * the bottom property, such that a bar added to the top anchor grows upward.
+ *
+ * @param {string} name the anchor name; either a string or a property function.
+ * @returns {pv.Anchor}
+ */
+pv.Rule.prototype.anchor = pv.Line.prototype.anchor;
+
+/** @private Sets width or height based on orientation. */
+pv.Rule.prototype.buildImplied = function(s) {
+  var l = s.left, r = s.right, t = s.top, b = s.bottom;
+
+  /* Determine horizontal or vertical orientation. */
+  if ((s.width != null)
+      || ((l == null) && (r == null))
+      || ((r != null) && (l != null))) {
+    s.height = 0;
+  } else {
+    s.width = 0;
+  }
+
+  pv.Mark.prototype.buildImplied.call(this, s);
+};
+/**
+ * Constructs a new, empty panel with default properties. Panels, with the
+ * exception of the root panel, are not typically constructed directly; instead,
+ * they are added to an existing panel or mark via {@link pv.Mark#add}.
+ *
+ * @class Represents a container mark. Panels allow repeated or nested
+ * structures, commonly used in small multiple displays where a small
+ * visualization is tiled to facilitate comparison across one or more
+ * dimensions. Other types of visualizations may benefit from repeated and
+ * possibly overlapping structure as well, such as stacked area charts. Panels
+ * can also offset the position of marks to provide padding from surrounding
+ * content.
+ *
+ * <p>All Protovis displays have at least one panel; this is the root panel to
+ * which marks are rendered. The box model properties (four margins, width and
+ * height) are used to offset the positions of contained marks. The data
+ * property determines the panel count: a panel is generated once per associated
+ * datum. When nested panels are used, property functions can declare additional
+ * arguments to access the data associated with enclosing panels.
+ *
+ * <p>Panels can be rendered inline, facilitating the creation of sparklines.
+ * This allows designers to reuse browser layout features, such as text flow and
+ * tables; designers can also overlay HTML elements such as rich text and
+ * images.
+ *
+ * <p>All panels have a <tt>children</tt> array (possibly empty) containing the
+ * child marks in the order they were added. Panels also have a <tt>root</tt>
+ * field which points to the root (outermost) panel; the root panel's root field
+ * points to itself.
+ *
+ * <p>See also the <a href="../../api/">Protovis guide</a>.
+ *
+ * @extends pv.Bar
+ */
+pv.Panel = function() {
+  pv.Bar.call(this);
+
+  /**
+   * The child marks; zero or more {@link pv.Mark}s in the order they were
+   * added.
+   *
+   * @see #add
+   * @type pv.Mark[]
+   */
+  this.children = [];
+  this.root = this;
+
+  /**
+   * The internal $dom field is set by the Protovis loader; see lang/init.js. It
+   * refers to the script element that contains the Protovis specification, so
+   * that the panel knows where in the DOM to insert the generated SVG element.
+   *
+   * @private
+   */
+  this.$dom = pv.$ && pv.$.s;
+};
+
+pv.Panel.prototype = pv.extend(pv.Bar)
+    .property("transform")
+    .property("overflow", String)
+    .property("canvas", function(c) {
+        return (typeof c == "string")
+            ? document.getElementById(c)
+            : c; // assume that c is the passed-in element
+      });
+
+pv.Panel.prototype.type = "panel";
+
+/**
+ * The canvas element; either the string ID of the canvas element in the current
+ * document, or a reference to the canvas element itself. If null, a canvas
+ * element will be created and inserted into the document at the location of the
+ * script element containing the current Protovis specification. This property
+ * only applies to root panels and is ignored on nested panels.
+ *
+ * <p>Note: the "canvas" element here refers to a <tt>div</tt> (or other suitable
+ * HTML container element), <i>not</i> a <tt>canvas</tt> element. The name of
+ * this property is a historical anachronism from the first implementation that
+ * used HTML 5 canvas, rather than SVG.
+ *
+ * @type string
+ * @name pv.Panel.prototype.canvas
+ */
+
+/**
+ * Specifies whether child marks are clipped when they overflow this panel.
+ * This affects the clipping of all this panel's descendant marks.
+ *
+ * @type string
+ * @name pv.Panel.prototype.overflow
+ * @see <a href="http://www.w3.org/TR/CSS2/visufx.html#overflow">CSS2</a>
+ */
+
+/**
+ * The transform to be applied to child marks. The default transform is
+ * identity, which has no effect. Note that the panel's own fill and stroke are
+ * not affected by the transform, and panel's transform only affects the
+ * <tt>scale</tt> of child marks, not the panel itself.
+ *
+ * @type pv.Transform
+ * @name pv.Panel.prototype.transform
+ * @see pv.Mark#scale
+ */
+
+/**
+ * Default properties for panels. By default, the margins are zero, the fill
+ * style is transparent.
+ *
+ * @type pv.Panel
+ */
+pv.Panel.prototype.defaults = new pv.Panel()
+    .extend(pv.Bar.prototype.defaults)
+    .fillStyle(null) // override Bar default
+    .overflow("visible");
+
+/**
+ * Returns an anchor with the specified name. This method is overridden such
+ * that adding to a panel's anchor adds to the panel, rather than to the panel's
+ * parent.
+ *
+ * @param {string} name the anchor name; either a string or a property function.
+ * @returns {pv.Anchor} the new anchor.
+ */
+pv.Panel.prototype.anchor = function(name) {
+  var anchor = pv.Bar.prototype.anchor.call(this, name);
+  anchor.parent = this;
+  return anchor;
+};
+
+/**
+ * Adds a new mark of the specified type to this panel. Unlike the normal
+ * {@link Mark#add} behavior, adding a mark to a panel does not cause the mark
+ * to inherit from the panel. Since the contained marks are offset by the panel
+ * margins already, inheriting properties is generally undesirable; of course,
+ * it is always possible to change this behavior by calling {@link Mark#extend}
+ * explicitly.
+ *
+ * @param {function} type the type of the new mark to add.
+ * @returns {pv.Mark} the new mark.
+ */
+pv.Panel.prototype.add = function(type) {
+  var child = new type();
+  child.parent = this;
+  child.root = this.root;
+  child.childIndex = this.children.length;
+  this.children.push(child);
+  return child;
+};
+
+/** @private Bind this panel, then any child marks recursively. */
+pv.Panel.prototype.bind = function() {
+  pv.Mark.prototype.bind.call(this);
+  for (var i = 0; i < this.children.length; i++) {
+    this.children[i].bind();
+  }
+};
+
+/**
+ * @private Evaluates all of the properties for this panel for the specified
+ * instance <tt>s</tt> in the scene graph, including recursively building the
+ * scene graph for child marks.
+ *
+ * @param s a node in the scene graph; the instance of the panel to build.
+ * @see Mark#scene
+ */
+pv.Panel.prototype.buildInstance = function(s) {
+  pv.Bar.prototype.buildInstance.call(this, s);
+  if (!s.visible) return;
+  if (!s.children) s.children = [];
+
+  /*
+   * Multiply the current scale factor by this panel's transform. Also clear the
+   * default index as we recurse into child marks; it will be reset to the
+   * current index when the next panel instance is built.
+   */
+  var scale = this.scale * s.transform.k, child, n = this.children.length;
+  pv.Mark.prototype.index = -1;
+
+  /*
+   * Build each child, passing in the parent (this panel) scene graph node. The
+   * child mark's scene is initialized from the corresponding entry in the
+   * existing scene graph, such that properties from the previous build can be
+   * reused; this is largely to facilitate the recycling of SVG elements.
+   */
+  for (var i = 0; i < n; i++) {
+    child = this.children[i];
+    child.scene = s.children[i]; // possibly undefined
+    child.scale = scale;
+    child.build();
+  }
+
+  /*
+   * Once the child marks have been built, the new scene graph nodes are removed
+   * from the child marks and placed into the scene graph. The nodes cannot
+   * remain on the child nodes because this panel (or a parent panel) may be
+   * instantiated multiple times!
+   */
+  for (var i = 0; i < n; i++) {
+    child = this.children[i];
+    s.children[i] = child.scene;
+    delete child.scene;
+    delete child.scale;
+  }
+
+  /* Delete any expired child scenes. */
+  s.children.length = n;
+};
+
+/**
+ * @private Computes the implied properties for this panel for the specified
+ * instance <tt>s</tt> in the scene graph. Panels have two implied
+ * properties:<ul>
+ *
+ * <li>The <tt>canvas</tt> property references the DOM element, typically a DIV,
+ * that contains the SVG element that is used to display the visualization. This
+ * property may be specified as a string, referring to the unique ID of the
+ * element in the DOM. The string is converted to a reference to the DOM
+ * element. The width and height of the SVG element is inferred from this DOM
+ * element. If no canvas property is specified, a new SVG element is created and
+ * inserted into the document, using the panel dimensions; see
+ * {@link #createCanvas}.
+ *
+ * <li>The <tt>children</tt> array, while not a property per se, contains the
+ * scene graph for each child mark. This array is initialized to be empty, and
+ * is populated above in {@link #buildInstance}.
+ *
+ * </ul>The current implementation creates the SVG element, if necessary, during
+ * the build phase; in the future, it may be preferrable to move this to the
+ * update phase, although then the canvas property would be undefined. In
+ * addition, DOM inspection is necessary to define the implied width and height
+ * properties that may be inferred from the DOM.
+ *
+ * @param s a node in the scene graph; the instance of the panel to build.
+ */
+pv.Panel.prototype.buildImplied = function(s) {
+  if (!this.parent) {
+    var c = s.canvas;
+    if (c) {
+      /* Clear the container if it's not associated with this panel. */
+      if (c.$panel != this) {
+        c.$panel = this;
+        while (c.lastChild) c.removeChild(c.lastChild);
+      }
+
+      /* If width and height weren't specified, inspect the container. */
+      var w, h;
+      if (s.width == null) {
+        w = parseFloat(pv.css(c, "width"));
+        s.width = w - s.left - s.right;
+      }
+      if (s.height == null) {
+        h = parseFloat(pv.css(c, "height"));
+        s.height = h - s.top - s.bottom;
+      }
+    } else {
+      var cache = this.$canvas || (this.$canvas = []);
+      if (!(c = cache[this.index])) {
+        c = cache[this.index] = document.createElement("span");
+        if (this.$dom) { // script element for text/javascript+protovis
+          this.$dom.parentNode.insertBefore(c, this.$dom);
+        } else { // find the last element in the body
+          var n = document.body;
+          while (n.lastChild && n.lastChild.tagName) n = n.lastChild;
+          if (n != document.body) n = n.parentNode;
+          n.appendChild(c);
+        }
+      }
+    }
+    s.canvas = c;
+  }
+  if (!s.transform) s.transform = pv.Transform.identity;
+  pv.Mark.prototype.buildImplied.call(this, s);
+};
+/**
+ * Constructs a new image with default properties. Images are not typically
+ * constructed directly, but by adding to a panel or an existing mark via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents an image, either a static resource or a dynamically-
+ * generated pixel buffer. Images share the same layout and style properties as
+ * bars. The external image resource is specified via the {@link #url}
+ * property. The optional fill, if specified, appears beneath the image, while
+ * the optional stroke appears above the image.
+ *
+ * <p>Dynamic images such as heatmaps are supported using the {@link #image}
+ * psuedo-property. This function is passed the <i>x</i> and <i>y</i> index, in
+ * addition to the current data stack. The return value is a {@link pv.Color},
+ * or null for transparent. A string can also be returned, which will be parsed
+ * into a color; however, it is typically much faster to return an object with
+ * <tt>r</tt>, <tt>g</tt>, <tt>b</tt> and <tt>a</tt> attributes, to avoid the
+ * cost of parsing and object instantiation.
+ *
+ * <p>See {@link pv.Bar} for details on positioning properties.
+ *
+ * @extends pv.Bar
+ */
+pv.Image = function() {
+  pv.Bar.call(this);
+};
+
+pv.Image.prototype = pv.extend(pv.Bar)
+    .property("url", String)
+    .property("imageWidth", Number)
+    .property("imageHeight", Number);
+
+pv.Image.prototype.type = "image";
+
+/**
+ * The URL of the image to display. The set of supported image types is
+ * browser-dependent; PNG and JPEG are recommended.
+ *
+ * @type string
+ * @name pv.Image.prototype.url
+ */
+
+/**
+ * The width of the image in pixels. For static images, this property is
+ * computed implicitly from the loaded image resources. For dynamic images, this
+ * property can be used to specify the width of the pixel buffer; otherwise, the
+ * value is derived from the <tt>width</tt> property.
+ *
+ * @type number
+ * @name pv.Image.prototype.imageWidth
+ */
+
+/**
+ * The height of the image in pixels. For static images, this property is
+ * computed implicitly from the loaded image resources. For dynamic images, this
+ * property can be used to specify the height of the pixel buffer; otherwise, the
+ * value is derived from the <tt>height</tt> property.
+ *
+ * @type number
+ * @name pv.Image.prototype.imageHeight
+ */
+
+/**
+ * Default properties for images. By default, there is no stroke or fill style.
+ *
+ * @type pv.Image
+ */
+pv.Image.prototype.defaults = new pv.Image()
+    .extend(pv.Bar.prototype.defaults)
+    .fillStyle(null);
+
+/**
+ * Specifies the dynamic image function. By default, no image function is
+ * specified and the <tt>url</tt> property is used to load a static image
+ * resource. If an image function is specified, it will be invoked for each
+ * pixel in the image, based on the related <tt>imageWidth</tt> and
+ * <tt>imageHeight</tt> properties.
+ *
+ * <p>For example, given a two-dimensional array <tt>heatmap</tt>, containing
+ * numbers in the range [0, 1] in row-major order, a simple monochrome heatmap
+ * image can be specified as:
+ *
+ * <pre>vis.add(pv.Image)
+ *     .imageWidth(heatmap[0].length)
+ *     .imageHeight(heatmap.length)
+ *     .image(pv.ramp("white", "black").by(function(x, y) heatmap[y][x]));</pre>
+ *
+ * For fastest performance, use an ordinal scale which caches the fixed color
+ * palette, or return an object literal with <tt>r</tt>, <tt>g</tt>, <tt>b</tt>
+ * and <tt>a</tt> attributes. A {@link pv.Color} or string can also be returned,
+ * though this typically results in slower performance.
+ *
+ * @param {function} f the new sizing function.
+ * @returns {pv.Layout.Pack} this.
+ */
+pv.Image.prototype.image = function(f) {
+  /** @private */
+  this.$image = function() {
+      var c = f.apply(this, arguments);
+      return c == null ? pv.Color.transparent
+          : typeof c == "string" ? pv.color(c)
+          : c;
+    };
+  return this;
+};
+
+/** @private Scan the proto chain for an image function. */
+pv.Image.prototype.bind = function() {
+  pv.Bar.prototype.bind.call(this);
+  var binds = this.binds, mark = this;
+  do {
+    binds.image = mark.$image;
+  } while (!binds.image && (mark = mark.proto));
+};
+
+/** @private */
+pv.Image.prototype.buildImplied = function(s) {
+  pv.Bar.prototype.buildImplied.call(this, s);
+  if (!s.visible) return;
+
+  /* Compute the implied image dimensions. */
+  if (s.imageWidth == null) s.imageWidth = s.width;
+  if (s.imageHeight == null) s.imageHeight = s.height;
+
+  /* Compute the pixel values. */
+  if ((s.url == null) && this.binds.image) {
+
+    /* Cache the canvas element to reuse across renders. */
+    var canvas = this.$canvas || (this.$canvas = document.createElement("canvas")),
+        context = canvas.getContext("2d"),
+        w = s.imageWidth,
+        h = s.imageHeight,
+        stack = pv.Mark.stack,
+        data;
+
+    /* Evaluate the image function, storing into a CanvasPixelArray. */
+    canvas.width = w;
+    canvas.height = h;
+    data = (s.image = context.createImageData(w, h)).data;
+    stack.unshift(null, null);
+    for (var y = 0, p = 0; y < h; y++) {
+      stack[1] = y;
+      for (var x = 0; x < w; x++) {
+        stack[0] = x;
+        var color = this.binds.image.apply(this, stack);
+        data[p++] = color.r;
+        data[p++] = color.g;
+        data[p++] = color.b;
+        data[p++] = 255 * color.a;
+      }
+    }
+    stack.splice(0, 2);
+  }
+};
+/**
+ * Constructs a new wedge with default properties. Wedges are not typically
+ * constructed directly, but by adding to a panel or an existing mark via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents a wedge, or pie slice. Specified in terms of start and end
+ * angle, inner and outer radius, wedges can be used to construct donut charts
+ * and polar bar charts as well. If the {@link #angle} property is used, the end
+ * angle is implied by adding this value to start angle. By default, the start
+ * angle is the previously-generated wedge's end angle. This design allows
+ * explicit control over the wedge placement if desired, while offering
+ * convenient defaults for the construction of radial graphs.
+ *
+ * <p>The center point of the circle is positioned using the standard box model.
+ * The wedge can be stroked and filled, similar to {@link pv.Bar}.
+ *
+ * <p>See also the <a href="../../api/Wedge.html">Wedge guide</a>.
+ *
+ * @extends pv.Mark
+ */
+pv.Wedge = function() {
+  pv.Mark.call(this);
+};
+
+pv.Wedge.prototype = pv.extend(pv.Mark)
+    .property("startAngle", Number)
+    .property("endAngle", Number)
+    .property("angle", Number)
+    .property("innerRadius", Number)
+    .property("outerRadius", Number)
+    .property("lineWidth", Number)
+    .property("strokeStyle", pv.color)
+    .property("fillStyle", pv.color);
+
+pv.Wedge.prototype.type = "wedge";
+
+/**
+ * The start angle of the wedge, in radians. The start angle is measured
+ * clockwise from the 3 o'clock position. The default value of this property is
+ * the end angle of the previous instance (the {@link Mark#sibling}), or -PI / 2
+ * for the first wedge; for pie and donut charts, typically only the
+ * {@link #angle} property needs to be specified.
+ *
+ * @type number
+ * @name pv.Wedge.prototype.startAngle
+ */
+
+/**
+ * The end angle of the wedge, in radians. If not specified, the end angle is
+ * implied as the start angle plus the {@link #angle}.
+ *
+ * @type number
+ * @name pv.Wedge.prototype.endAngle
+ */
+
+/**
+ * The angular span of the wedge, in radians. This property is used if end angle
+ * is not specified.
+ *
+ * @type number
+ * @name pv.Wedge.prototype.angle
+ */
+
+/**
+ * The inner radius of the wedge, in pixels. The default value of this property
+ * is zero; a positive value will produce a donut slice rather than a pie slice.
+ * The inner radius can vary per-wedge.
+ *
+ * @type number
+ * @name pv.Wedge.prototype.innerRadius
+ */
+
+/**
+ * The outer radius of the wedge, in pixels. This property is required. For
+ * pies, only this radius is required; for donuts, the inner radius must be
+ * specified as well. The outer radius can vary per-wedge.
+ *
+ * @type number
+ * @name pv.Wedge.prototype.outerRadius
+ */
+
+/**
+ * The width of stroked lines, in pixels; used in conjunction with
+ * <tt>strokeStyle</tt> to stroke the wedge's border.
+ *
+ * @type number
+ * @name pv.Wedge.prototype.lineWidth
+ */
+
+/**
+ * The style of stroked lines; used in conjunction with <tt>lineWidth</tt> to
+ * stroke the wedge's border. The default value of this property is null,
+ * meaning wedges are not stroked by default.
+ *
+ * @type string
+ * @name pv.Wedge.prototype.strokeStyle
+ * @see pv.color
+ */
+
+/**
+ * The wedge fill style; if non-null, the interior of the wedge is filled with
+ * the specified color. The default value of this property is a categorical
+ * color.
+ *
+ * @type string
+ * @name pv.Wedge.prototype.fillStyle
+ * @see pv.color
+ */
+
+/**
+ * Default properties for wedges. By default, there is no stroke and the fill
+ * style is a categorical color.
+ *
+ * @type pv.Wedge
+ */
+pv.Wedge.prototype.defaults = new pv.Wedge()
+    .extend(pv.Mark.prototype.defaults)
+    .startAngle(function() {
+        var s = this.sibling();
+        return s ? s.endAngle : -Math.PI / 2;
+      })
+    .innerRadius(0)
+    .lineWidth(1.5)
+    .strokeStyle(null)
+    .fillStyle(pv.Colors.category20().by(pv.index));
+
+/**
+ * Returns the mid-radius of the wedge, which is defined as half-way between the
+ * inner and outer radii.
+ *
+ * @see #innerRadius
+ * @see #outerRadius
+ * @returns {number} the mid-radius, in pixels.
+ */
+pv.Wedge.prototype.midRadius = function() {
+  return (this.innerRadius() + this.outerRadius()) / 2;
+};
+
+/**
+ * Returns the mid-angle of the wedge, which is defined as half-way between the
+ * start and end angles.
+ *
+ * @see #startAngle
+ * @see #endAngle
+ * @returns {number} the mid-angle, in radians.
+ */
+pv.Wedge.prototype.midAngle = function() {
+  return (this.startAngle() + this.endAngle()) / 2;
+};
+
+/**
+ * Constructs a new wedge anchor with default properties. Wedges support five
+ * different anchors:<ul>
+ *
+ * <li>outer
+ * <li>inner
+ * <li>center
+ * <li>start
+ * <li>end
+ *
+ * </ul>In addition to positioning properties (left, right, top bottom), the
+ * anchors support text rendering properties (text-align, text-baseline,
+ * textAngle). Text is rendered to appear inside the wedge.
+ *
+ * @param {string} name the anchor name; either a string or a property function.
+ * @returns {pv.Anchor}
+ */
+pv.Wedge.prototype.anchor = function(name) {
+  function partial(s) { return s.innerRadius || s.angle < 2 * Math.PI; }
+  function midRadius(s) { return (s.innerRadius + s.outerRadius) / 2; }
+  function midAngle(s) { return (s.startAngle + s.endAngle) / 2; }
+  return pv.Mark.prototype.anchor.call(this, name)
+    .left(function() {
+        var s = this.scene.target[this.index];
+        if (partial(s)) switch (this.name()) {
+          case "outer": return s.left + s.outerRadius * Math.cos(midAngle(s));
+          case "inner": return s.left + s.innerRadius * Math.cos(midAngle(s));
+          case "start": return s.left + midRadius(s) * Math.cos(s.startAngle);
+          case "center": return s.left + midRadius(s) * Math.cos(midAngle(s));
+          case "end": return s.left + midRadius(s) * Math.cos(s.endAngle);
+        }
+        return s.left;
+      })
+    .top(function() {
+        var s = this.scene.target[this.index];
+        if (partial(s)) switch (this.name()) {
+          case "outer": return s.top + s.outerRadius * Math.sin(midAngle(s));
+          case "inner": return s.top + s.innerRadius * Math.sin(midAngle(s));
+          case "start": return s.top + midRadius(s) * Math.sin(s.startAngle);
+          case "center": return s.top + midRadius(s) * Math.sin(midAngle(s));
+          case "end": return s.top + midRadius(s) * Math.sin(s.endAngle);
+        }
+        return s.top;
+      })
+    .textAlign(function() {
+        var s = this.scene.target[this.index];
+        if (partial(s)) switch (this.name()) {
+          case "outer": return pv.Wedge.upright(midAngle(s)) ? "right" : "left";
+          case "inner": return pv.Wedge.upright(midAngle(s)) ? "left" : "right";
+        }
+        return "center";
+      })
+    .textBaseline(function() {
+        var s = this.scene.target[this.index];
+        if (partial(s)) switch (this.name()) {
+          case "start": return pv.Wedge.upright(s.startAngle) ? "top" : "bottom";
+          case "end": return pv.Wedge.upright(s.endAngle) ? "bottom" : "top";
+        }
+        return "middle";
+      })
+    .textAngle(function() {
+        var s = this.scene.target[this.index], a = 0;
+        if (partial(s)) switch (this.name()) {
+          case "center":
+          case "inner":
+          case "outer": a = midAngle(s); break;
+          case "start": a = s.startAngle; break;
+          case "end": a = s.endAngle; break;
+        }
+        return pv.Wedge.upright(a) ? a : (a + Math.PI);
+      });
+};
+
+/**
+ * Returns true if the specified angle is considered "upright", as in, text
+ * rendered at that angle would appear upright. If the angle is not upright,
+ * text is rotated 180 degrees to be upright, and the text alignment properties
+ * are correspondingly changed.
+ *
+ * @param {number} angle an angle, in radius.
+ * @returns {boolean} true if the specified angle is upright.
+ */
+pv.Wedge.upright = function(angle) {
+  angle = angle % (2 * Math.PI);
+  angle = (angle < 0) ? (2 * Math.PI + angle) : angle;
+  return (angle < Math.PI / 2) || (angle >= 3 * Math.PI / 2);
+};
+
+/** @private Sets angle based on endAngle or vice versa. */
+pv.Wedge.prototype.buildImplied = function(s) {
+  if (s.angle == null) s.angle = s.endAngle - s.startAngle;
+  else if (s.endAngle == null) s.endAngle = s.startAngle + s.angle;
+  pv.Mark.prototype.buildImplied.call(this, s);
+};
+/**
+ * Abstract; not implemented. There is no explicit constructor; this class
+ * merely serves to document the attributes that are used on particles in
+ * physics simulations.
+ *
+ * @class A weighted particle that can participate in a force simulation.
+ *
+ * @name pv.Particle
+ */
+
+/**
+ * The next particle in the simulation. Particles form a singly-linked list.
+ *
+ * @field
+ * @type pv.Particle
+ * @name pv.Particle.prototype.next
+ */
+
+/**
+ * The <i>x</i>-position of the particle.
+ *
+ * @field
+ * @type number
+ * @name pv.Particle.prototype.x
+ */
+
+/**
+ * The <i>y</i>-position of the particle.
+ *
+ * @field
+ * @type number
+ * @name pv.Particle.prototype.y
+ */
+
+/**
+ * The <i>x</i>-velocity of the particle.
+ *
+ * @field
+ * @type number
+ * @name pv.Particle.prototype.vx
+ */
+
+/**
+ * The <i>y</i>-velocity of the particle.
+ *
+ * @field
+ * @type number
+ * @name pv.Particle.prototype.vy
+ */
+
+/**
+ * The <i>x</i>-position of the particle at -dt.
+ *
+ * @field
+ * @type number
+ * @name pv.Particle.prototype.px
+ */
+
+/**
+ * The <i>y</i>-position of the particle at -dt.
+ *
+ * @field
+ * @type number
+ * @name pv.Particle.prototype.py
+ */
+
+/**
+ * The <i>x</i>-force on the particle.
+ *
+ * @field
+ * @type number
+ * @name pv.Particle.prototype.fx
+ */
+
+/**
+ * The <i>y</i>-force on the particle.
+ *
+ * @field
+ * @type number
+ * @name pv.Particle.prototype.fy
+ */
+/**
+ * Constructs a new empty simulation.
+ *
+ * @param {array} particles
+ * @returns {pv.Simulation} a new simulation for the specified particles.
+ * @see pv.Simulation
+ */
+pv.simulation = function(particles) {
+  return new pv.Simulation(particles);
+};
+
+/**
+ * Constructs a new simulation for the specified particles.
+ *
+ * @class Represents a particle simulation. Particles are massive points in
+ * two-dimensional space. Forces can be applied to these particles, causing them
+ * to move. Constraints can also be applied to restrict particle movement, for
+ * example, constraining particles to a fixed position, or simulating collision
+ * between circular particles with area.
+ *
+ * <p>The simulation uses <a
+ * href="http://en.wikipedia.org/wiki/Verlet_integration">Position Verlet</a>
+ * integration, due to the ease with which <a
+ * href="http://www.teknikus.dk/tj/gdc2001.htm">geometric constraints</a> can be
+ * implemented. For each time step, Verlet integration is performed, new forces
+ * are accumulated, and then constraints are applied.
+ *
+ * <p>The simulation makes two simplifying assumptions: all particles are
+ * equal-mass, and the time step of the simulation is fixed. It would be easy to
+ * incorporate variable-mass particles as a future enhancement. Variable time
+ * steps are also possible, but are likely to introduce instability in the
+ * simulation.
+ *
+ * <p>This class can be used directly to simulate particle interaction.
+ * Alternatively, for network diagrams, see {@link pv.Layout.Force}.
+ *
+ * @param {array} particles an array of {@link pv.Particle}s to simulate.
+ * @see pv.Layout.Force
+ * @see pv.Force
+ * @see pv.Constraint
+ */
+pv.Simulation = function(particles) {
+  for (var i = 0; i < particles.length; i++) this.particle(particles[i]);
+};
+
+/**
+ * The particles in the simulation. Particles are stored as a linked list; this
+ * field represents the first particle in the simulation.
+ *
+ * @field
+ * @type pv.Particle
+ * @name pv.Simulation.prototype.particles
+ */
+
+/**
+ * The forces in the simulation. Forces are stored as a linked list; this field
+ * represents the first force in the simulation.
+ *
+ * @field
+ * @type pv.Force
+ * @name pv.Simulation.prototype.forces
+ */
+
+/**
+ * The constraints in the simulation. Constraints are stored as a linked list;
+ * this field represents the first constraint in the simulation.
+ *
+ * @field
+ * @type pv.Constraint
+ * @name pv.Simulation.prototype.constraints
+ */
+
+/**
+ * Adds the specified particle to the simulation.
+ *
+ * @param {pv.Particle} p the new particle.
+ * @returns {pv.Simulation} this.
+ */
+pv.Simulation.prototype.particle = function(p) {
+  p.next = this.particles;
+  /* Default velocities and forces to zero if unset. */
+  if (isNaN(p.px)) p.px = p.x;
+  if (isNaN(p.py)) p.py = p.y;
+  if (isNaN(p.fx)) p.fx = 0;
+  if (isNaN(p.fy)) p.fy = 0;
+  this.particles = p;
+  return this;
+};
+
+/**
+ * Adds the specified force to the simulation.
+ *
+ * @param {pv.Force} f the new force.
+ * @returns {pv.Simulation} this.
+ */
+pv.Simulation.prototype.force = function(f) {
+  f.next = this.forces;
+  this.forces = f;
+  return this;
+};
+
+/**
+ * Adds the specified constraint to the simulation.
+ *
+ * @param {pv.Constraint} c the new constraint.
+ * @returns {pv.Simulation} this.
+ */
+pv.Simulation.prototype.constraint = function(c) {
+  c.next = this.constraints;
+  this.constraints = c;
+  return this;
+};
+
+/**
+ * Apply constraints, and then set the velocities to zero.
+ *
+ * @returns {pv.Simulation} this.
+ */
+pv.Simulation.prototype.stabilize = function(n) {
+  var c;
+  if (!arguments.length) n = 3; // TODO use cooling schedule
+  for (var i = 0; i < n; i++) {
+    var q = new pv.Quadtree(this.particles);
+    for (c = this.constraints; c; c = c.next) c.apply(this.particles, q);
+  }
+  for (var p = this.particles; p; p = p.next) {
+    p.px = p.x;
+    p.py = p.y;
+  }
+  return this;
+};
+
+/**
+ * Advances the simulation one time-step.
+ */
+pv.Simulation.prototype.step = function() {
+  var p, f, c;
+
+  /*
+   * Assumptions:
+   * - The mass (m) of every particles is 1.
+   * - The time step (dt) is 1.
+   */
+
+  /* Position Verlet integration. */
+  for (p = this.particles; p; p = p.next) {
+    var px = p.px, py = p.py;
+    p.px = p.x;
+    p.py = p.y;
+    p.x += p.vx = ((p.x - px) + p.fx);
+    p.y += p.vy = ((p.y - py) + p.fy);
+  }
+
+  /* Apply constraints, then accumulate new forces. */
+  var q = new pv.Quadtree(this.particles);
+  for (c = this.constraints; c; c = c.next) c.apply(this.particles, q);
+  for (p = this.particles; p; p = p.next) p.fx = p.fy = 0;
+  for (f = this.forces; f; f = f.next) f.apply(this.particles, q);
+};
+/**
+ * Constructs a new quadtree for the specified array of particles.
+ *
+ * @class Represents a quadtree: a two-dimensional recursive spatial
+ * subdivision. This particular implementation uses square partitions, dividing
+ * each square into four equally-sized squares. Each particle exists in a unique
+ * node; if multiple particles are in the same position, some particles may be
+ * stored on internal nodes rather than leaf nodes.
+ *
+ * <p>This quadtree can be used to accelerate various spatial operations, such
+ * as the Barnes-Hut approximation for computing n-body forces, or collision
+ * detection.
+ *
+ * @see pv.Force.charge
+ * @see pv.Constraint.collision
+ * @param {pv.Particle} particles the linked list of particles.
+ */
+pv.Quadtree = function(particles) {
+  var p;
+
+  /* Compute bounds. */
+  var x1 = Number.POSITIVE_INFINITY, y1 = x1,
+      x2 = Number.NEGATIVE_INFINITY, y2 = x2;
+  for (p = particles; p; p = p.next) {
+    if (p.x < x1) x1 = p.x;
+    if (p.y < y1) y1 = p.y;
+    if (p.x > x2) x2 = p.x;
+    if (p.y > y2) y2 = p.y;
+  }
+
+  /* Squarify the bounds. */
+  var dx = x2 - x1, dy = y2 - y1;
+  if (dx > dy) y2 = y1 + dx;
+  else x2 = x1 + dy;
+  this.xMin = x1;
+  this.yMin = y1;
+  this.xMax = x2;
+  this.yMax = y2;
+
+  /**
+   * @ignore Recursively inserts the specified particle <i>p</i> at the node
+   * <i>n</i> or one of its descendants. The bounds are defined by [<i>x1</i>,
+   * <i>x2</i>] and [<i>y1</i>, <i>y2</i>].
+   */
+  function insert(n, p, x1, y1, x2, y2) {
+    if (isNaN(p.x) || isNaN(p.y)) return; // ignore invalid particles
+    if (n.leaf) {
+      if (n.p) {
+        /*
+         * If the particle at this leaf node is at the same position as the new
+         * particle we are adding, we leave the particle associated with the
+         * internal node while adding the new particle to a child node. This
+         * avoids infinite recursion.
+         */
+        if ((Math.abs(n.p.x - p.x) + Math.abs(n.p.y - p.y)) < .01) {
+          insertChild(n, p, x1, y1, x2, y2);
+        } else {
+          var v = n.p;
+          n.p = null;
+          insertChild(n, v, x1, y1, x2, y2);
+          insertChild(n, p, x1, y1, x2, y2);
+        }
+      } else {
+        n.p = p;
+      }
+    } else {
+      insertChild(n, p, x1, y1, x2, y2);
+    }
+  }
+
+  /**
+   * @ignore Recursively inserts the specified particle <i>p</i> into a
+   * descendant of node <i>n</i>. The bounds are defined by [<i>x1</i>,
+   * <i>x2</i>] and [<i>y1</i>, <i>y2</i>].
+   */
+  function insertChild(n, p, x1, y1, x2, y2) {
+    /* Compute the split point, and the quadrant in which to insert p. */
+    var sx = (x1 + x2) * .5,
+        sy = (y1 + y2) * .5,
+        right = p.x >= sx,
+        bottom = p.y >= sy;
+
+    /* Recursively insert into the child node. */
+    n.leaf = false;
+    switch ((bottom << 1) + right) {
+      case 0: n = n.c1 || (n.c1 = new pv.Quadtree.Node()); break;
+      case 1: n = n.c2 || (n.c2 = new pv.Quadtree.Node()); break;
+      case 2: n = n.c3 || (n.c3 = new pv.Quadtree.Node()); break;
+      case 3: n = n.c4 || (n.c4 = new pv.Quadtree.Node()); break;
+    }
+
+    /* Update the bounds as we recurse. */
+    if (right) x1 = sx; else x2 = sx;
+    if (bottom) y1 = sy; else y2 = sy;
+    insert(n, p, x1, y1, x2, y2);
+  }
+
+  /* Insert all particles. */
+  this.root = new pv.Quadtree.Node();
+  for (p = particles; p; p = p.next) insert(this.root, p, x1, y1, x2, y2);
+};
+
+/**
+ * The root node of the quadtree.
+ *
+ * @type pv.Quadtree.Node
+ * @name pv.Quadtree.prototype.root
+ */
+
+/**
+ * The minimum x-coordinate value of all contained particles.
+ *
+ * @type number
+ * @name pv.Quadtree.prototype.xMin
+ */
+
+/**
+ * The maximum x-coordinate value of all contained particles.
+ *
+ * @type number
+ * @name pv.Quadtree.prototype.xMax
+ */
+
+/**
+ * The minimum y-coordinate value of all contained particles.
+ *
+ * @type number
+ * @name pv.Quadtree.prototype.yMin
+ */
+
+/**
+ * The maximum y-coordinate value of all contained particles.
+ *
+ * @type number
+ * @name pv.Quadtree.prototype.yMax
+ */
+
+/**
+ * Constructs a new node.
+ *
+ * @class A node in a quadtree.
+ *
+ * @see pv.Quadtree
+ */
+pv.Quadtree.Node = function() {
+  /*
+   * Prepopulating all attributes significantly increases performance! Also,
+   * letting the language interpreter manage garbage collection was moderately
+   * faster than creating a cache pool.
+   */
+  this.leaf = true;
+  this.c1 = null;
+  this.c2 = null;
+  this.c3 = null;
+  this.c4 = null;
+  this.p = null;
+};
+
+/**
+ * True if this node is a leaf node; i.e., it has no children. Note that both
+ * leaf nodes and non-leaf (internal) nodes may have associated particles. If
+ * this is a non-leaf node, then at least one of {@link #c1}, {@link #c2},
+ * {@link #c3} or {@link #c4} is guaranteed to be non-null.
+ *
+ * @type boolean
+ * @name pv.Quadtree.Node.prototype.leaf
+ */
+
+/**
+ * The particle associated with this node, if any.
+ *
+ * @type pv.Particle
+ * @name pv.Quadtree.Node.prototype.p
+ */
+
+/**
+ * The child node for the second quadrant, if any.
+ *
+ * @type pv.Quadtree.Node
+ * @name pv.Quadtree.Node.prototype.c2
+ */
+
+/**
+ * The child node for the third quadrant, if any.
+ *
+ * @type pv.Quadtree.Node
+ * @name pv.Quadtree.Node.prototype.c3
+ */
+
+/**
+ * The child node for the fourth quadrant, if any.
+ *
+ * @type pv.Quadtree.Node
+ * @name pv.Quadtree.Node.prototype.c4
+ */
+/**
+ * Abstract; see an implementing class.
+ *
+ * @class Represents a force that acts on particles. Note that this interface
+ * does not specify how to bind a force to specific particles; in general,
+ * forces are applied globally to all particles. However, some forces may be
+ * applied to specific particles or between particles, such as spring forces,
+ * through additional specialization.
+ *
+ * @see pv.Simulation
+ * @see pv.Particle
+ * @see pv.Force.charge
+ * @see pv.Force.drag
+ * @see pv.Force.spring
+ */
+pv.Force = {};
+
+/**
+ * Applies this force to the specified particles.
+ *
+ * @function
+ * @name pv.Force.prototype.apply
+ * @param {pv.Particle} particles particles to which to apply this force.
+ * @param {pv.Quadtree} q a quadtree for spatial acceleration.
+ */
+/**
+ * Constructs a new charge force, with an optional charge constant. The charge
+ * constant can be negative for repulsion (e.g., particles with electrical
+ * charge of equal sign), or positive for attraction (e.g., massive particles
+ * with mutual gravity). The default charge constant is -40.
+ *
+ * @class An n-body force, as defined by Coulomb's law or Newton's law of
+ * gravitation, inversely proportional to the square of the distance between
+ * particles. Note that the force is independent of the <i>mass</i> of the
+ * associated particles, and that the particles do not have charges of varying
+ * magnitude; instead, the attraction or repulsion of all particles is globally
+ * specified as the charge {@link #constant}.
+ *
+ * <p>This particular implementation uses the Barnes-Hut algorithm. For details,
+ * see <a
+ * href="http://www.nature.com/nature/journal/v324/n6096/abs/324446a0.html">"A
+ * hierarchical O(N log N) force-calculation algorithm"</a>, J. Barnes &amp;
+ * P. Hut, <i>Nature</i> 1986.
+ *
+ * @name pv.Force.charge
+ * @param {number} [k] the charge constant.
+ */
+pv.Force.charge = function(k) {
+  var min = 2, // minimum distance at which to observe forces
+      min1 = 1 / min,
+      max = 500, // maximum distance at which to observe forces
+      max1 = 1 / max,
+      theta = .9, // Barnes-Hut theta approximation constant
+      force = {};
+
+  if (!arguments.length) k = -40; // default charge constant (repulsion)
+
+  /**
+   * Sets or gets the charge constant. If an argument is specified, it is the
+   * new charge constant. The charge constant can be negative for repulsion
+   * (e.g., particles with electrical charge of equal sign), or positive for
+   * attraction (e.g., massive particles with mutual gravity). The default
+   * charge constant is -40.
+   *
+   * @function
+   * @name pv.Force.charge.prototype.constant
+   * @param {number} x the charge constant.
+   * @returns {pv.Force.charge} this.
+   */
+  force.constant = function(x) {
+    if (arguments.length) {
+      k = Number(x);
+      return force;
+    }
+    return k;
+  };
+
+  /**
+   * Sets or gets the domain; specifies the minimum and maximum domain within
+   * which charge forces are applied. A minimum distance threshold avoids
+   * applying forces that are two strong (due to granularity of the simulation's
+   * numeric integration). A maximum distance threshold improves performance by
+   * skipping force calculations for particles that are far apart.
+   *
+   * <p>The default domain is [2, 500].
+   *
+   * @function
+   * @name pv.Force.charge.prototype.domain
+   * @param {number} a
+   * @param {number} b
+   * @returns {pv.Force.charge} this.
+   */
+  force.domain = function(a, b) {
+    if (arguments.length) {
+      min = Number(a);
+      min1 = 1 / min;
+      max = Number(b);
+      max1 = 1 / max;
+      return force;
+    }
+    return [min, max];
+  };
+
+  /**
+   * Sets or gets the Barnes-Hut approximation factor. The Barnes-Hut
+   * approximation criterion is the ratio of the size of the quadtree node to
+   * the distance from the point to the node's center of mass is beneath some
+   * threshold.
+   *
+   * @function
+   * @name pv.Force.charge.prototype.theta
+   * @param {number} x the new Barnes-Hut approximation factor.
+   * @returns {pv.Force.charge} this.
+   */
+  force.theta = function(x) {
+    if (arguments.length) {
+      theta = Number(x);
+      return force;
+    }
+    return theta;
+  };
+
+  /**
+   * @ignore Recursively computes the center of charge for each node in the
+   * quadtree. This is equivalent to the center of mass, assuming that all
+   * particles have unit weight.
+   */
+  function accumulate(n) {
+    var cx = 0, cy = 0;
+    n.cn = 0;
+    function accumulateChild(c) {
+      accumulate(c);
+      n.cn += c.cn;
+      cx += c.cn * c.cx;
+      cy += c.cn * c.cy;
+    }
+    if (!n.leaf) {
+      if (n.c1) accumulateChild(n.c1);
+      if (n.c2) accumulateChild(n.c2);
+      if (n.c3) accumulateChild(n.c3);
+      if (n.c4) accumulateChild(n.c4);
+    }
+    if (n.p) {
+      n.cn += k;
+      cx += k * n.p.x;
+      cy += k * n.p.y;
+    }
+    n.cx = cx / n.cn;
+    n.cy = cy / n.cn;
+  }
+
+  /**
+   * @ignore Recursively computes forces on the given particle using the given
+   * quadtree node. The Barnes-Hut approximation criterion is the ratio of the
+   * size of the quadtree node to the distance from the point to the node's
+   * center of mass is beneath some threshold.
+   */
+  function forces(n, p, x1, y1, x2, y2) {
+    var dx = n.cx - p.x,
+        dy = n.cy - p.y,
+        dn = 1 / Math.sqrt(dx * dx + dy * dy);
+
+    /* Barnes-Hut criterion. */
+    if ((n.leaf && (n.p != p)) || ((x2 - x1) * dn < theta)) {
+      if (dn < max1) return;
+      if (dn > min1) dn = min1;
+      var kc = n.cn * dn * dn * dn,
+          fx = dx * kc,
+          fy = dy * kc;
+      p.fx += fx;
+      p.fy += fy;
+    } else if (!n.leaf) {
+      var sx = (x1 + x2) * .5, sy = (y1 + y2) * .5;
+      if (n.c1) forces(n.c1, p, x1, y1, sx, sy);
+      if (n.c2) forces(n.c2, p, sx, y1, x2, sy);
+      if (n.c3) forces(n.c3, p, x1, sy, sx, y2);
+      if (n.c4) forces(n.c4, p, sx, sy, x2, y2);
+      if (dn < max1) return;
+      if (dn > min1) dn = min1;
+      if (n.p && (n.p != p)) {
+        var kc = k * dn * dn * dn,
+            fx = dx * kc,
+            fy = dy * kc;
+        p.fx += fx;
+        p.fy += fy;
+      }
+    }
+  }
+
+  /**
+   * Applies this force to the specified particles. The force is applied between
+   * all pairs of particles within the domain, using the specified quadtree to
+   * accelerate n-body force calculation using the Barnes-Hut approximation
+   * criterion.
+   *
+   * @function
+   * @name pv.Force.charge.prototype.apply
+   * @param {pv.Particle} particles particles to which to apply this force.
+   * @param {pv.Quadtree} q a quadtree for spatial acceleration.
+   */
+  force.apply = function(particles, q) {
+    accumulate(q.root);
+    for (var p = particles; p; p = p.next) {
+      forces(q.root, p, q.xMin, q.yMin, q.xMax, q.yMax);
+    }
+  };
+
+  return force;
+};
+/**
+ * Constructs a new drag force with the specified constant.
+ *
+ * @class Implements a drag force, simulating friction. The drag force is
+ * applied in the opposite direction of the particle's velocity. Since Position
+ * Verlet integration does not track velocities explicitly, the error term with
+ * this estimate of velocity is fairly high, so the drag force may be
+ * inaccurate.
+ *
+ * @extends pv.Force
+ * @param {number} k the drag constant.
+ * @see #constant
+ */
+pv.Force.drag = function(k) {
+  var force = {};
+
+  if (!arguments.length) k = .1; // default drag constant
+
+  /**
+   * Sets or gets the drag constant, in the range [0,1]. The default drag
+   * constant is 0.1. The drag forces scales linearly with the particle's
+   * velocity based on the given drag constant.
+   *
+   * @function
+   * @name pv.Force.drag.prototype.constant
+   * @param {number} x the new drag constant.
+   * @returns {pv.Force.drag} this, or the current drag constant.
+   */
+  force.constant = function(x) {
+    if (arguments.length) { k = x; return force; }
+    return k;
+  };
+
+  /**
+   * Applies this force to the specified particles.
+   *
+   * @function
+   * @name pv.Force.drag.prototype.apply
+   * @param {pv.Particle} particles particles to which to apply this force.
+   */
+  force.apply = function(particles) {
+    if (k) for (var p = particles; p; p = p.next) {
+      p.fx -= k * p.vx;
+      p.fy -= k * p.vy;
+    }
+  };
+
+  return force;
+};
+/**
+ * Constructs a new spring force with the specified constant. The links
+ * associated with this spring force must be specified before the spring force
+ * can be applied.
+ *
+ * @class Implements a spring force, per Hooke's law. The spring force can be
+ * configured with a tension constant, rest length, and damping factor. The
+ * tension and damping will automatically be normalized using the inverse square
+ * root of the maximum link degree of attached nodes; this makes springs weaker
+ * between nodes of high link degree.
+ *
+ * <p>Unlike other forces (such as charge and drag forces) which may be applied
+ * globally, spring forces are only applied between linked particles. Therefore,
+ * an array of links must be specified before this force can be applied; the
+ * links should be an array of {@link pv.Layout.Network.Link}s. See also
+ * {@link pv.Layout.Force} for an example of using spring and charge forces for
+ * network layout.
+ *
+ * @extends pv.Force
+ * @param {number} k the spring constant.
+ * @see #constant
+ * @see #links
+ */
+pv.Force.spring = function(k) {
+  var d = .1, // default damping factor
+      l = 20, // default rest length
+      links, // links on which to apply spring forces
+      kl, // per-spring normalization
+      force = {};
+
+  if (!arguments.length) k = .1; // default spring constant (tension)
+
+  /**
+   * Sets or gets the links associated with this spring force. Unlike other
+   * forces (such as charge and drag forces) which may be applied globally,
+   * spring forces are only applied between linked particles. Therefore, an
+   * array of links must be specified before this force can be applied; the
+   * links should be an array of {@link pv.Layout.Network.Link}s.
+   *
+   * @function
+   * @name pv.Force.spring.prototype.links
+   * @param {array} x the new array of links.
+   * @returns {pv.Force.spring} this, or the current array of links.
+   */
+  force.links = function(x) {
+    if (arguments.length) {
+      links = x;
+      kl = x.map(function(l) {
+          return 1 / Math.sqrt(Math.max(
+              l.sourceNode.linkDegree,
+              l.targetNode.linkDegree));
+        });
+      return force;
+    }
+    return links;
+  };
+
+  /**
+   * Sets or gets the spring constant. The default value is 0.1; greater values
+   * will result in stronger tension. The spring tension is automatically
+   * normalized using the inverse square root of the maximum link degree of
+   * attached nodes.
+   *
+   * @function
+   * @name pv.Force.spring.prototype.constant
+   * @param {number} x the new spring constant.
+   * @returns {pv.Force.spring} this, or the current spring constant.
+   */
+  force.constant = function(x) {
+    if (arguments.length) {
+      k = Number(x);
+      return force;
+    }
+    return k;
+  };
+
+  /**
+   * The spring damping factor, in the range [0,1]. Damping functions
+   * identically to drag forces, damping spring bounciness by applying a force
+   * in the opposite direction of attached nodes' velocities. The default value
+   * is 0.1. The spring damping is automatically normalized using the inverse
+   * square root of the maximum link degree of attached nodes.
+   *
+   * @function
+   * @name pv.Force.spring.prototype.damping
+   * @param {number} x the new spring damping factor.
+   * @returns {pv.Force.spring} this, or the current spring damping factor.
+   */
+  force.damping = function(x) {
+    if (arguments.length) {
+      d = Number(x);
+      return force;
+    }
+    return d;
+  };
+
+  /**
+   * The spring rest length. The default value is 20 pixels.
+   *
+   * @function
+   * @name pv.Force.spring.prototype.length
+   * @param {number} x the new spring rest length.
+   * @returns {pv.Force.spring} this, or the current spring rest length.
+   */
+  force.length = function(x) {
+    if (arguments.length) {
+      l = Number(x);
+      return force;
+    }
+    return l;
+  };
+
+  /**
+   * Applies this force to the specified particles.
+   *
+   * @function
+   * @name pv.Force.spring.prototype.apply
+   * @param {pv.Particle} particles particles to which to apply this force.
+   */
+  force.apply = function(particles) {
+    for (var i = 0; i < links.length; i++) {
+      var a = links[i].sourceNode,
+          b = links[i].targetNode,
+          dx = a.x - b.x,
+          dy = a.y - b.y,
+          dn = Math.sqrt(dx * dx + dy * dy),
+          dd = dn ? (1 / dn) : 1,
+          ks = k * kl[i], // normalized tension
+          kd = d * kl[i], // normalized damping
+          kk = (ks * (dn - l) + kd * (dx * (a.vx - b.vx) + dy * (a.vy - b.vy)) * dd) * dd,
+          fx = -kk * (dn ? dx : (.01 * (.5 - Math.random()))),
+          fy = -kk * (dn ? dy : (.01 * (.5 - Math.random())));
+      a.fx += fx;
+      a.fy += fy;
+      b.fx -= fx;
+      b.fy -= fy;
+    }
+  };
+
+  return force;
+};
+/**
+ * Abstract; see an implementing class.
+ *
+ * @class Represents a constraint that acts on particles. Note that this
+ * interface does not specify how to bind a constraint to specific particles; in
+ * general, constraints are applied globally to all particles. However, some
+ * constraints may be applied to specific particles or between particles, such
+ * as position constraints, through additional specialization.
+ *
+ * @see pv.Simulation
+ * @see pv.Particle
+ * @see pv.Constraint.bound
+ * @see pv.Constraint.collision
+ * @see pv.Constraint.position
+ */
+pv.Constraint = {};
+
+/**
+ * Applies this constraint to the specified particles.
+ *
+ * @function
+ * @name pv.Constraint.prototype.apply
+ * @param {pv.Particle} particles particles to which to apply this constraint.
+ * @param {pv.Quadtree} q a quadtree for spatial acceleration.
+ * @returns {pv.Constraint} this.
+ */
+/**
+ * Constructs a new collision constraint. The default search radius is 10, and
+ * the default repeat count is 1. A radius function must be specified to compute
+ * the radius of particles.
+ *
+ * @class Constraints circles to avoid overlap. Each particle is treated as a
+ * circle, with the radius of the particle computed using a specified function.
+ * For example, if the particle has an <tt>r</tt> attribute storing the radius,
+ * the radius <tt>function(d) d.r</tt> specifies a collision constraint using
+ * this radius. The radius function is passed each {@link pv.Particle} as the
+ * first argument.
+ *
+ * <p>To accelerate collision detection, this implementation uses a quadtree and
+ * a search radius. The search radius is computed as the maximum radius of all
+ * particles in the simulation.
+ *
+ * @see pv.Constraint
+ * @param {function} radius the radius function.
+ */
+pv.Constraint.collision = function(radius) {
+  var n = 1, // number of times to repeat the constraint
+      r1,
+      px1,
+      py1,
+      px2,
+      py2,
+      constraint = {};
+
+  if (!arguments.length) r1 = 10; // default search radius
+
+  /**
+   * Sets or gets the repeat count. If the repeat count is greater than 1, the
+   * constraint will be applied repeatedly; this is a form of the Gauss-Seidel
+   * method for constraints relaxation. Repeating the collision constraint makes
+   * the constraint have more of an effect when there is a potential for many
+   * co-occurring collisions.
+   *
+   * @function
+   * @name pv.Constraint.collision.prototype.repeat
+   * @param {number} x the number of times to repeat this constraint.
+   * @returns {pv.Constraint.collision} this.
+   */
+  constraint.repeat = function(x) {
+    if (arguments.length) {
+      n = Number(x);
+      return constraint;
+    }
+    return n;
+  };
+
+  /** @private */
+  function constrain(n, p, x1, y1, x2, y2) {
+    if (!n.leaf) {
+      var sx = (x1 + x2) * .5,
+          sy = (y1 + y2) * .5,
+          top = sy > py1,
+          bottom = sy < py2,
+          left = sx > px1,
+          right = sx < px2;
+      if (top) {
+        if (n.c1 && left) constrain(n.c1, p, x1, y1, sx, sy);
+        if (n.c2 && right) constrain(n.c2, p, sx, y1, x2, sy);
+      }
+      if (bottom) {
+        if (n.c3 && left) constrain(n.c3, p, x1, sy, sx, y2);
+        if (n.c4 && right) constrain(n.c4, p, sx, sy, x2, y2);
+      }
+    }
+    if (n.p && (n.p != p)) {
+      var dx = p.x - n.p.x,
+          dy = p.y - n.p.y,
+          l = Math.sqrt(dx * dx + dy * dy),
+          d = r1 + radius(n.p);
+      if (l < d) {
+        var k = (l - d) / l * .5;
+        dx *= k;
+        dy *= k;
+        p.x -= dx;
+        p.y -= dy;
+        n.p.x += dx;
+        n.p.y += dy;
+      }
+    }
+  }
+
+  /**
+   * Applies this constraint to the specified particles.
+   *
+   * @function
+   * @name pv.Constraint.collision.prototype.apply
+   * @param {pv.Particle} particles particles to which to apply this constraint.
+   * @param {pv.Quadtree} q a quadtree for spatial acceleration.
+   */
+  constraint.apply = function(particles, q) {
+    var p, r, max = -Infinity;
+    for (p = particles; p; p = p.next) {
+      r = radius(p);
+      if (r > max) max = r;
+    }
+    for (var i = 0; i < n; i++) {
+      for (p = particles; p; p = p.next) {
+        r = (r1 = radius(p)) + max;
+        px1 = p.x - r;
+        px2 = p.x + r;
+        py1 = p.y - r;
+        py2 = p.y + r;
+        constrain(q.root, p, q.xMin, q.yMin, q.xMax, q.yMax);
+      }
+    }
+  };
+
+  return constraint;
+};
+/**
+ * Constructs a default position constraint using the <tt>fix</tt> attribute.
+ * An optional position function can be specified to determine how the fixed
+ * position per-particle is determined.
+ *
+ * @class Constraints particles to a fixed position. The fixed position per
+ * particle is determined using a given position function, which defaults to
+ * <tt>function(d) d.fix</tt>.
+ *
+ * <p>If the position function returns null, then no position constraint is
+ * applied to the given particle. Otherwise, the particle's position is set to
+ * the returned position, as expressed by a {@link pv.Vector}. (Note: the
+ * position does not need to be an instance of <tt>pv.Vector</tt>, but simply an
+ * object with <tt>x</tt> and <tt>y</tt> attributes.)
+ *
+ * <p>This constraint also supports a configurable alpha parameter, which
+ * defaults to 1. If the alpha parameter is in the range [0,1], then rather than
+ * setting the particle's new position directly to the position returned by the
+ * supplied position function, the particle's position is interpolated towards
+ * the fixed position. This results is a smooth (exponential) drift towards the
+ * fixed position, which can increase the stability of the physics simulation.
+ * In addition, the alpha parameter can be decayed over time, relaxing the
+ * position constraint, which helps to stabilize on an optimal solution.
+ *
+ * @param {function} [f] the position function.
+ */
+pv.Constraint.position = function(f) {
+  var a = 1, // default alpha
+      constraint = {};
+
+  if (!arguments.length) /** @ignore */ f = function(p) { return p.fix; };
+
+  /**
+   * Sets or gets the alpha parameter for position interpolation. If the alpha
+   * parameter is in the range [0,1], then rather than setting the particle's
+   * new position directly to the position returned by the supplied position
+   * function, the particle's position is interpolated towards the fixed
+   * position.
+   *
+   * @function
+   * @name pv.Constraint.position.prototype.alpha
+   * @param {number} x the new alpha parameter, in the range [0,1].
+   * @returns {pv.Constraint.position} this.
+   */
+  constraint.alpha = function(x) {
+    if (arguments.length) {
+      a = Number(x);
+      return constraint;
+    }
+    return a;
+  };
+
+  /**
+   * Applies this constraint to the specified particles.
+   *
+   * @function
+   * @name pv.Constraint.position.prototype.apply
+   * @param {pv.Particle} particles particles to which to apply this constraint.
+   */
+  constraint.apply = function(particles) {
+    for (var p = particles; p; p = p.next) {
+      var v = f(p);
+      if (v) {
+        p.x += (v.x - p.x) * a;
+        p.y += (v.y - p.y) * a;
+        p.fx = p.fy = p.vx = p.vy = 0;
+      }
+    }
+  };
+
+  return constraint;
+};
+/**
+ * Constructs a new bound constraint. Before the constraint can be used, the
+ * {@link #x} and {@link #y} methods must be call to specify the bounds.
+ *
+ * @class Constrains particles to within fixed rectangular bounds. For example,
+ * this constraint can be used to constrain particles in a physics simulation
+ * within the bounds of an enclosing panel.
+ *
+ * <p>Note that the current implementation treats particles as points, with no
+ * area. If the particles are rendered as dots, be sure to include some
+ * additional padding to inset the bounds such that the edges of the dots do not
+ * get clipped by the panel bounds. If the particles have different radii, this
+ * constraint would need to be extended using a radius function, similar to
+ * {@link pv.Constraint.collision}.
+ *
+ * @see pv.Layout.Force
+ * @extends pv.Constraint
+ */
+pv.Constraint.bound = function() {
+  var constraint = {},
+      x,
+      y;
+
+  /**
+   * Sets or gets the bounds on the x-coordinate.
+   *
+   * @function
+   * @name pv.Constraint.bound.prototype.x
+   * @param {number} min the minimum allowed x-coordinate.
+   * @param {number} max the maximum allowed x-coordinate.
+   * @returns {pv.Constraint.bound} this.
+   */
+  constraint.x = function(min, max) {
+    if (arguments.length) {
+      x = {min: Math.min(min, max), max: Math.max(min, max)};
+      return this;
+    }
+    return x;
+  };
+
+  /**
+   * Sets or gets the bounds on the y-coordinate.
+   *
+   * @function
+   * @name pv.Constraint.bound.prototype.y
+   * @param {number} min the minimum allowed y-coordinate.
+   * @param {number} max the maximum allowed y-coordinate.
+   * @returns {pv.Constraint.bound} this.
+   */
+  constraint.y = function(min, max) {
+    if (arguments.length) {
+      y = {min: Math.min(min, max), max: Math.max(min, max)};
+      return this;
+    }
+    return y;
+  };
+
+  /**
+   * Applies this constraint to the specified particles.
+   *
+   * @function
+   * @name pv.Constraint.bound.prototype.apply
+   * @param {pv.Particle} particles particles to which to apply this constraint.
+   */
+  constraint.apply = function(particles) {
+    if (x) for (var p = particles; p; p = p.next) {
+      p.x = p.x < x.min ? x.min : (p.x > x.max ? x.max : p.x);
+    }
+    if (y) for (var p = particles; p; p = p.next) {
+      p.y = p.y < y.min ? y.min : (p.y > y.max ? y.max : p.y);
+    }
+  };
+
+  return constraint;
+};
+/**
+ * Constructs a new, empty layout with default properties. Layouts are not
+ * typically constructed directly; instead, a concrete subclass is added to an
+ * existing panel via {@link pv.Mark#add}.
+ *
+ * @class Represents an abstract layout, encapsulating a visualization technique
+ * such as a streamgraph or treemap. Layouts are themselves containers,
+ * extending from {@link pv.Panel}, and defining a set of mark prototypes as
+ * children. These mark prototypes provide default properties that together
+ * implement the given visualization technique.
+ *
+ * <p>Layouts do not initially contain any marks; any exported marks (such as a
+ * network layout's <tt>link</tt> and <tt>node</tt>) are intended to be used as
+ * prototypes. By adding a concrete mark, such as a {@link pv.Bar}, to the
+ * appropriate mark prototype, the mark is added to the layout and inherits the
+ * given properties. This approach allows further customization of the layout,
+ * either by choosing a different mark type to add, or more simply by overriding
+ * some of the layout's defined properties.
+ *
+ * <p>Each concrete layout, such as treemap or circle-packing, has different
+ * behavior and may export different mark prototypes, depending on what marks
+ * are typically needed to render the desired visualization. Therefore it is
+ * important to understand how each layout is structured, such that the provided
+ * mark prototypes are used appropriately.
+ *
+ * <p>In addition to the mark prototypes, layouts may define custom properties
+ * that affect the overall behavior of the layout. For example, a treemap layout
+ * might use a property to specify which layout algorithm to use. These
+ * properties are just like other mark properties, and can be defined as
+ * constants or as functions. As with panels, the data property can be used to
+ * replicate layouts, and properties can be defined to in terms of layout data.
+ *
+ * @extends pv.Panel
+ */
+pv.Layout = function() {
+  pv.Panel.call(this);
+};
+
+pv.Layout.prototype = pv.extend(pv.Panel);
+
+/**
+ * @private Defines a local property with the specified name and cast. Note that
+ * although the property method is only defined locally, the cast function is
+ * global, which is necessary since properties are inherited!
+ *
+ * @param {string} name the property name.
+ * @param {function} [cast] the cast function for this property.
+ */
+pv.Layout.prototype.property = function(name, cast) {
+  if (!this.hasOwnProperty("properties")) {
+    this.properties = pv.extend(this.properties);
+  }
+  this.properties[name] = true;
+  this.propertyMethod(name, false, pv.Mark.cast[name] = cast);
+  return this;
+};
+/**
+ * Constructs a new, empty network layout. Layouts are not typically constructed
+ * directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents an abstract layout for network diagrams. This class
+ * provides the basic structure for both node-link diagrams (such as
+ * force-directed graph layout) and space-filling network diagrams (such as
+ * sunbursts and treemaps). Note that "network" here is a general term that
+ * includes hierarchical structures; a tree is represented using links from
+ * child to parent.
+ *
+ * <p>Network layouts require the graph data structure to be defined using two
+ * properties:<ul>
+ *
+ * <li><tt>nodes</tt> - an array of objects representing nodes. Objects in this
+ * array must conform to the {@link pv.Layout.Network.Node} interface; which is
+ * to say, be careful to avoid naming collisions with automatic attributes such
+ * as <tt>index</tt> and <tt>linkDegree</tt>. If the nodes property is defined
+ * as an array of primitives, such as numbers or strings, these primitives are
+ * automatically wrapped in an object; the resulting object's <tt>nodeValue</tt>
+ * attribute points to the original primitive value.
+ *
+ * <p><li><tt>links</tt> - an array of objects representing links. Objects in
+ * this array must conform to the {@link pv.Layout.Network.Link} interface; at a
+ * minimum, either <tt>source</tt> and <tt>target</tt> indexes or
+ * <tt>sourceNode</tt> and <tt>targetNode</tt> references must be set. Note that
+ * if the links property is defined after the nodes property, the links can be
+ * defined in terms of <tt>this.nodes()</tt>.
+ *
+ * </ul>
+ *
+ * <p>Three standard mark prototypes are provided:<ul>
+ *
+ * <li><tt>node</tt> - for rendering nodes; typically a {@link pv.Dot}. The node
+ * mark is added directly to the layout, with the data property defined via the
+ * layout's <tt>nodes</tt> property. Properties such as <tt>strokeStyle</tt> and
+ * <tt>fillStyle</tt> can be overridden to compute properties from node data
+ * dynamically.
+ *
+ * <p><li><tt>link</tt> - for rendering links; typically a {@link pv.Line}. The
+ * link mark is added to a child panel, whose data property is defined as
+ * layout's <tt>links</tt> property. The link's data property is then a
+ * two-element array of the source node and target node. Thus, poperties such as
+ * <tt>strokeStyle</tt> and <tt>fillStyle</tt> can be overridden to compute
+ * properties from either the node data (the first argument) or the link data
+ * (the second argument; the parent panel data) dynamically.
+ *
+ * <p><li><tt>label</tt> - for rendering node labels; typically a
+ * {@link pv.Label}. The label mark is added directly to the layout, with the
+ * data property defined via the layout's <tt>nodes</tt> property. Properties
+ * such as <tt>strokeStyle</tt> and <tt>fillStyle</tt> can be overridden to
+ * compute properties from node data dynamically.
+ *
+ * </ul>Note that some network implementations may not support all three
+ * standard mark prototypes; for example, space-filling hierarchical layouts
+ * typically do not use a <tt>link</tt> prototype, as the parent-child links are
+ * implied by the structure of the space-filling <tt>node</tt> marks.  Check the
+ * specific network layout for implementation details.
+ *
+ * <p>Network layout properties, including <tt>nodes</tt> and <tt>links</tt>,
+ * are typically cached rather than re-evaluated with every call to render. This
+ * is a performance optimization, as network layout algorithms can be
+ * expensive. If the network structure changes, call {@link #reset} to clear the
+ * cache before rendering. Note that although the network layout properties are
+ * cached, child mark properties, such as the marks used to render the nodes and
+ * links, <i>are not</i>. Therefore, non-structural changes to the network
+ * layout, such as changing the color of a mark on mouseover, do not need to
+ * reset the layout.
+ *
+ * @see pv.Layout.Hierarchy
+ * @see pv.Layout.Force
+ * @see pv.Layout.Matrix
+ * @see pv.Layout.Arc
+ * @see pv.Layout.Rollup
+ * @extends pv.Layout
+ */
+pv.Layout.Network = function() {
+  pv.Layout.call(this);
+  var that = this;
+
+  /* @private Version tracking to cache layout state, improving performance. */
+  this.$id = pv.id();
+
+  /**
+   * The node prototype. This prototype is intended to be used with a Dot mark
+   * in conjunction with the link prototype.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Network.prototype.node
+   */
+  (this.node = new pv.Mark()
+      .data(function() { return that.nodes(); })
+      .strokeStyle("#1f77b4")
+      .fillStyle("#fff")
+      .left(function(n) { return n.x; })
+      .top(function(n) { return n.y; })).parent = this;
+
+  /**
+   * The link prototype, which renders edges between source nodes and target
+   * nodes. This prototype is intended to be used with a Line mark in
+   * conjunction with the node prototype.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Network.prototype.link
+   */
+  this.link = new pv.Mark()
+      .extend(this.node)
+      .data(function(p) { return [p.sourceNode, p.targetNode]; })
+      .fillStyle(null)
+      .lineWidth(function(d, p) { return p.linkValue * 1.5; })
+      .strokeStyle("rgba(0,0,0,.2)");
+
+  this.link.add = function(type) {
+    return that.add(pv.Panel)
+        .data(function() { return that.links(); })
+      .add(type)
+        .extend(this);
+  };
+
+  /**
+   * The node label prototype, which renders the node name adjacent to the node.
+   * This prototype is provided as an alternative to using the anchor on the
+   * node mark; it is primarily intended to be used with radial node-link
+   * layouts, since it provides a convenient mechanism to set the text angle.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Network.prototype.label
+   */
+  (this.label = new pv.Mark()
+      .extend(this.node)
+      .textMargin(7)
+      .textBaseline("middle")
+      .text(function(n) { return n.nodeName || n.nodeValue; })
+      .textAngle(function(n) {
+          var a = n.midAngle;
+          return pv.Wedge.upright(a) ? a : (a + Math.PI);
+        })
+      .textAlign(function(n) {
+          return pv.Wedge.upright(n.midAngle) ? "left" : "right";
+        })).parent = this;
+};
+
+/**
+ * @class Represents a node in a network layout. There is no explicit
+ * constructor; this class merely serves to document the attributes that are
+ * used on nodes in network layouts. (Note that hierarchical nodes place
+ * additional requirements on node representation, vis {@link pv.Dom.Node}.)
+ *
+ * @see pv.Layout.Network
+ * @name pv.Layout.Network.Node
+ */
+
+/**
+ * The node index, zero-based. This attribute is populated automatically based
+ * on the index in the array returned by the <tt>nodes</tt> property.
+ *
+ * @type number
+ * @name pv.Layout.Network.Node.prototype.index
+ */
+
+/**
+ * The link degree; the sum of link values for all incoming and outgoing links.
+ * This attribute is populated automatically.
+ *
+ * @type number
+ * @name pv.Layout.Network.Node.prototype.linkDegree
+ */
+
+/**
+ * The node name; optional. If present, this attribute will be used to provide
+ * the text for node labels. If not present, the label text will fallback to the
+ * <tt>nodeValue</tt> attribute.
+ *
+ * @type string
+ * @name pv.Layout.Network.Node.prototype.nodeName
+ */
+
+/**
+ * The node value; optional. If present, and no <tt>nodeName</tt> attribute is
+ * present, the node value will be used as the label text. This attribute is
+ * also automatically populated if the nodes are specified as an array of
+ * primitives, such as strings or numbers.
+ *
+ * @type object
+ * @name pv.Layout.Network.Node.prototype.nodeValue
+ */
+
+/**
+ * @class Represents a link in a network layout. There is no explicit
+ * constructor; this class merely serves to document the attributes that are
+ * used on links in network layouts. For hierarchical layouts, this class is
+ * used to represent the parent-child links.
+ *
+ * @see pv.Layout.Network
+ * @name pv.Layout.Network.Link
+ */
+
+/**
+ * The link value, or weight; optional. If not specified (or not a number), the
+ * default value of 1 is used.
+ *
+ * @type number
+ * @name pv.Layout.Network.Link.prototype.linkValue
+ */
+
+/**
+ * The link's source node. If not set, this value will be derived from the
+ * <tt>source</tt> attribute index.
+ *
+ * @type pv.Layout.Network.Node
+ * @name pv.Layout.Network.Link.prototype.sourceNode
+ */
+
+/**
+ * The link's target node. If not set, this value will be derived from the
+ * <tt>target</tt> attribute index.
+ *
+ * @type pv.Layout.Network.Node
+ * @name pv.Layout.Network.Link.prototype.targetNode
+ */
+
+/**
+ * Alias for <tt>sourceNode</tt>, as expressed by the index of the source node.
+ * This attribute is not populated automatically, but may be used as a more
+ * convenient identification of the link's source, for example in a static JSON
+ * representation.
+ *
+ * @type number
+ * @name pv.Layout.Network.Link.prototype.source
+ */
+
+/**
+ * Alias for <tt>targetNode</tt>, as expressed by the index of the target node.
+ * This attribute is not populated automatically, but may be used as a more
+ * convenient identification of the link's target, for example in a static JSON
+ * representation.
+ *
+ * @type number
+ * @name pv.Layout.Network.Link.prototype.target
+ */
+
+/**
+ * Alias for <tt>linkValue</tt>. This attribute is not populated automatically,
+ * but may be used instead of the <tt>linkValue</tt> attribute when specifying
+ * links.
+ *
+ * @type number
+ * @name pv.Layout.Network.Link.prototype.value
+ */
+
+/** @private Transform nodes and links on cast. */
+pv.Layout.Network.prototype = pv.extend(pv.Layout)
+    .property("nodes", function(v) {
+        return v.map(function(d, i) {
+            if (typeof d != "object") d = {nodeValue: d};
+            d.index = i;
+            return d;
+          });
+      })
+    .property("links", function(v) {
+        return v.map(function(d) {
+            if (isNaN(d.linkValue)) d.linkValue = isNaN(d.value) ? 1 : d.value;
+            return d;
+          });
+      });
+
+/**
+ * Resets the cache, such that changes to layout property definitions will be
+ * visible on subsequent render. Unlike normal marks (and normal layouts),
+ * properties associated with network layouts are not automatically re-evaluated
+ * on render; the properties are cached, and any expensive layout algorithms are
+ * only run after the layout is explicitly reset.
+ *
+ * @returns {pv.Layout.Network} this.
+ */
+pv.Layout.Network.prototype.reset = function() {
+  this.$id = pv.id();
+  return this;
+};
+
+/** @private Skip evaluating properties if cached. */
+pv.Layout.Network.prototype.buildProperties = function(s, properties) {
+  if ((s.$id || 0) < this.$id) {
+    pv.Layout.prototype.buildProperties.call(this, s, properties);
+  }
+};
+
+/** @private Compute link degrees; map source and target indexes to nodes. */
+pv.Layout.Network.prototype.buildImplied = function(s) {
+  pv.Layout.prototype.buildImplied.call(this, s);
+  if (s.$id >= this.$id) return true;
+  s.$id = this.$id;
+  s.nodes.forEach(function(d) {
+      d.linkDegree = 0;
+    });
+  s.links.forEach(function(d) {
+      var v = d.linkValue;
+      (d.sourceNode || (d.sourceNode = s.nodes[d.source])).linkDegree += v;
+      (d.targetNode || (d.targetNode = s.nodes[d.target])).linkDegree += v;
+    });
+};
+/**
+ * Constructs a new, empty hierarchy layout. Layouts are not typically
+ * constructed directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Represents an abstract layout for hierarchy diagrams. This class is a
+ * specialization of {@link pv.Layout.Network}, providing the basic structure
+ * for both hierarchical node-link diagrams (such as Reingold-Tilford trees) and
+ * space-filling hierarchy diagrams (such as sunbursts and treemaps).
+ *
+ * <p>Unlike general network layouts, the <tt>links</tt> property need not be
+ * defined explicitly. Instead, the links are computed implicitly from the
+ * <tt>parentNode</tt> attribute of the node objects, as defined by the
+ * <tt>nodes</tt> property. This implementation is also available as
+ * {@link #links}, for reuse with non-hierarchical layouts; for example, to
+ * render a tree using force-directed layout.
+ *
+ * <p>Correspondingly, the <tt>nodes</tt> property is represented as a union of
+ * {@link pv.Layout.Network.Node} and {@link pv.Dom.Node}. To construct a node
+ * hierarchy from a simple JSON map, use the {@link pv.Dom} operator; this
+ * operator also provides an easy way to sort nodes before passing them to the
+ * layout.
+ *
+ * <p>For more details on how to use this layout, see
+ * {@link pv.Layout.Network}.
+ *
+ * @see pv.Layout.Cluster
+ * @see pv.Layout.Partition
+ * @see pv.Layout.Tree
+ * @see pv.Layout.Treemap
+ * @see pv.Layout.Indent
+ * @see pv.Layout.Pack
+ * @extends pv.Layout.Network
+ */
+pv.Layout.Hierarchy = function() {
+  pv.Layout.Network.call(this);
+  this.link.strokeStyle("#ccc");
+};
+
+pv.Layout.Hierarchy.prototype = pv.extend(pv.Layout.Network);
+
+/** @private Compute the implied links. (Links are null by default.) */
+pv.Layout.Hierarchy.prototype.buildImplied = function(s) {
+  if (!s.links) s.links = pv.Layout.Hierarchy.links.call(this);
+  pv.Layout.Network.prototype.buildImplied.call(this, s);
+};
+
+/** The implied links; computes links using the <tt>parentNode</tt> attribute. */
+pv.Layout.Hierarchy.links = function() {
+  return this.nodes()
+      .filter(function(n) { return n.parentNode; })
+      .map(function(n) {
+          return {
+              sourceNode: n,
+              targetNode: n.parentNode,
+              linkValue: 1
+            };
+      });
+};
+
+/** @private Provides standard node-link layout based on breadth & depth. */
+pv.Layout.Hierarchy.NodeLink = {
+
+  /** @private */
+  buildImplied: function(s) {
+    var nodes = s.nodes,
+        orient = s.orient,
+        horizontal = /^(top|bottom)$/.test(orient),
+        w = s.width,
+        h = s.height;
+
+    /* Compute default inner and outer radius. */
+    if (orient == "radial") {
+      var ir = s.innerRadius, or = s.outerRadius;
+      if (ir == null) ir = 0;
+      if (or == null) or = Math.min(w, h) / 2;
+    }
+
+    /** @private Returns the radius of the given node. */
+    function radius(n) {
+      return n.parentNode ? (n.depth * (or - ir) + ir) : 0;
+    }
+
+    /** @private Returns the angle of the given node. */
+    function midAngle(n) {
+      return (n.parentNode ? (n.breadth - .25) * 2 * Math.PI : 0);
+    }
+
+    /** @private */
+    function x(n) {
+      switch (orient) {
+        case "left": return n.depth * w;
+        case "right": return w - n.depth * w;
+        case "top": return n.breadth * w;
+        case "bottom": return w - n.breadth * w;
+        case "radial": return w / 2 + radius(n) * Math.cos(n.midAngle);
+      }
+    }
+
+    /** @private */
+    function y(n) {
+      switch (orient) {
+        case "left": return n.breadth * h;
+        case "right": return h - n.breadth * h;
+        case "top": return n.depth * h;
+        case "bottom": return h - n.depth * h;
+        case "radial": return h / 2 + radius(n) * Math.sin(n.midAngle);
+      }
+    }
+
+    for (var i = 0; i < nodes.length; i++) {
+      var n = nodes[i];
+      n.midAngle = orient == "radial" ? midAngle(n)
+          : horizontal ? Math.PI / 2 : 0;
+      n.x = x(n);
+      n.y = y(n);
+      if (n.firstChild) n.midAngle += Math.PI;
+    }
+  }
+};
+
+/** @private Provides standard space-filling layout based on breadth & depth. */
+pv.Layout.Hierarchy.Fill = {
+
+  /** @private */
+  constructor: function() {
+    this.node
+        .strokeStyle("#fff")
+        .fillStyle("#ccc")
+        .width(function(n) { return n.dx; })
+        .height(function(n) { return n.dy; })
+        .innerRadius(function(n) { return n.innerRadius; })
+        .outerRadius(function(n) { return n.outerRadius; })
+        .startAngle(function(n) { return n.startAngle; })
+        .angle(function(n) { return n.angle; });
+
+    this.label
+        .textAlign("center")
+        .left(function(n) { return n.x + (n.dx / 2); })
+        .top(function(n) { return n.y + (n.dy / 2); });
+
+    /* Hide unsupported link. */
+    delete this.link;
+  },
+
+  /** @private */
+  buildImplied: function(s) {
+    var nodes = s.nodes,
+        orient = s.orient,
+        horizontal = /^(top|bottom)$/.test(orient),
+        w = s.width,
+        h = s.height,
+        depth = -nodes[0].minDepth;
+
+    /* Compute default inner and outer radius. */
+    if (orient == "radial") {
+      var ir = s.innerRadius, or = s.outerRadius;
+      if (ir == null) ir = 0;
+      if (ir) depth *= 2; // use full depth step for root
+      if (or == null) or = Math.min(w, h) / 2;
+    }
+
+    /** @private Scales the specified depth for a space-filling layout. */
+    function scale(d, depth) {
+      return (d + depth) / (1 + depth);
+    }
+
+    /** @private */
+    function x(n) {
+      switch (orient) {
+        case "left": return scale(n.minDepth, depth) * w;
+        case "right": return (1 - scale(n.maxDepth, depth)) * w;
+        case "top": return n.minBreadth * w;
+        case "bottom": return (1 - n.maxBreadth) * w;
+        case "radial": return w / 2;
+      }
+    }
+
+    /** @private */
+    function y(n) {
+      switch (orient) {
+        case "left": return n.minBreadth * h;
+        case "right": return (1 - n.maxBreadth) * h;
+        case "top": return scale(n.minDepth, depth) * h;
+        case "bottom": return (1 - scale(n.maxDepth, depth)) * h;
+        case "radial": return h / 2;
+      }
+    }
+
+    /** @private */
+    function dx(n) {
+      switch (orient) {
+        case "left":
+        case "right": return (n.maxDepth - n.minDepth) / (1 + depth) * w;
+        case "top":
+        case "bottom": return (n.maxBreadth - n.minBreadth) * w;
+        case "radial": return n.parentNode ? (n.innerRadius + n.outerRadius) * Math.cos(n.midAngle) : 0;
+      }
+    }
+
+    /** @private */
+    function dy(n) {
+      switch (orient) {
+        case "left":
+        case "right": return (n.maxBreadth - n.minBreadth) * h;
+        case "top":
+        case "bottom": return (n.maxDepth - n.minDepth) / (1 + depth) * h;
+        case "radial": return n.parentNode ? (n.innerRadius + n.outerRadius) * Math.sin(n.midAngle) : 0;
+      }
+    }
+
+    /** @private */
+    function innerRadius(n) {
+      return Math.max(0, scale(n.minDepth, depth / 2)) * (or - ir) + ir;
+    }
+
+    /** @private */
+    function outerRadius(n) {
+      return scale(n.maxDepth, depth / 2) * (or - ir) + ir;
+    }
+
+    /** @private */
+    function startAngle(n) {
+      return (n.parentNode ? n.minBreadth - .25 : 0) * 2 * Math.PI;
+    }
+
+    /** @private */
+    function angle(n) {
+      return (n.parentNode ? n.maxBreadth - n.minBreadth : 1) * 2 * Math.PI;
+    }
+
+    for (var i = 0; i < nodes.length; i++) {
+      var n = nodes[i];
+      n.x = x(n);
+      n.y = y(n);
+      if (orient == "radial") {
+        n.innerRadius = innerRadius(n);
+        n.outerRadius = outerRadius(n);
+        n.startAngle = startAngle(n);
+        n.angle = angle(n);
+        n.midAngle = n.startAngle + n.angle / 2;
+      } else {
+        n.midAngle = horizontal ? -Math.PI / 2 : 0;
+      }
+      n.dx = dx(n);
+      n.dy = dy(n);
+    }
+  }
+};
+/**
+ * Constructs a new, empty grid layout. Layouts are not typically constructed
+ * directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a grid layout with regularly-sized rows and columns. The
+ * number of rows and columns are determined from their respective
+ * properties. For example, the 2&times;3 array:
+ *
+ * <pre>1 2 3
+ * 4 5 6</pre>
+ *
+ * can be represented using the <tt>rows</tt> property as:
+ *
+ * <pre>[[1, 2, 3], [4, 5, 6]]</pre>
+ *
+ * If your data is in column-major order, you can equivalently use the
+ * <tt>columns</tt> property. If the <tt>rows</tt> property is an array, it
+ * takes priority over the <tt>columns</tt> property. The data is implicitly
+ * transposed, as if the {@link pv.transpose} operator were applied.
+ *
+ * <p>This layout exports a single <tt>cell</tt> mark prototype, which is
+ * intended to be used with a bar, panel, layout, or subclass thereof. The data
+ * property of the cell prototype is defined as the elements in the array. For
+ * example, if the array is a two-dimensional array of values in the range
+ * [0,1], a simple heatmap can be generated as:
+ *
+ * <pre>vis.add(pv.Layout.Grid)
+ *     .rows(arrays)
+ *   .cell.add(pv.Bar)
+ *     .fillStyle(pv.ramp("white", "black"))</pre>
+ *
+ * The grid subdivides the full width and height of the parent panel into equal
+ * rectangles. Note, however, that for large, interactive, or animated heatmaps,
+ * you may see significantly better performance through dynamic {@link pv.Image}
+ * generation.
+ *
+ * <p>For irregular grids using value-based spatial partitioning, see {@link
+ * pv.Layout.Treemap}.
+ *
+ * @extends pv.Layout
+ */
+pv.Layout.Grid = function() {
+  pv.Layout.call(this);
+  var that = this;
+
+  /**
+   * The cell prototype. This prototype is intended to be used with a bar,
+   * panel, or layout (or subclass thereof) to render the grid cells.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Grid.prototype.cell
+   */
+  (this.cell = new pv.Mark()
+      .data(function() {
+          return that.scene[that.index].$grid;
+        })
+      .width(function() {
+          return that.width() / that.cols();
+        })
+      .height(function() {
+          return that.height() / that.rows();
+        })
+      .left(function() {
+          return this.width() * (this.index % that.cols());
+        })
+      .top(function() {
+          return this.height() * Math.floor(this.index / that.cols());
+        })).parent = this;
+};
+
+pv.Layout.Grid.prototype = pv.extend(pv.Layout)
+    .property("rows")
+    .property("cols");
+
+/**
+ * Default properties for grid layouts. By default, there is one row and one
+ * column, and the data is the propagated to the child cell.
+ *
+ * @type pv.Layout.Grid
+ */
+pv.Layout.Grid.prototype.defaults = new pv.Layout.Grid()
+    .extend(pv.Layout.prototype.defaults)
+    .rows(1)
+    .cols(1);
+
+/** @private */
+pv.Layout.Grid.prototype.buildImplied = function(s) {
+  pv.Layout.prototype.buildImplied.call(this, s);
+  var r = s.rows, c = s.cols;
+  if (typeof c == "object") r = pv.transpose(c);
+  if (typeof r == "object") {
+    s.$grid = pv.blend(r);
+    s.rows = r.length;
+    s.cols = r[0] ? r[0].length : 0;
+  } else {
+    s.$grid = pv.repeat([s.data], r * c);
+  }
+};
+
+/**
+ * The number of rows. This property can also be specified as the data in
+ * row-major order; in this case, the rows property is implicitly set to the
+ * length of the array, and the cols property is set to the length of the first
+ * element in the array.
+ *
+ * @type number
+ * @name pv.Layout.Grid.prototype.rows
+ */
+
+/**
+ * The number of columns. This property can also be specified as the data in
+ * column-major order; in this case, the cols property is implicitly set to the
+ * length of the array, and the rows property is set to the length of the first
+ * element in the array.
+ *
+ * @type number
+ * @name pv.Layout.Grid.prototype.cols
+ */
+/**
+ * Constructs a new, empty stack layout. Layouts are not typically constructed
+ * directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a layout for stacked visualizations, ranging from simple
+ * stacked bar charts to more elaborate "streamgraphs" composed of stacked
+ * areas. Stack layouts uses length as a visual encoding, as opposed to
+ * position, as the layers do not share an aligned axis.
+ *
+ * <p>Marks can be stacked vertically or horizontally. For example,
+ *
+ * <pre>vis.add(pv.Layout.Stack)
+ *     .layers([[1, 1.2, 1.7, 1.5, 1.7],
+ *              [.5, 1, .8, 1.1, 1.3],
+ *              [.2, .5, .8, .9, 1]])
+ *     .x(function() this.index * 35)
+ *     .y(function(d) d * 40)
+ *   .layer.add(pv.Area);</pre>
+ *
+ * specifies a vertically-stacked area chart, using the default "bottom-left"
+ * orientation with "zero" offset. This visualization can be easily changed into
+ * a streamgraph using the "wiggle" offset, which attempts to minimize change in
+ * slope weighted by layer thickness. See the {@link #offset} property for more
+ * supported streamgraph algorithms.
+ *
+ * <p>In the simplest case, the layer data can be specified as a two-dimensional
+ * array of numbers. The <tt>x</tt> and <tt>y</tt> psuedo-properties are used to
+ * define the thickness of each layer at the given position, respectively; in
+ * the above example of the "bottom-left" orientation, the <tt>x</tt> and
+ * <tt>y</tt> psuedo-properties are equivalent to the <tt>left</tt> and
+ * <tt>height</tt> properties that you might use if you implemented a stacked
+ * area by hand.
+ *
+ * <p>The advantage of using the stack layout is that the baseline, i.e., the
+ * <tt>bottom</tt> property is computed automatically using the specified offset
+ * algorithm. In addition, the order of layers can be computed using a built-in
+ * algorithm via the <tt>order</tt> property.
+ *
+ * <p>With the exception of the "expand" <tt>offset</tt>, the stack layout does
+ * not perform any automatic scaling of data; the values returned from
+ * <tt>x</tt> and <tt>y</tt> specify pixel sizes. To simplify scaling math, use
+ * this layout in conjunction with {@link pv.Scale.linear} or similar.
+ *
+ * <p>In other cases, the <tt>values</tt> psuedo-property can be used to define
+ * the data more flexibly. As with a typical panel &amp; area, the
+ * <tt>layers</tt> property corresponds to the data in the enclosing panel,
+ * while the <tt>values</tt> psuedo-property corresponds to the data for the
+ * area within the panel. For example, given an array of data values:
+ *
+ * <pre>var crimea = [
+ *  { date: "4/1854", wounds: 0, other: 110, disease: 110 },
+ *  { date: "5/1854", wounds: 0, other: 95, disease: 105 },
+ *  { date: "6/1854", wounds: 0, other: 40, disease: 95 },
+ *  ...</pre>
+ *
+ * and a corresponding array of series names:
+ *
+ * <pre>var causes = ["wounds", "other", "disease"];</pre>
+ *
+ * Separate layers can be defined for each cause like so:
+ *
+ * <pre>vis.add(pv.Layout.Stack)
+ *     .layers(causes)
+ *     .values(crimea)
+ *     .x(function(d) x(d.date))
+ *     .y(function(d, p) y(d[p]))
+ *   .layer.add(pv.Area)
+ *     ...</pre>
+ *
+ * As with the panel &amp; area case, the datum that is passed to the
+ * psuedo-properties <tt>x</tt> and <tt>y</tt> are the values (an element in
+ * <tt>crimea</tt>); the second argument is the layer data (a string in
+ * <tt>causes</tt>). Additional arguments specify the data of enclosing panels,
+ * if any.
+ *
+ * @extends pv.Layout
+ */
+pv.Layout.Stack = function() {
+  pv.Layout.call(this);
+  var that = this,
+      /** @ignore */ none = function() { return null; },
+      prop = {t: none, l: none, r: none, b: none, w: none, h: none},
+      values,
+      buildImplied = that.buildImplied;
+
+  /** @private Proxy the given property on the layer. */
+  function proxy(name) {
+    return function() {
+        return prop[name](this.parent.index, this.index);
+      };
+  }
+
+  /** @private Compute the layout! */
+  this.buildImplied = function(s) {
+    buildImplied.call(this, s);
+
+    var data = s.layers,
+        n = data.length,
+        m,
+        orient = s.orient,
+        horizontal = /^(top|bottom)\b/.test(orient),
+        h = this.parent[horizontal ? "height" : "width"](),
+        x = [],
+        y = [],
+        dy = [];
+
+    /*
+     * Iterate over the data, evaluating the values, x and y functions. The
+     * context in which the x and y psuedo-properties are evaluated is a
+     * pseudo-mark that is a grandchild of this layout.
+     */
+    var stack = pv.Mark.stack, o = {parent: {parent: this}};
+    stack.unshift(null);
+    values = [];
+    for (var i = 0; i < n; i++) {
+      dy[i] = [];
+      y[i] = [];
+      o.parent.index = i;
+      stack[0] = data[i];
+      values[i] = this.$values.apply(o.parent, stack);
+      if (!i) m = values[i].length;
+      stack.unshift(null);
+      for (var j = 0; j < m; j++) {
+        stack[0] = values[i][j];
+        o.index = j;
+        if (!i) x[j] = this.$x.apply(o, stack);
+        dy[i][j] = this.$y.apply(o, stack);
+      }
+      stack.shift();
+    }
+    stack.shift();
+
+    /* order */
+    var index;
+    switch (s.order) {
+      case "inside-out": {
+        var max = dy.map(function(v) { return pv.max.index(v); }),
+            map = pv.range(n).sort(function(a, b) { return max[a] - max[b]; }),
+            sums = dy.map(function(v) { return pv.sum(v); }),
+            top = 0,
+            bottom = 0,
+            tops = [],
+            bottoms = [];
+        for (var i = 0; i < n; i++) {
+          var j = map[i];
+          if (top < bottom) {
+            top += sums[j];
+            tops.push(j);
+          } else {
+            bottom += sums[j];
+            bottoms.push(j);
+          }
+        }
+        index = bottoms.reverse().concat(tops);
+        break;
+      }
+      case "reverse": index = pv.range(n - 1, -1, -1); break;
+      default: index = pv.range(n); break;
+    }
+
+    /* offset */
+    switch (s.offset) {
+      case "silohouette": {
+        for (var j = 0; j < m; j++) {
+          var o = 0;
+          for (var i = 0; i < n; i++) o += dy[i][j];
+          y[index[0]][j] = (h - o) / 2;
+        }
+        break;
+      }
+      case "wiggle": {
+        var o = 0;
+        for (var i = 0; i < n; i++) o += dy[i][0];
+        y[index[0]][0] = o = (h - o) / 2;
+        for (var j = 1; j < m; j++) {
+          var s1 = 0, s2 = 0, dx = x[j] - x[j - 1];
+          for (var i = 0; i < n; i++) s1 += dy[i][j];
+          for (var i = 0; i < n; i++) {
+            var s3 = (dy[index[i]][j] - dy[index[i]][j - 1]) / (2 * dx);
+            for (var k = 0; k < i; k++) {
+              s3 += (dy[index[k]][j] - dy[index[k]][j - 1]) / dx;
+            }
+            s2 += s3 * dy[index[i]][j];
+          }
+          y[index[0]][j] = o -= s1 ? s2 / s1 * dx : 0;
+        }
+        break;
+      }
+      case "expand": {
+        for (var j = 0; j < m; j++) {
+          y[index[0]][j] = 0;
+          var k = 0;
+          for (var i = 0; i < n; i++) k += dy[i][j];
+          if (k) {
+            k = h / k;
+            for (var i = 0; i < n; i++) dy[i][j] *= k;
+          } else {
+            k = h / n;
+            for (var i = 0; i < n; i++) dy[i][j] = k;
+          }
+        }
+        break;
+      }
+      default: {
+        for (var j = 0; j < m; j++) y[index[0]][j] = 0;
+        break;
+      }
+    }
+
+    /* Propagate the offset to the other series. */
+    for (var j = 0; j < m; j++) {
+      var o = y[index[0]][j];
+      for (var i = 1; i < n; i++) {
+        o += dy[index[i - 1]][j];
+        y[index[i]][j] = o;
+      }
+    }
+
+    /* Find the property definitions for dynamic substitution. */
+    var i = orient.indexOf("-"),
+        pdy = horizontal ? "h" : "w",
+        px = i < 0 ? (horizontal ? "l" : "b") : orient.charAt(i + 1),
+        py = orient.charAt(0);
+    for (var p in prop) prop[p] = none;
+    prop[px] = function(i, j) { return x[j]; };
+    prop[py] = function(i, j) { return y[i][j]; };
+    prop[pdy] = function(i, j) { return dy[i][j]; };
+  };
+
+  /**
+   * The layer prototype. This prototype is intended to be used with an area,
+   * bar or panel mark (or subclass thereof). Other mark types may be possible,
+   * though note that the stack layout is not currently designed to support
+   * radial stacked visualizations using wedges.
+   *
+   * <p>The layer is not a direct child of the stack layout; a hidden panel is
+   * used to replicate layers.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Stack.prototype.layer
+   */
+  this.layer = new pv.Mark()
+      .data(function() { return values[this.parent.index]; })
+      .top(proxy("t"))
+      .left(proxy("l"))
+      .right(proxy("r"))
+      .bottom(proxy("b"))
+      .width(proxy("w"))
+      .height(proxy("h"));
+
+  this.layer.add = function(type) {
+    return that.add(pv.Panel)
+        .data(function() { return that.layers(); })
+      .add(type)
+        .extend(this);
+  };
+};
+
+pv.Layout.Stack.prototype = pv.extend(pv.Layout)
+    .property("orient", String)
+    .property("offset", String)
+    .property("order", String)
+    .property("layers");
+
+/**
+ * Default properties for stack layouts. The default orientation is
+ * "bottom-left", the default offset is "zero", and the default layers is
+ * <tt>[[]]</tt>.
+ *
+ * @type pv.Layout.Stack
+ */
+pv.Layout.Stack.prototype.defaults = new pv.Layout.Stack()
+    .extend(pv.Layout.prototype.defaults)
+    .orient("bottom-left")
+    .offset("zero")
+    .layers([[]]);
+
+/** @private */
+pv.Layout.Stack.prototype.$x
+    = /** @private */ pv.Layout.Stack.prototype.$y
+    = function() { return 0; };
+
+/**
+ * The x psuedo-property; determines the position of the value within the layer.
+ * This typically corresponds to the independent variable. For example, with the
+ * default "bottom-left" orientation, this function defines the "left" property.
+ *
+ * @param {function} f the x function.
+ * @returns {pv.Layout.Stack} this.
+ */
+pv.Layout.Stack.prototype.x = function(f) {
+  /** @private */ this.$x = pv.functor(f);
+  return this;
+};
+
+/**
+ * The y psuedo-property; determines the thickness of the layer at the given
+ * value.  This typically corresponds to the dependent variable. For example,
+ * with the default "bottom-left" orientation, this function defines the
+ * "height" property.
+ *
+ * @param {function} f the y function.
+ * @returns {pv.Layout.Stack} this.
+ */
+pv.Layout.Stack.prototype.y = function(f) {
+  /** @private */ this.$y = pv.functor(f);
+  return this;
+};
+
+/** @private The default value function; identity. */
+pv.Layout.Stack.prototype.$values = pv.identity;
+
+/**
+ * The values function; determines the values for a given layer. The default
+ * value is the identity function, which assumes that the layers property is
+ * specified as a two-dimensional (i.e., nested) array.
+ *
+ * @param {function} f the values function.
+ * @returns {pv.Layout.Stack} this.
+ */
+pv.Layout.Stack.prototype.values = function(f) {
+  this.$values = pv.functor(f);
+  return this;
+};
+
+/**
+ * The layer data in row-major order. The value of this property is typically a
+ * two-dimensional (i.e., nested) array, but any array can be used, provided the
+ * values psuedo-property is defined accordingly.
+ *
+ * @type array[]
+ * @name pv.Layout.Stack.prototype.layers
+ */
+
+/**
+ * The layer orientation. The following values are supported:<ul>
+ *
+ * <li>bottom-left == bottom
+ * <li>bottom-right
+ * <li>top-left == top
+ * <li>top-right
+ * <li>left-top
+ * <li>left-bottom == left
+ * <li>right-top
+ * <li>right-bottom == right
+ *
+ * </ul>. The default value is "bottom-left", which means that the layers will
+ * be built from the bottom-up, and the values within layers will be laid out
+ * from left-to-right.
+ *
+ * <p>Note that with non-zero baselines, some orientations may give similar
+ * results. For example, offset("silohouette") centers the layers, resulting in
+ * a streamgraph. Thus, the orientations "bottom-left" and "top-left" will
+ * produce similar results, differing only in the layer order.
+ *
+ * @type string
+ * @name pv.Layout.Stack.prototype.orient
+ */
+
+/**
+ * The layer order. The following values are supported:<ul>
+ *
+ * <li><i>null</i> - use given layer order.
+ * <li>inside-out - sort by maximum value, with balanced order.
+ * <li>reverse - use reverse of given layer order.
+ *
+ * </ul>For details on the inside-out order algorithm, refer to "Stacked Graphs
+ * -- Geometry &amp; Aesthetics" by L. Byron and M. Wattenberg, IEEE TVCG
+ * November/December 2008.
+ *
+ * @type string
+ * @name pv.Layout.Stack.prototype.order
+ */
+
+/**
+ * The layer offset; the y-position of the bottom of the lowest layer. The
+ * following values are supported:<ul>
+ *
+ * <li>zero - use a zero baseline, i.e., the y-axis.
+ * <li>silohouette - center the stream, i.e., ThemeRiver.
+ * <li>wiggle - minimize weighted change in slope.
+ * <li>expand - expand layers to fill the enclosing layout dimensions.
+ *
+ * </ul>For details on these offset algorithms, refer to "Stacked Graphs --
+ * Geometry &amp; Aesthetics" by L. Byron and M. Wattenberg, IEEE TVCG
+ * November/December 2008.
+ *
+ * @type string
+ * @name pv.Layout.Stack.prototype.offset
+ */
+/**
+ * Constructs a new, empty treemap layout. Layouts are not typically
+ * constructed directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a space-filling rectangular layout, with the hierarchy
+ * represented via containment. Treemaps represent nodes as boxes, with child
+ * nodes placed within parent boxes. The size of each box is proportional to the
+ * size of the node in the tree. This particular algorithm is taken from Bruls,
+ * D.M., C. Huizing, and J.J. van Wijk, <a
+ * href="http://www.win.tue.nl/~vanwijk/stm.pdf">"Squarified Treemaps"</a> in
+ * <i>Data Visualization 2000, Proceedings of the Joint Eurographics and IEEE
+ * TCVG Sumposium on Visualization</i>, 2000, pp. 33-42.
+ *
+ * <p>The meaning of the exported mark prototypes changes slightly in the
+ * space-filling implementation:<ul>
+ *
+ * <li><tt>node</tt> - for rendering nodes; typically a {@link pv.Bar}. The node
+ * data is populated with <tt>dx</tt> and <tt>dy</tt> attributes, in addition to
+ * the standard <tt>x</tt> and <tt>y</tt> position attributes.
+ *
+ * <p><li><tt>leaf</tt> - for rendering leaf nodes only, with no fill or stroke
+ * style by default; typically a {@link pv.Panel} or another layout!
+ *
+ * <p><li><tt>link</tt> - unsupported; undefined. Links are encoded implicitly
+ * in the arrangement of the space-filling nodes.
+ *
+ * <p><li><tt>label</tt> - for rendering node labels; typically a
+ * {@link pv.Label}.
+ *
+ * </ul>For more details on how to use this layout, see
+ * {@link pv.Layout.Hierarchy}.
+ *
+ * @extends pv.Layout.Hierarchy
+ */
+pv.Layout.Treemap = function() {
+  pv.Layout.Hierarchy.call(this);
+
+  this.node
+      .strokeStyle("#fff")
+      .fillStyle("rgba(31, 119, 180, .25)")
+      .width(function(n) { return n.dx; })
+      .height(function(n) { return n.dy; });
+
+  this.label
+      .visible(function(n) { return !n.firstChild; })
+      .left(function(n) { return n.x + (n.dx / 2); })
+      .top(function(n) { return n.y + (n.dy / 2); })
+      .textAlign("center")
+      .textAngle(function(n) { return n.dx > n.dy ? 0 : -Math.PI / 2; });
+
+  (this.leaf = new pv.Mark()
+      .extend(this.node)
+      .fillStyle(null)
+      .strokeStyle(null)
+      .visible(function(n) { return !n.firstChild; })).parent = this;
+
+  /* Hide unsupported link. */
+  delete this.link;
+};
+
+pv.Layout.Treemap.prototype = pv.extend(pv.Layout.Hierarchy)
+    .property("round", Boolean)
+    .property("paddingLeft", Number)
+    .property("paddingRight", Number)
+    .property("paddingTop", Number)
+    .property("paddingBottom", Number)
+    .property("mode", String)
+    .property("order", String);
+
+/**
+ * Default propertiess for treemap layouts. The default mode is "squarify" and
+ * the default order is "ascending".
+ *
+ * @type pv.Layout.Treemap
+ */
+pv.Layout.Treemap.prototype.defaults = new pv.Layout.Treemap()
+    .extend(pv.Layout.Hierarchy.prototype.defaults)
+    .mode("squarify") // squarify, slice-and-dice, slice, dice
+    .order("ascending"); // ascending, descending, reverse, null
+
+/**
+ * Whether node sizes should be rounded to integer values. This has a similar
+ * effect to setting <tt>antialias(false)</tt> for node values, but allows the
+ * treemap algorithm to accumulate error related to pixel rounding.
+ *
+ * @type boolean
+ * @name pv.Layout.Treemap.prototype.round
+ */
+
+/**
+ * The left inset between parent add child in pixels. Defaults to 0.
+ *
+ * @type number
+ * @name pv.Layout.Treemap.prototype.paddingLeft
+ * @see #padding
+ */
+
+/**
+ * The right inset between parent add child in pixels. Defaults to 0.
+ *
+ * @type number
+ * @name pv.Layout.Treemap.prototype.paddingRight
+ * @see #padding
+ */
+
+/**
+ * The top inset between parent and child in pixels. Defaults to 0.
+ *
+ * @type number
+ * @name pv.Layout.Treemap.prototype.paddingTop
+ * @see #padding
+ */
+
+/**
+ * The bottom inset between parent and child in pixels. Defaults to 0.
+ *
+ * @type number
+ * @name pv.Layout.Treemap.prototype.paddingBottom
+ * @see #padding
+ */
+
+/**
+ * The treemap algorithm. The default value is "squarify". The "slice-and-dice"
+ * algorithm may also be used, which alternates between horizontal and vertical
+ * slices for different depths. In addition, the "slice" and "dice" algorithms
+ * may be specified explicitly to control whether horizontal or vertical slices
+ * are used, which may be useful for nested treemap layouts.
+ *
+ * @type string
+ * @name pv.Layout.Treemap.prototype.mode
+ * @see <a
+ * href="ftp://ftp.cs.umd.edu/pub/hcil/Reports-Abstracts-Bibliography/2001-06html/2001-06.pdf"
+ * >"Ordered Treemap Layouts"</a> by B. Shneiderman &amp; M. Wattenberg, IEEE
+ * InfoVis 2001.
+ */
+
+/**
+ * The sibling node order. A <tt>null</tt> value means to use the sibling order
+ * specified by the nodes property as-is; "reverse" will reverse the given
+ * order. The default value "ascending" will sort siblings in ascending order of
+ * size, while "descending" will do the reverse. For sorting based on data
+ * attributes other than size, use the default <tt>null</tt> for the order
+ * property, and sort the nodes beforehand using the {@link pv.Dom} operator.
+ *
+ * @type string
+ * @name pv.Layout.Treemap.prototype.order
+ */
+
+/**
+ * Alias for setting the left, right, top and bottom padding properties
+ * simultaneously.
+ *
+ * @see #paddingLeft
+ * @see #paddingRight
+ * @see #paddingTop
+ * @see #paddingBottom
+ * @returns {pv.Layout.Treemap} this.
+ */
+pv.Layout.Treemap.prototype.padding = function(n) {
+  return this.paddingLeft(n).paddingRight(n).paddingTop(n).paddingBottom(n);
+};
+
+/** @private The default size function. */
+pv.Layout.Treemap.prototype.$size = function(d) {
+  return Number(d.nodeValue);
+};
+
+/**
+ * Specifies the sizing function. By default, the size function uses the
+ * <tt>nodeValue</tt> attribute of nodes as a numeric value: <tt>function(d)
+ * Number(d.nodeValue)</tt>.
+ *
+ * <p>The sizing function is invoked for each leaf node in the tree, per the
+ * <tt>nodes</tt> property. For example, if the tree data structure represents a
+ * file system, with files as leaf nodes, and each file has a <tt>bytes</tt>
+ * attribute, you can specify a size function as:
+ *
+ * <pre>    .size(function(d) d.bytes)</pre>
+ *
+ * @param {function} f the new sizing function.
+ * @returns {pv.Layout.Treemap} this.
+ */
+pv.Layout.Treemap.prototype.size = function(f) {
+  this.$size = pv.functor(f);
+  return this;
+};
+
+/** @private */
+pv.Layout.Treemap.prototype.buildImplied = function(s) {
+  if (pv.Layout.Hierarchy.prototype.buildImplied.call(this, s)) return;
+
+  var that = this,
+      nodes = s.nodes,
+      root = nodes[0],
+      stack = pv.Mark.stack,
+      left = s.paddingLeft,
+      right = s.paddingRight,
+      top = s.paddingTop,
+      bottom = s.paddingBottom,
+      /** @ignore */ size = function(n) { return n.size; },
+      round = s.round ? Math.round : Number,
+      mode = s.mode;
+
+  /** @private */
+  function slice(row, sum, horizontal, x, y, w, h) {
+    for (var i = 0, d = 0; i < row.length; i++) {
+      var n = row[i];
+      if (horizontal) {
+        n.x = x + d;
+        n.y = y;
+        d += n.dx = round(w * n.size / sum);
+        n.dy = h;
+      } else {
+        n.x = x;
+        n.y = y + d;
+        n.dx = w;
+        d += n.dy = round(h * n.size / sum);
+      }
+    }
+    if (n) { // correct on-axis rounding error
+      if (horizontal) {
+        n.dx += w - d;
+      } else {
+        n.dy += h - d;
+      }
+    }
+  }
+
+  /** @private */
+  function ratio(row, l) {
+    var rmax = -Infinity, rmin = Infinity, s = 0;
+    for (var i = 0; i < row.length; i++) {
+      var r = row[i].size;
+      if (r < rmin) rmin = r;
+      if (r > rmax) rmax = r;
+      s += r;
+    }
+    s = s * s;
+    l = l * l;
+    return Math.max(l * rmax / s, s / (l * rmin));
+  }
+
+  /** @private */
+  function layout(n, i) {
+    var x = n.x + left,
+        y = n.y + top,
+        w = n.dx - left - right,
+        h = n.dy - top - bottom;
+
+    /* Assume squarify by default. */
+    if (mode != "squarify") {
+      slice(n.childNodes, n.size,
+          mode == "slice" ? true
+          : mode == "dice" ? false
+          : i & 1, x, y, w, h);
+      return;
+    }
+
+    var row = [],
+        mink = Infinity,
+        l = Math.min(w, h),
+        k = w * h / n.size;
+
+    /* Abort if the size is nonpositive. */
+    if (n.size <= 0) return;
+
+    /* Scale the sizes to fill the current subregion. */
+    n.visitBefore(function(n) { n.size *= k; });
+
+    /** @private Position the specified nodes along one dimension. */
+    function position(row) {
+      var horizontal = w == l,
+          sum = pv.sum(row, size),
+          r = l ? round(sum / l) : 0;
+      slice(row, sum, horizontal, x, y, horizontal ? w : r, horizontal ? r : h);
+      if (horizontal) {
+        y += r;
+        h -= r;
+      } else {
+        x += r;
+        w -= r;
+      }
+      l = Math.min(w, h);
+      return horizontal;
+    }
+
+    var children = n.childNodes.slice(); // copy
+    while (children.length) {
+      var child = children[children.length - 1];
+      if (!child.size) {
+        children.pop();
+        continue;
+      }
+      row.push(child);
+
+      var k = ratio(row, l);
+      if (k <= mink) {
+        children.pop();
+        mink = k;
+      } else {
+        row.pop();
+        position(row);
+        row.length = 0;
+        mink = Infinity;
+      }
+    }
+
+    /* correct off-axis rounding error */
+    if (position(row)) for (var i = 0; i < row.length; i++) {
+      row[i].dy += h;
+    } else for (var i = 0; i < row.length; i++) {
+      row[i].dx += w;
+    }
+  }
+
+  /* Recursively compute the node depth and size. */
+  stack.unshift(null);
+  root.visitAfter(function(n, i) {
+      n.depth = i;
+      n.x = n.y = n.dx = n.dy = 0;
+      n.size = n.firstChild
+          ? pv.sum(n.childNodes, function(n) { return n.size; })
+          : that.$size.apply(that, (stack[0] = n, stack));
+    });
+  stack.shift();
+
+  /* Sort. */
+  switch (s.order) {
+    case "ascending": {
+      root.sort(function(a, b) { return a.size - b.size; });
+      break;
+    }
+    case "descending": {
+      root.sort(function(a, b) { return b.size - a.size; });
+      break;
+    }
+    case "reverse": root.reverse(); break;
+  }
+
+  /* Recursively compute the layout. */
+  root.x = 0;
+  root.y = 0;
+  root.dx = s.width;
+  root.dy = s.height;
+  root.visitBefore(layout);
+};
+/**
+ * Constructs a new, empty tree layout. Layouts are not typically constructed
+ * directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a node-link tree diagram using the Reingold-Tilford "tidy"
+ * tree layout algorithm. The specific algorithm used by this layout is based on
+ * <a href="http://citeseer.ist.psu.edu/buchheim02improving.html">"Improving
+ * Walker's Algorithm to Run in Linear Time"</A> by C. Buchheim, M. J&uuml;nger
+ * &amp; S. Leipert, Graph Drawing 2002. This layout supports both cartesian and
+ * radial orientations orientations for node-link diagrams.
+ *
+ * <p>The tree layout supports a "group" property, which if true causes siblings
+ * to be positioned closer together than unrelated nodes at the same depth. The
+ * layout can be configured using the <tt>depth</tt> and <tt>breadth</tt>
+ * properties, which control the increments in pixel space between nodes in both
+ * dimensions, similar to the indent layout.
+ *
+ * <p>For more details on how to use this layout, see
+ * {@link pv.Layout.Hierarchy}.
+ *
+ * @extends pv.Layout.Hierarchy
+ */
+pv.Layout.Tree = function() {
+  pv.Layout.Hierarchy.call(this);
+};
+
+pv.Layout.Tree.prototype = pv.extend(pv.Layout.Hierarchy)
+    .property("group", Number)
+    .property("breadth", Number)
+    .property("depth", Number)
+    .property("orient", String);
+
+/**
+ * Default properties for tree layouts. The default orientation is "top", the
+ * default group parameter is 1, and the default breadth and depth offsets are
+ * 15 and 60 respectively.
+ *
+ * @type pv.Layout.Tree
+ */
+pv.Layout.Tree.prototype.defaults = new pv.Layout.Tree()
+    .extend(pv.Layout.Hierarchy.prototype.defaults)
+    .group(1)
+    .breadth(15)
+    .depth(60)
+    .orient("top");
+
+/** @private */
+pv.Layout.Tree.prototype.buildImplied = function(s) {
+  if (pv.Layout.Hierarchy.prototype.buildImplied.call(this, s)) return;
+
+  var nodes = s.nodes,
+      orient = s.orient,
+      depth = s.depth,
+      breadth = s.breadth,
+      group = s.group,
+      w = s.width,
+      h = s.height;
+
+  /** @private */
+  function firstWalk(v) {
+    var l, r, a;
+    if (!v.firstChild) {
+      if (l = v.previousSibling) {
+        v.prelim = l.prelim + distance(v.depth, true);
+      }
+    } else {
+      l = v.firstChild;
+      r = v.lastChild;
+      a = l; // default ancestor
+      for (var c = l; c; c = c.nextSibling) {
+        firstWalk(c);
+        a = apportion(c, a);
+      }
+      executeShifts(v);
+      var midpoint = .5 * (l.prelim + r.prelim);
+      if (l = v.previousSibling) {
+        v.prelim = l.prelim + distance(v.depth, true);
+        v.mod = v.prelim - midpoint;
+      } else {
+        v.prelim = midpoint;
+      }
+    }
+  }
+
+  /** @private */
+  function secondWalk(v, m, depth) {
+    v.breadth = v.prelim + m;
+    m += v.mod;
+    for (var c = v.firstChild; c; c = c.nextSibling) {
+      secondWalk(c, m, depth);
+    }
+  }
+
+  /** @private */
+  function apportion(v, a) {
+    var w = v.previousSibling;
+    if (w) {
+      var vip = v,
+          vop = v,
+          vim = w,
+          vom = v.parentNode.firstChild,
+          sip = vip.mod,
+          sop = vop.mod,
+          sim = vim.mod,
+          som = vom.mod,
+          nr = nextRight(vim),
+          nl = nextLeft(vip);
+      while (nr && nl) {
+        vim = nr;
+        vip = nl;
+        vom = nextLeft(vom);
+        vop = nextRight(vop);
+        vop.ancestor = v;
+        var shift = (vim.prelim + sim) - (vip.prelim + sip) + distance(vim.depth, false);
+        if (shift > 0) {
+          moveSubtree(ancestor(vim, v, a), v, shift);
+          sip += shift;
+          sop += shift;
+        }
+        sim += vim.mod;
+        sip += vip.mod;
+        som += vom.mod;
+        sop += vop.mod;
+        nr = nextRight(vim);
+        nl = nextLeft(vip);
+      }
+      if (nr && !nextRight(vop)) {
+        vop.thread = nr;
+        vop.mod += sim - sop;
+      }
+      if (nl && !nextLeft(vom)) {
+        vom.thread = nl;
+        vom.mod += sip - som;
+        a = v;
+      }
+    }
+    return a;
+  }
+
+  /** @private */
+  function nextLeft(v) {
+    return v.firstChild || v.thread;
+  }
+
+  /** @private */
+  function nextRight(v) {
+    return v.lastChild || v.thread;
+  }
+
+  /** @private */
+  function moveSubtree(wm, wp, shift) {
+    var subtrees = wp.number - wm.number;
+    wp.change -= shift / subtrees;
+    wp.shift += shift;
+    wm.change += shift / subtrees;
+    wp.prelim += shift;
+    wp.mod += shift;
+  }
+
+  /** @private */
+  function executeShifts(v) {
+    var shift = 0, change = 0;
+    for (var c = v.lastChild; c; c = c.previousSibling) {
+      c.prelim += shift;
+      c.mod += shift;
+      change += c.change;
+      shift += c.shift + change;
+    }
+  }
+
+  /** @private */
+  function ancestor(vim, v, a) {
+    return (vim.ancestor.parentNode == v.parentNode) ? vim.ancestor : a;
+  }
+
+  /** @private */
+  function distance(depth, siblings) {
+    return (siblings ? 1 : (group + 1)) / ((orient == "radial") ? depth : 1);
+  }
+
+  /* Initialize temporary layout variables. TODO: store separately. */
+  var root = nodes[0];
+  root.visitAfter(function(v, i) {
+      v.ancestor = v;
+      v.prelim = 0;
+      v.mod = 0;
+      v.change = 0;
+      v.shift = 0;
+      v.number = v.previousSibling ? (v.previousSibling.number + 1) : 0;
+      v.depth = i;
+    });
+
+  /* Compute the layout using Buchheim et al.'s algorithm. */
+  firstWalk(root);
+  secondWalk(root, -root.prelim, 0);
+
+  /** @private Returns the angle of the given node. */
+  function midAngle(n) {
+    return (orient == "radial") ? n.breadth / depth : 0;
+  }
+
+  /** @private */
+  function x(n) {
+    switch (orient) {
+      case "left": return n.depth;
+      case "right": return w - n.depth;
+      case "top":
+      case "bottom": return n.breadth + w / 2;
+      case "radial": return w / 2 + n.depth * Math.cos(midAngle(n));
+    }
+  }
+
+  /** @private */
+  function y(n) {
+    switch (orient) {
+      case "left":
+      case "right": return n.breadth + h / 2;
+      case "top": return n.depth;
+      case "bottom": return h - n.depth;
+      case "radial": return h / 2 + n.depth * Math.sin(midAngle(n));
+    }
+  }
+
+  /* Clear temporary layout variables; transform depth and breadth. */
+  root.visitAfter(function(v) {
+      v.breadth *= breadth;
+      v.depth *= depth;
+      v.midAngle = midAngle(v);
+      v.x = x(v);
+      v.y = y(v);
+      if (v.firstChild) v.midAngle += Math.PI;
+      delete v.breadth;
+      delete v.depth;
+      delete v.ancestor;
+      delete v.prelim;
+      delete v.mod;
+      delete v.change;
+      delete v.shift;
+      delete v.number;
+      delete v.thread;
+    });
+};
+
+/**
+ * The offset between siblings nodes; defaults to 15.
+ *
+ * @type number
+ * @name pv.Layout.Tree.prototype.breadth
+ */
+
+/**
+ * The offset between parent and child nodes; defaults to 60.
+ *
+ * @type number
+ * @name pv.Layout.Tree.prototype.depth
+ */
+
+/**
+ * The orientation. The default orientation is "top", which means that the root
+ * node is placed on the top edge, leaf nodes appear at the bottom, and internal
+ * nodes are in-between. The following orientations are supported:<ul>
+ *
+ * <li>left - left-to-right.
+ * <li>right - right-to-left.
+ * <li>top - top-to-bottom.
+ * <li>bottom - bottom-to-top.
+ * <li>radial - radially, with the root at the center.</ul>
+ *
+ * @type string
+ * @name pv.Layout.Tree.prototype.orient
+ */
+
+/**
+ * The sibling grouping, i.e., whether differentiating space is placed between
+ * sibling groups. The default is 1 (or true), causing sibling leaves to be
+ * separated by one breadth offset. Setting this to false (or 0) causes
+ * non-siblings to be adjacent.
+ *
+ * @type number
+ * @name pv.Layout.Tree.prototype.group
+ */
+/**
+ * Constructs a new, empty indent layout. Layouts are not typically constructed
+ * directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a hierarchical layout using the indent algorithm. This
+ * layout implements a node-link diagram where the nodes are presented in
+ * preorder traversal, and nodes are indented based on their depth from the
+ * root. This technique is used ubiquitously by operating systems to represent
+ * file directories; although it requires much vertical space, indented trees
+ * allow efficient <i>interactive</i> exploration of trees to find a specific
+ * node. In addition they allow rapid scanning of node labels, and multivariate
+ * data such as file sizes can be displayed adjacent to the hierarchy.
+ *
+ * <p>The indent layout can be configured using the <tt>depth</tt> and
+ * <tt>breadth</tt> properties, which control the increments in pixel space for
+ * each indent and row in the layout. This layout does not support multiple
+ * orientations; the root node is rendered in the top-left, while
+ * <tt>breadth</tt> is a vertical offset from the top, and <tt>depth</tt> is a
+ * horizontal offset from the left.
+ *
+ * <p>For more details on how to use this layout, see
+ * {@link pv.Layout.Hierarchy}.
+ *
+ * @extends pv.Layout.Hierarchy
+ */
+pv.Layout.Indent = function() {
+  pv.Layout.Hierarchy.call(this);
+  this.link.interpolate("step-after");
+};
+
+pv.Layout.Indent.prototype = pv.extend(pv.Layout.Hierarchy)
+    .property("depth", Number)
+    .property("breadth", Number);
+
+/**
+ * The horizontal offset between different levels of the tree; defaults to 15.
+ *
+ * @type number
+ * @name pv.Layout.Indent.prototype.depth
+ */
+
+/**
+ * The vertical offset between nodes; defaults to 15.
+ *
+ * @type number
+ * @name pv.Layout.Indent.prototype.breadth
+ */
+
+/**
+ * Default properties for indent layouts. By default the depth and breadth
+ * offsets are 15 pixels.
+ *
+ * @type pv.Layout.Indent
+ */
+pv.Layout.Indent.prototype.defaults = new pv.Layout.Indent()
+    .extend(pv.Layout.Hierarchy.prototype.defaults)
+    .depth(15)
+    .breadth(15);
+
+/** @private */
+pv.Layout.Indent.prototype.buildImplied = function(s) {
+  if (pv.Layout.Hierarchy.prototype.buildImplied.call(this, s)) return;
+
+  var nodes = s.nodes,
+      bspace = s.breadth,
+      dspace = s.depth,
+      ax = 0,
+      ay = 0;
+
+  /** @private */
+  function position(n, breadth, depth) {
+    n.x = ax + depth++ * dspace;
+    n.y = ay + breadth++ * bspace;
+    n.midAngle = 0;
+    for (var c = n.firstChild; c; c = c.nextSibling) {
+      breadth = position(c, breadth, depth);
+    }
+    return breadth;
+  }
+
+  position(nodes[0], 1, 1);
+};
+/**
+ * Constructs a new, empty circle-packing layout. Layouts are not typically
+ * constructed directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a hierarchical layout using circle-packing. The meaning of
+ * the exported mark prototypes changes slightly in the space-filling
+ * implementation:<ul>
+ *
+ * <li><tt>node</tt> - for rendering nodes; typically a {@link pv.Dot}.
+ *
+ * <p><li><tt>link</tt> - unsupported; undefined. Links are encoded implicitly
+ * in the arrangement of the space-filling nodes.
+ *
+ * <p><li><tt>label</tt> - for rendering node labels; typically a
+ * {@link pv.Label}.
+ *
+ * </ul>The pack layout support dynamic sizing for leaf nodes, if a
+ * {@link #size} psuedo-property is specified. The default size function returns
+ * 1, causing all leaf nodes to be sized equally, and all internal nodes to be
+ * sized by the number of leaf nodes they have as descendants.
+ *
+ * <p>The size function can be used in conjunction with the order property,
+ * which allows the nodes to the sorted by the computed size. Note: for sorting
+ * based on other data attributes, simply use the default <tt>null</tt> for the
+ * order property, and sort the nodes beforehand using the {@link pv.Dom}
+ * operator.
+ *
+ * <p>For more details on how to use this layout, see
+ * {@link pv.Layout.Hierarchy}.
+ *
+ * @extends pv.Layout.Hierarchy
+ * @see <a href="http://portal.acm.org/citation.cfm?id=1124772.1124851"
+ * >"Visualization of large hierarchical data by circle packing"</a> by W. Wang,
+ * H. Wang, G. Dai, and H. Wang, ACM CHI 2006.
+ */
+pv.Layout.Pack = function() {
+  pv.Layout.Hierarchy.call(this);
+
+  this.node
+      .radius(function(n) { return n.radius; })
+      .strokeStyle("rgb(31, 119, 180)")
+      .fillStyle("rgba(31, 119, 180, .25)");
+
+  this.label
+      .textAlign("center");
+
+  /* Hide unsupported link. */
+  delete this.link;
+};
+
+pv.Layout.Pack.prototype = pv.extend(pv.Layout.Hierarchy)
+    .property("spacing", Number)
+    .property("order", String); // ascending, descending, reverse, null
+
+/**
+ * Default properties for circle-packing layouts. The default spacing parameter
+ * is 1 and the default order is "ascending".
+ *
+ * @type pv.Layout.Pack
+ */
+pv.Layout.Pack.prototype.defaults = new pv.Layout.Pack()
+    .extend(pv.Layout.Hierarchy.prototype.defaults)
+    .spacing(1)
+    .order("ascending");
+
+/**
+ * The spacing parameter; defaults to 1, which provides a little bit of padding
+ * between sibling nodes and the enclosing circle. Larger values increase the
+ * spacing, by making the sibling nodes smaller; a value of zero makes the leaf
+ * nodes as large as possible, with no padding on enclosing circles.
+ *
+ * @type number
+ * @name pv.Layout.Pack.prototype.spacing
+ */
+
+/**
+ * The sibling node order. The default order is <tt>null</tt>, which means to
+ * use the sibling order specified by the nodes property as-is. A value of
+ * "ascending" will sort siblings in ascending order of size, while "descending"
+ * will do the reverse. For sorting based on data attributes other than size,
+ * use the default <tt>null</tt> for the order property, and sort the nodes
+ * beforehand using the {@link pv.Dom} operator.
+ *
+ * @see pv.Dom.Node#sort
+ * @type string
+ * @name pv.Layout.Pack.prototype.order
+ */
+
+/** @private The default size function. */
+pv.Layout.Pack.prototype.$radius = function() { return 1; };
+
+// TODO is it possible for spacing to operate in pixel space?
+// Right now it appears to be multiples of the smallest radius.
+
+/**
+ * Specifies the sizing function. By default, a sizing function is disabled and
+ * all nodes are given constant size. The sizing function is invoked for each
+ * leaf node in the tree (passed to the constructor).
+ *
+ * <p>For example, if the tree data structure represents a file system, with
+ * files as leaf nodes, and each file has a <tt>bytes</tt> attribute, you can
+ * specify a size function as:
+ *
+ * <pre>    .size(function(d) d.bytes)</pre>
+ *
+ * As with other properties, a size function may specify additional arguments to
+ * access the data associated with the layout and any enclosing panels.
+ *
+ * @param {function} f the new sizing function.
+ * @returns {pv.Layout.Pack} this.
+ */
+pv.Layout.Pack.prototype.size = function(f) {
+  this.$radius = typeof f == "function"
+      ? function() { return Math.sqrt(f.apply(this, arguments)); }
+      : (f = Math.sqrt(f), function() { return f; });
+  return this;
+};
+
+/** @private */
+pv.Layout.Pack.prototype.buildImplied = function(s) {
+  if (pv.Layout.Hierarchy.prototype.buildImplied.call(this, s)) return;
+
+  var that = this,
+      nodes = s.nodes,
+      root = nodes[0];
+
+  /** @private Compute the radii of the leaf nodes. */
+  function radii(nodes) {
+    var stack = pv.Mark.stack;
+    stack.unshift(null);
+    for (var i = 0, n = nodes.length; i < n; i++) {
+      var c = nodes[i];
+      if (!c.firstChild) {
+        c.radius = that.$radius.apply(that, (stack[0] = c, stack));
+      }
+    }
+    stack.shift();
+  }
+
+  /** @private */
+  function packTree(n) {
+    var nodes = [];
+    for (var c = n.firstChild; c; c = c.nextSibling) {
+      if (c.firstChild) c.radius = packTree(c);
+      c.n = c.p = c;
+      nodes.push(c);
+    }
+
+    /* Sort. */
+    switch (s.order) {
+      case "ascending": {
+        nodes.sort(function(a, b) { return a.radius - b.radius; });
+        break;
+      }
+      case "descending": {
+        nodes.sort(function(a, b) { return b.radius - a.radius; });
+        break;
+      }
+      case "reverse": nodes.reverse(); break;
+    }
+
+    return packCircle(nodes);
+  }
+
+  /** @private */
+  function packCircle(nodes) {
+    var xMin = Infinity,
+        xMax = -Infinity,
+        yMin = Infinity,
+        yMax = -Infinity,
+        a, b, c, j, k;
+
+    /** @private */
+    function bound(n) {
+      xMin = Math.min(n.x - n.radius, xMin);
+      xMax = Math.max(n.x + n.radius, xMax);
+      yMin = Math.min(n.y - n.radius, yMin);
+      yMax = Math.max(n.y + n.radius, yMax);
+    }
+
+    /** @private */
+    function insert(a, b) {
+      var c = a.n;
+      a.n = b;
+      b.p = a;
+      b.n = c;
+      c.p = b;
+    }
+
+    /** @private */
+    function splice(a, b) {
+      a.n = b;
+      b.p = a;
+    }
+
+    /** @private */
+    function intersects(a, b) {
+      var dx = b.x - a.x,
+          dy = b.y - a.y,
+          dr = a.radius + b.radius;
+      return (dr * dr - dx * dx - dy * dy) > .001; // within epsilon
+    }
+
+    /* Create first node. */
+    a = nodes[0];
+    a.x = -a.radius;
+    a.y = 0;
+    bound(a);
+
+    /* Create second node. */
+    if (nodes.length > 1) {
+      b = nodes[1];
+      b.x = b.radius;
+      b.y = 0;
+      bound(b);
+
+      /* Create third node and build chain. */
+      if (nodes.length > 2) {
+        c = nodes[2];
+        place(a, b, c);
+        bound(c);
+        insert(a, c);
+        a.p = c;
+        insert(c, b);
+        b = a.n;
+
+        /* Now iterate through the rest. */
+        for (var i = 3; i < nodes.length; i++) {
+          place(a, b, c = nodes[i]);
+
+          /* Search for the closest intersection. */
+          var isect = 0, s1 = 1, s2 = 1;
+          for (j = b.n; j != b; j = j.n, s1++) {
+            if (intersects(j, c)) {
+              isect = 1;
+              break;
+            }
+          }
+          if (isect == 1) {
+            for (k = a.p; k != j.p; k = k.p, s2++) {
+              if (intersects(k, c)) {
+                if (s2 < s1) {
+                  isect = -1;
+                  j = k;
+                }
+                break;
+              }
+            }
+          }
+
+          /* Update node chain. */
+          if (isect == 0) {
+            insert(a, c);
+            b = c;
+            bound(c);
+          } else if (isect > 0) {
+            splice(a, j);
+            b = j;
+            i--;
+          } else if (isect < 0) {
+            splice(j, b);
+            a = j;
+            i--;
+          }
+        }
+      }
+    }
+
+    /* Re-center the circles and return the encompassing radius. */
+    var cx = (xMin + xMax) / 2,
+        cy = (yMin + yMax) / 2,
+        cr = 0;
+    for (var i = 0; i < nodes.length; i++) {
+      var n = nodes[i];
+      n.x -= cx;
+      n.y -= cy;
+      cr = Math.max(cr, n.radius + Math.sqrt(n.x * n.x + n.y * n.y));
+    }
+    return cr + s.spacing;
+  }
+
+  /** @private */
+  function place(a, b, c) {
+    var da = b.radius + c.radius,
+        db = a.radius + c.radius,
+        dx = b.x - a.x,
+        dy = b.y - a.y,
+        dc = Math.sqrt(dx * dx + dy * dy),
+        cos = (db * db + dc * dc - da * da) / (2 * db * dc),
+        theta = Math.acos(cos),
+        x = cos * db,
+        h = Math.sin(theta) * db;
+    dx /= dc;
+    dy /= dc;
+    c.x = a.x + x * dx + h * dy;
+    c.y = a.y + x * dy - h * dx;
+  }
+
+  /** @private */
+  function transform(n, x, y, k) {
+    for (var c = n.firstChild; c; c = c.nextSibling) {
+      c.x += n.x;
+      c.y += n.y;
+      transform(c, x, y, k);
+    }
+    n.x = x + k * n.x;
+    n.y = y + k * n.y;
+    n.radius *= k;
+  }
+
+  radii(nodes);
+
+  /* Recursively compute the layout. */
+  root.x = 0;
+  root.y = 0;
+  root.radius = packTree(root);
+
+  var w = this.width(),
+      h = this.height(),
+      k = 1 / Math.max(2 * root.radius / w, 2 * root.radius / h);
+  transform(root, w / 2, h / 2, k);
+};
+/**
+ * Constructs a new, empty force-directed layout. Layouts are not typically
+ * constructed directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements force-directed network layout as a node-link diagram. This
+ * layout uses the Fruchterman-Reingold algorithm, which applies an attractive
+ * spring force between neighboring nodes, and a repulsive electrical charge
+ * force between all nodes. An additional drag force improves stability of the
+ * simulation. See {@link pv.Force.spring}, {@link pv.Force.drag} and {@link
+ * pv.Force.charge} for more details; note that the n-body charge force is
+ * approximated using the Barnes-Hut algorithm.
+ *
+ * <p>This layout is implemented on top of {@link pv.Simulation}, which can be
+ * used directly for more control over simulation parameters. The simulation
+ * uses Position Verlet integration, which does not compute velocities
+ * explicitly, but allows for easy geometric constraints, such as bounding the
+ * nodes within the layout panel. Many of the configuration properties supported
+ * by this layout are simply passed through to the underlying forces and
+ * constraints of the simulation.
+ *
+ * <p>Force layouts are typically interactive. The gradual movement of the nodes
+ * as they stabilize to a local stress minimum can help reveal the structure of
+ * the network, as can {@link pv.Behavior.drag}, which allows the user to pick
+ * up nodes and reposition them while the physics simulation continues. This
+ * layout can also be used with pan &amp; zoom behaviors for interaction.
+ *
+ * <p>To facilitate interaction, this layout by default automatically re-renders
+ * using a <tt>setInterval</tt> every 42 milliseconds. This can be disabled via
+ * the <tt>iterations</tt> property, which if non-null specifies the number of
+ * simulation iterations to run before the force-directed layout is finalized.
+ * Be careful not to use too high an iteration count, as this can lead to an
+ * annoying delay on page load.
+ *
+ * <p>As with other network layouts, the network data can be updated
+ * dynamically, provided the property cache is reset. See
+ * {@link pv.Layout.Network} for details. New nodes are initialized with random
+ * positions near the center. Alternatively, positions can be specified manually
+ * by setting the <tt>x</tt> and <tt>y</tt> attributes on nodes.
+ *
+ * @extends pv.Layout.Network
+ * @see <a href="http://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.13.8444&rep=rep1&type=pdf"
+ * >"Graph Drawing by Force-directed Placement"</a> by T. Fruchterman &amp;
+ * E. Reingold, Software--Practice &amp; Experience, November 1991.
+ */
+pv.Layout.Force = function() {
+  pv.Layout.Network.call(this);
+
+  /* Force-directed graphs can be messy, so reduce the link width. */
+  this.link.lineWidth(function(d, p) { return Math.sqrt(p.linkValue) * 1.5; });
+  this.label.textAlign("center");
+};
+
+pv.Layout.Force.prototype = pv.extend(pv.Layout.Network)
+    .property("bound", Boolean)
+    .property("iterations", Number)
+    .property("dragConstant", Number)
+    .property("chargeConstant", Number)
+    .property("chargeMinDistance", Number)
+    .property("chargeMaxDistance", Number)
+    .property("chargeTheta", Number)
+    .property("springConstant", Number)
+    .property("springDamping", Number)
+    .property("springLength", Number);
+
+/**
+ * The bound parameter; true if nodes should be constrained within the layout
+ * panel. Bounding is disabled by default. Currently the layout does not observe
+ * the radius of the nodes; strictly speaking, only the center of the node is
+ * constrained to be within the panel, with an additional 6-pixel offset for
+ * padding. A future enhancement could extend the bound constraint to observe
+ * the node's radius, which would also support bounding for variable-size nodes.
+ *
+ * <p>Note that if this layout is used in conjunction with pan &amp; zoom
+ * behaviors, those behaviors should have their bound parameter set to the same
+ * value.
+ *
+ * @type boolean
+ * @name pv.Layout.Force.prototype.bound
+ */
+
+/**
+ * The number of simulation iterations to run, or null if this layout is
+ * interactive. Force-directed layouts are interactive by default, using a
+ * <tt>setInterval</tt> to advance the physics simulation and re-render
+ * automatically.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.iterations
+ */
+
+/**
+ * The drag constant, in the range [0,1]. A value of 0 means no drag (a
+ * perfectly frictionless environment), while a value of 1 means friction
+ * immediately cancels all momentum. The default value is 0.1, which provides a
+ * minimum amount of drag that helps stabilize bouncy springs; lower values may
+ * result in excessive bounciness, while higher values cause the simulation to
+ * take longer to converge.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.dragConstant
+ * @see pv.Force.drag#constant
+ */
+
+/**
+ * The charge constant, which should be a negative number. The default value is
+ * -40; more negative values will result in a stronger repulsive force, which
+ * may lead to faster convergence at the risk of instability. Too strong
+ * repulsive charge forces can cause comparatively weak springs to be stretched
+ * well beyond their rest length, emphasizing global structure over local
+ * structure. A nonnegative value will break the Fruchterman-Reingold algorithm,
+ * and is for entertainment purposes only.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.chargeConstant
+ * @see pv.Force.charge#constant
+ */
+
+/**
+ * The minimum distance at which charge forces are applied. The default minimum
+ * distance of 2 avoids applying forces that are two strong; because the physics
+ * simulation is run at discrete time intervals, it is possible for two same-
+ * charged particles to become very close or even a singularity! Since the
+ * charge force is inversely proportional to the square of the distance, very
+ * small distances can break the simulation.
+ *
+ * <p>In rare cases, two particles can become stuck on top of each other, as a
+ * minimum distance threshold will prevent the charge force from repelling them.
+ * However, this occurs very rarely because other forces and momentum typically
+ * cause the particles to become separated again, at which point the repulsive
+ * charge force kicks in.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.chargeMinDistance
+ * @see pv.Force.charge#domain
+ */
+
+/**
+ * The maximum distance at which charge forces are applied. This improves
+ * performance by ignoring weak charge forces at great distances. Note that this
+ * parameter is partly redundant, as the Barnes-Hut algorithm for n-body forces
+ * already improves performance for far-away particles through approximation.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.chargeMaxDistance
+ * @see pv.Force.charge#domain
+ */
+
+/**
+ * The Barnes-Hut approximation factor. The Barnes-Hut approximation criterion
+ * is the ratio of the size of the quadtree node to the distance from the point
+ * to the node's center of mass is beneath some threshold. The default value is
+ * 0.9.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.chargeTheta
+ * @see pv.Force.charge#theta
+ */
+
+/**
+ * The spring constant, which should be a positive number. The default value is
+ * 0.1; greater values will result in a stronger attractive force, which may
+ * lead to faster convergence at the risk of instability. Too strong spring
+ * forces can cause comparatively weak charge forces to be ignored, emphasizing
+ * local structure over global structure. A nonpositive value will break the
+ * Fruchterman-Reingold algorithm, and is for entertainment purposes only.
+ *
+ * <p>The spring tension is automatically normalized using the inverse square
+ * root of the maximum link degree of attached nodes.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.springConstant
+ * @see pv.Force.spring#constant
+ */
+
+/**
+ * The spring damping factor, in the range [0,1]. Damping functions identically
+ * to drag forces, damping spring bounciness by applying a force in the opposite
+ * direction of attached nodes' velocities. The default value is 0.3.
+ *
+ * <p>The spring damping is automatically normalized using the inverse square
+ * root of the maximum link degree of attached nodes.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.springDamping
+ * @see pv.Force.spring#damping
+ */
+
+/**
+ * The spring rest length. The default value is 20 pixels. Larger values may be
+ * appropriate if the layout panel is larger, or if the nodes are rendered
+ * larger than the default dot size of 20.
+ *
+ * @type number
+ * @name pv.Layout.Force.prototype.springLength
+ * @see pv.Force.spring#length
+ */
+
+/**
+ * Default properties for force-directed layouts. The default drag constant is
+ * 0.1, the default charge constant is -40 (with a domain of [2, 500] and theta
+ * of 0.9), and the default spring constant is 0.1 (with a damping of 0.3 and a
+ * rest length of 20).
+ *
+ * @type pv.Layout.Force
+ */
+pv.Layout.Force.prototype.defaults = new pv.Layout.Force()
+    .extend(pv.Layout.Network.prototype.defaults)
+    .dragConstant(.1)
+    .chargeConstant(-40)
+    .chargeMinDistance(2)
+    .chargeMaxDistance(500)
+    .chargeTheta(.9)
+    .springConstant(.1)
+    .springDamping(.3)
+    .springLength(20);
+
+/** @private Initialize the physics simulation. */
+pv.Layout.Force.prototype.buildImplied = function(s) {
+
+  /* Any cached interactive layouts need to be rebound for the timer. */
+  if (pv.Layout.Network.prototype.buildImplied.call(this, s)) {
+    var f = s.$force;
+    if (f) {
+      f.next = this.binds.$force;
+      this.binds.$force = f;
+    }
+    return;
+  }
+
+  var that = this,
+      nodes = s.nodes,
+      links = s.links,
+      k = s.iterations,
+      w = s.width,
+      h = s.height;
+
+  /* Initialize positions randomly near the center. */
+  for (var i = 0, n; i < nodes.length; i++) {
+    n = nodes[i];
+    if (isNaN(n.x)) n.x = w / 2 + 40 * Math.random() - 20;
+    if (isNaN(n.y)) n.y = h / 2 + 40 * Math.random() - 20;
+  }
+
+  /* Initialize the simulation. */
+  var sim = pv.simulation(nodes);
+
+  /* Drag force. */
+  sim.force(pv.Force.drag(s.dragConstant));
+
+  /* Charge (repelling) force. */
+  sim.force(pv.Force.charge(s.chargeConstant)
+      .domain(s.chargeMinDistance, s.chargeMaxDistance)
+      .theta(s.chargeTheta));
+
+  /* Spring (attracting) force. */
+  sim.force(pv.Force.spring(s.springConstant)
+      .damping(s.springDamping)
+      .length(s.springLength)
+      .links(links));
+
+  /* Position constraint (for interactive dragging). */
+  sim.constraint(pv.Constraint.position());
+
+  /* Optionally add bound constraint. TODO: better padding. */
+  if (s.bound) {
+    sim.constraint(pv.Constraint.bound().x(6, w - 6).y(6, h - 6));
+  }
+
+  /** @private Returns the speed of the given node, to determine cooling. */
+  function speed(n) {
+    return n.fix ? 1 : n.vx * n.vx + n.vy * n.vy;
+  }
+
+  /*
+   * If the iterations property is null (the default), the layout is
+   * interactive. The simulation is run until the fastest particle drops below
+   * an arbitrary minimum speed. Although the timer keeps firing, this speed
+   * calculation is fast so there is minimal CPU overhead. Note: if a particle
+   * is fixed for interactivity, treat this as a high speed and resume
+   * simulation.
+   */
+  if (k == null) {
+    sim.step(); // compute initial previous velocities
+    sim.step(); // compute initial velocities
+
+    /* Add the simulation state to the bound list. */
+    var force = s.$force = this.binds.$force = {
+      next: this.binds.$force,
+      nodes: nodes,
+      min: 1e-4 * (links.length + 1),
+      sim: sim
+    };
+
+    /* Start the timer, if not already started. */
+    if (!this.$timer) this.$timer = setInterval(function() {
+      var render = false;
+      for (var f = that.binds.$force; f; f = f.next) {
+        if (pv.max(f.nodes, speed) > f.min) {
+          f.sim.step();
+          render = true;
+        }
+      }
+      if (render) that.render();
+    }, 42);
+  } else for (var i = 0; i < k; i++) {
+    sim.step();
+  }
+};
+/**
+ * Constructs a new, empty cluster layout. Layouts are not typically
+ * constructed directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a hierarchical layout using the cluster (or dendrogram)
+ * algorithm. This layout provides both node-link and space-filling
+ * implementations of cluster diagrams. In many ways it is similar to
+ * {@link pv.Layout.Partition}, except that leaf nodes are positioned at maximum
+ * depth, and the depth of internal nodes is based on their distance from their
+ * deepest descendant, rather than their distance from the root.
+ *
+ * <p>The cluster layout supports a "group" property, which if true causes
+ * siblings to be positioned closer together than unrelated nodes at the same
+ * depth. Unlike the partition layout, this layout does not support dynamic
+ * sizing for leaf nodes; all leaf nodes are the same size.
+ *
+ * <p>For more details on how to use this layout, see
+ * {@link pv.Layout.Hierarchy}.
+ *
+ * @see pv.Layout.Cluster.Fill
+ * @extends pv.Layout.Hierarchy
+ */
+pv.Layout.Cluster = function() {
+  pv.Layout.Hierarchy.call(this);
+  var interpolate, // cached interpolate
+      buildImplied = this.buildImplied;
+
+  /** @private Cache layout state to optimize properties. */
+  this.buildImplied = function(s) {
+    buildImplied.call(this, s);
+    interpolate
+        = /^(top|bottom)$/.test(s.orient) ? "step-before"
+        : /^(left|right)$/.test(s.orient) ? "step-after"
+        : "linear";
+  };
+
+  this.link.interpolate(function() { return interpolate; });
+};
+
+pv.Layout.Cluster.prototype = pv.extend(pv.Layout.Hierarchy)
+    .property("group", Number)
+    .property("orient", String)
+    .property("innerRadius", Number)
+    .property("outerRadius", Number);
+
+/**
+ * The group parameter; defaults to 0, disabling grouping of siblings. If this
+ * parameter is set to a positive number (or true, which is equivalent to 1),
+ * then additional space will be allotted between sibling groups. In other
+ * words, siblings (nodes that share the same parent) will be positioned more
+ * closely than nodes at the same depth that do not share a parent.
+ *
+ * @type number
+ * @name pv.Layout.Cluster.prototype.group
+ */
+
+/**
+ * The orientation. The default orientation is "top", which means that the root
+ * node is placed on the top edge, leaf nodes appear on the bottom edge, and
+ * internal nodes are in-between. The following orientations are supported:<ul>
+ *
+ * <li>left - left-to-right.
+ * <li>right - right-to-left.
+ * <li>top - top-to-bottom.
+ * <li>bottom - bottom-to-top.
+ * <li>radial - radially, with the root at the center.</ul>
+ *
+ * @type string
+ * @name pv.Layout.Cluster.prototype.orient
+ */
+
+/**
+ * The inner radius; defaults to 0. This property applies only to radial
+ * orientations, and can be used to compress the layout radially. Note that for
+ * the node-link implementation, the root node is always at the center,
+ * regardless of the value of this property; this property only affects internal
+ * and leaf nodes. For the space-filling implementation, a non-zero value of
+ * this property will result in the root node represented as a ring rather than
+ * a circle.
+ *
+ * @type number
+ * @name pv.Layout.Cluster.prototype.innerRadius
+ */
+
+/**
+ * The outer radius; defaults to fill the containing panel, based on the height
+ * and width of the layout. If the layout has no height and width specified, it
+ * will extend to fill the enclosing panel.
+ *
+ * @type number
+ * @name pv.Layout.Cluster.prototype.outerRadius
+ */
+
+/**
+ * Defaults for cluster layouts. The default group parameter is 0 and the
+ * default orientation is "top".
+ *
+ * @type pv.Layout.Cluster
+ */
+pv.Layout.Cluster.prototype.defaults = new pv.Layout.Cluster()
+    .extend(pv.Layout.Hierarchy.prototype.defaults)
+    .group(0)
+    .orient("top");
+
+/** @private */
+pv.Layout.Cluster.prototype.buildImplied = function(s) {
+  if (pv.Layout.Hierarchy.prototype.buildImplied.call(this, s)) return;
+
+  var root = s.nodes[0],
+      group = s.group,
+      breadth,
+      depth,
+      leafCount = 0,
+      leafIndex = .5 - group / 2;
+
+  /* Count the leaf nodes and compute the depth of descendants. */
+  var p = undefined;
+  root.visitAfter(function(n) {
+      if (n.firstChild) {
+        n.depth = 1 + pv.max(n.childNodes, function(n) { return n.depth; });
+      } else {
+        if (group && (p != n.parentNode)) {
+          p = n.parentNode;
+          leafCount += group;
+        }
+        leafCount++;
+        n.depth = 0;
+      }
+    });
+  breadth = 1 / leafCount;
+  depth = 1 / root.depth;
+
+  /* Compute the unit breadth and depth of each node. */
+  var p = undefined;
+  root.visitAfter(function(n) {
+      if (n.firstChild) {
+        n.breadth = pv.mean(n.childNodes, function(n) { return n.breadth; });
+      } else {
+        if (group && (p != n.parentNode)) {
+          p = n.parentNode;
+          leafIndex += group;
+        }
+        n.breadth = breadth * leafIndex++;
+      }
+      n.depth = 1 - n.depth * depth;
+    });
+
+  /* Compute breadth and depth ranges for space-filling layouts. */
+  root.visitAfter(function(n) {
+      n.minBreadth = n.firstChild
+          ? n.firstChild.minBreadth
+          : (n.breadth - breadth / 2);
+      n.maxBreadth = n.firstChild
+          ? n.lastChild.maxBreadth
+          : (n.breadth + breadth / 2);
+    });
+  root.visitBefore(function(n) {
+      n.minDepth = n.parentNode
+          ? n.parentNode.maxDepth
+          : 0;
+      n.maxDepth = n.parentNode
+          ? (n.depth + root.depth)
+          : (n.minDepth + 2 * root.depth);
+    });
+  root.minDepth = -depth;
+
+  pv.Layout.Hierarchy.NodeLink.buildImplied.call(this, s);
+};
+
+/**
+ * Constructs a new, empty space-filling cluster layout. Layouts are not
+ * typically constructed directly; instead, they are added to an existing panel
+ * via {@link pv.Mark#add}.
+ *
+ * @class A variant of cluster layout that is space-filling. The meaning of the
+ * exported mark prototypes changes slightly in the space-filling
+ * implementation:<ul>
+ *
+ * <li><tt>node</tt> - for rendering nodes; typically a {@link pv.Bar} for
+ * non-radial orientations, and a {@link pv.Wedge} for radial orientations.
+ *
+ * <p><li><tt>link</tt> - unsupported; undefined. Links are encoded implicitly
+ * in the arrangement of the space-filling nodes.
+ *
+ * <p><li><tt>label</tt> - for rendering node labels; typically a
+ * {@link pv.Label}.
+ *
+ * </ul>For more details on how to use this layout, see
+ * {@link pv.Layout.Cluster}.
+ *
+ * @extends pv.Layout.Cluster
+ */
+pv.Layout.Cluster.Fill = function() {
+  pv.Layout.Cluster.call(this);
+  pv.Layout.Hierarchy.Fill.constructor.call(this);
+};
+
+pv.Layout.Cluster.Fill.prototype = pv.extend(pv.Layout.Cluster);
+
+/** @private */
+pv.Layout.Cluster.Fill.prototype.buildImplied = function(s) {
+  if (pv.Layout.Cluster.prototype.buildImplied.call(this, s)) return;
+  pv.Layout.Hierarchy.Fill.buildImplied.call(this, s);
+};
+/**
+ * Constructs a new, empty partition layout. Layouts are not typically
+ * constructed directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implemeents a hierarchical layout using the partition (or sunburst,
+ * icicle) algorithm. This layout provides both node-link and space-filling
+ * implementations of partition diagrams. In many ways it is similar to
+ * {@link pv.Layout.Cluster}, except that leaf nodes are positioned based on
+ * their distance from the root.
+ *
+ * <p>The partition layout support dynamic sizing for leaf nodes, if a
+ * {@link #size} psuedo-property is specified. The default size function returns
+ * 1, causing all leaf nodes to be sized equally, and all internal nodes to be
+ * sized by the number of leaf nodes they have as descendants.
+ *
+ * <p>The size function can be used in conjunction with the order property,
+ * which allows the nodes to the sorted by the computed size. Note: for sorting
+ * based on other data attributes, simply use the default <tt>null</tt> for the
+ * order property, and sort the nodes beforehand using the {@link pv.Dom}
+ * operator.
+ *
+ * <p>For more details on how to use this layout, see
+ * {@link pv.Layout.Hierarchy}.
+ *
+ * @see pv.Layout.Partition.Fill
+ * @extends pv.Layout.Hierarchy
+ */
+pv.Layout.Partition = function() {
+  pv.Layout.Hierarchy.call(this);
+};
+
+pv.Layout.Partition.prototype = pv.extend(pv.Layout.Hierarchy)
+    .property("order", String) // null, ascending, descending?
+    .property("orient", String) // top, left, right, bottom, radial
+    .property("innerRadius", Number)
+    .property("outerRadius", Number);
+
+/**
+ * The sibling node order. The default order is <tt>null</tt>, which means to
+ * use the sibling order specified by the nodes property as-is. A value of
+ * "ascending" will sort siblings in ascending order of size, while "descending"
+ * will do the reverse. For sorting based on data attributes other than size,
+ * use the default <tt>null</tt> for the order property, and sort the nodes
+ * beforehand using the {@link pv.Dom} operator.
+ *
+ * @see pv.Dom.Node#sort
+ * @type string
+ * @name pv.Layout.Partition.prototype.order
+ */
+
+/**
+ * The orientation. The default orientation is "top", which means that the root
+ * node is placed on the top edge, leaf nodes appear at the bottom, and internal
+ * nodes are in-between. The following orientations are supported:<ul>
+ *
+ * <li>left - left-to-right.
+ * <li>right - right-to-left.
+ * <li>top - top-to-bottom.
+ * <li>bottom - bottom-to-top.
+ * <li>radial - radially, with the root at the center.</ul>
+ *
+ * @type string
+ * @name pv.Layout.Partition.prototype.orient
+ */
+
+/**
+ * The inner radius; defaults to 0. This property applies only to radial
+ * orientations, and can be used to compress the layout radially. Note that for
+ * the node-link implementation, the root node is always at the center,
+ * regardless of the value of this property; this property only affects internal
+ * and leaf nodes. For the space-filling implementation, a non-zero value of
+ * this property will result in the root node represented as a ring rather than
+ * a circle.
+ *
+ * @type number
+ * @name pv.Layout.Partition.prototype.innerRadius
+ */
+
+/**
+ * The outer radius; defaults to fill the containing panel, based on the height
+ * and width of the layout. If the layout has no height and width specified, it
+ * will extend to fill the enclosing panel.
+ *
+ * @type number
+ * @name pv.Layout.Partition.prototype.outerRadius
+ */
+
+/**
+ * Default properties for partition layouts. The default orientation is "top".
+ *
+ * @type pv.Layout.Partition
+ */
+pv.Layout.Partition.prototype.defaults = new pv.Layout.Partition()
+    .extend(pv.Layout.Hierarchy.prototype.defaults)
+    .orient("top");
+
+/** @private */
+pv.Layout.Partition.prototype.$size = function() { return 1; };
+
+/**
+ * Specifies the sizing function. By default, a sizing function is disabled and
+ * all nodes are given constant size. The sizing function is invoked for each
+ * leaf node in the tree (passed to the constructor).
+ *
+ * <p>For example, if the tree data structure represents a file system, with
+ * files as leaf nodes, and each file has a <tt>bytes</tt> attribute, you can
+ * specify a size function as:
+ *
+ * <pre>    .size(function(d) d.bytes)</pre>
+ *
+ * As with other properties, a size function may specify additional arguments to
+ * access the data associated with the layout and any enclosing panels.
+ *
+ * @param {function} f the new sizing function.
+ * @returns {pv.Layout.Partition} this.
+ */
+pv.Layout.Partition.prototype.size = function(f) {
+  this.$size = f;
+  return this;
+};
+
+/** @private */
+pv.Layout.Partition.prototype.buildImplied = function(s) {
+  if (pv.Layout.Hierarchy.prototype.buildImplied.call(this, s)) return;
+
+  var that = this,
+      root = s.nodes[0],
+      stack = pv.Mark.stack,
+      maxDepth = 0;
+
+  /* Recursively compute the tree depth and node size. */
+  stack.unshift(null);
+  root.visitAfter(function(n, i) {
+      if (i > maxDepth) maxDepth = i;
+      n.size = n.firstChild
+          ? pv.sum(n.childNodes, function(n) { return n.size; })
+          : that.$size.apply(that, (stack[0] = n, stack));
+    });
+  stack.shift();
+
+  /* Order */
+  switch (s.order) {
+    case "ascending": root.sort(function(a, b) { return a.size - b.size; }); break;
+    case "descending": root.sort(function(b, a) { return a.size - b.size; }); break;
+  }
+
+  /* Compute the unit breadth and depth of each node. */
+  var ds = 1 / maxDepth;
+  root.minBreadth = 0;
+  root.breadth = .5;
+  root.maxBreadth = 1;
+  root.visitBefore(function(n) {
+    var b = n.minBreadth, s = n.maxBreadth - b;
+      for (var c = n.firstChild; c; c = c.nextSibling) {
+        c.minBreadth = b;
+        c.maxBreadth = b += (c.size / n.size) * s;
+        c.breadth = (b + c.minBreadth) / 2;
+      }
+    });
+  root.visitAfter(function(n, i) {
+      n.minDepth = (i - 1) * ds;
+      n.maxDepth = n.depth = i * ds;
+    });
+
+  pv.Layout.Hierarchy.NodeLink.buildImplied.call(this, s);
+};
+
+/**
+ * Constructs a new, empty space-filling partition layout. Layouts are not
+ * typically constructed directly; instead, they are added to an existing panel
+ * via {@link pv.Mark#add}.
+ *
+ * @class A variant of partition layout that is space-filling. The meaning of
+ * the exported mark prototypes changes slightly in the space-filling
+ * implementation:<ul>
+ *
+ * <li><tt>node</tt> - for rendering nodes; typically a {@link pv.Bar} for
+ * non-radial orientations, and a {@link pv.Wedge} for radial orientations.
+ *
+ * <p><li><tt>link</tt> - unsupported; undefined. Links are encoded implicitly
+ * in the arrangement of the space-filling nodes.
+ *
+ * <p><li><tt>label</tt> - for rendering node labels; typically a
+ * {@link pv.Label}.
+ *
+ * </ul>For more details on how to use this layout, see
+ * {@link pv.Layout.Partition}.
+ *
+ * @extends pv.Layout.Partition
+ */
+pv.Layout.Partition.Fill = function() {
+  pv.Layout.Partition.call(this);
+  pv.Layout.Hierarchy.Fill.constructor.call(this);
+};
+
+pv.Layout.Partition.Fill.prototype = pv.extend(pv.Layout.Partition);
+
+/** @private */
+pv.Layout.Partition.Fill.prototype.buildImplied = function(s) {
+  if (pv.Layout.Partition.prototype.buildImplied.call(this, s)) return;
+  pv.Layout.Hierarchy.Fill.buildImplied.call(this, s);
+};
+/**
+ * Constructs a new, empty arc layout. Layouts are not typically constructed
+ * directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a layout for arc diagrams. An arc diagram is a network
+ * visualization with a one-dimensional layout of nodes, using circular arcs to
+ * render links between nodes. For undirected networks, arcs are rendering on a
+ * single side; this makes arc diagrams useful as annotations to other
+ * two-dimensional network layouts, such as rollup, matrix or table layouts. For
+ * directed networks, links in opposite directions can be rendered on opposite
+ * sides using <tt>directed(true)</tt>.
+ *
+ * <p>Arc layouts are particularly sensitive to node ordering; for best results,
+ * order the nodes such that related nodes are close to each other. A poor
+ * (e.g., random) order may result in large arcs with crossovers that impede
+ * visual processing. A future improvement to this layout may include automatic
+ * reordering using, e.g., spectral graph layout or simulated annealing.
+ *
+ * <p>This visualization technique is related to that developed by
+ * M. Wattenberg, <a
+ * href="http://www.research.ibm.com/visual/papers/arc-diagrams.pdf">"Arc
+ * Diagrams: Visualizing Structure in Strings"</a> in <i>IEEE InfoVis</i>, 2002.
+ * However, this implementation is limited to simple node-link networks, as
+ * opposed to structures with hierarchical self-similarity (such as strings).
+ *
+ * <p>As with other network layouts, three mark prototypes are provided:<ul>
+ *
+ * <li><tt>node</tt> - for rendering nodes; typically a {@link pv.Dot}.
+ * <li><tt>link</tt> - for rendering links; typically a {@link pv.Line}.
+ * <li><tt>label</tt> - for rendering node labels; typically a {@link pv.Label}.
+ *
+ * </ul>For more details on how this layout is structured and can be customized,
+ * see {@link pv.Layout.Network}.
+ *
+ * @extends pv.Layout.Network
+ **/
+pv.Layout.Arc = function() {
+  pv.Layout.Network.call(this);
+  var interpolate, // cached interpolate
+      directed, // cached directed
+      reverse, // cached reverse
+      buildImplied = this.buildImplied;
+
+  /** @private Cache layout state to optimize properties. */
+  this.buildImplied = function(s) {
+    buildImplied.call(this, s);
+    directed = s.directed;
+    interpolate = s.orient == "radial" ? "linear" : "polar";
+    reverse = s.orient == "right" || s.orient == "top";
+  };
+
+  /* Override link properties to handle directedness and orientation. */
+  this.link
+      .data(function(p) {
+          var s = p.sourceNode, t = p.targetNode;
+          return reverse != (directed || (s.breadth < t.breadth)) ? [s, t] : [t, s];
+        })
+      .interpolate(function() { return interpolate; });
+};
+
+pv.Layout.Arc.prototype = pv.extend(pv.Layout.Network)
+    .property("orient", String)
+    .property("directed", Boolean);
+
+/**
+ * Default properties for arc layouts. By default, the orientation is "bottom".
+ *
+ * @type pv.Layout.Arc
+ */
+pv.Layout.Arc.prototype.defaults = new pv.Layout.Arc()
+    .extend(pv.Layout.Network.prototype.defaults)
+    .orient("bottom");
+
+/**
+ * Specifies an optional sort function. The sort function follows the same
+ * comparator contract required by {@link pv.Dom.Node#sort}. Specifying a sort
+ * function provides an alternative to sort the nodes as they are specified by
+ * the <tt>nodes</tt> property; the main advantage of doing this is that the
+ * comparator function can access implicit fields populated by the network
+ * layout, such as the <tt>linkDegree</tt>.
+ *
+ * <p>Note that arc diagrams are particularly sensitive to order. This is
+ * referred to as the seriation problem, and many different techniques exist to
+ * find good node orders that emphasize clusters, such as spectral layout and
+ * simulated annealing.
+ *
+ * @param {function} f comparator function for nodes.
+ * @returns {pv.Layout.Arc} this.
+ */
+pv.Layout.Arc.prototype.sort = function(f) {
+  this.$sort = f;
+  return this;
+};
+
+/** @private Populates the x, y and angle attributes on the nodes. */
+pv.Layout.Arc.prototype.buildImplied = function(s) {
+  if (pv.Layout.Network.prototype.buildImplied.call(this, s)) return;
+
+  var nodes = s.nodes,
+      orient = s.orient,
+      sort = this.$sort,
+      index = pv.range(nodes.length),
+      w = s.width,
+      h = s.height,
+      r = Math.min(w, h) / 2;
+
+  /* Sort the nodes. */
+  if (sort) index.sort(function(a, b) { return sort(nodes[a], nodes[b]); });
+
+  /** @private Returns the mid-angle, given the breadth. */
+  function midAngle(b) {
+    switch (orient) {
+      case "top": return -Math.PI / 2;
+      case "bottom": return Math.PI / 2;
+      case "left": return Math.PI;
+      case "right": return 0;
+      case "radial": return (b - .25) * 2 * Math.PI;
+    }
+  }
+
+  /** @private Returns the x-position, given the breadth. */
+  function x(b) {
+    switch (orient) {
+      case "top":
+      case "bottom": return b * w;
+      case "left": return 0;
+      case "right": return w;
+      case "radial": return w / 2 + r * Math.cos(midAngle(b));
+    }
+  }
+
+  /** @private Returns the y-position, given the breadth. */
+  function y(b) {
+    switch (orient) {
+      case "top": return 0;
+      case "bottom": return h;
+      case "left":
+      case "right": return b * h;
+      case "radial": return h / 2 + r * Math.sin(midAngle(b));
+    }
+  }
+
+  /* Populate the x, y and mid-angle attributes. */
+  for (var i = 0; i < nodes.length; i++) {
+    var n = nodes[index[i]], b = n.breadth = (i + .5) / nodes.length;
+    n.x = x(b);
+    n.y = y(b);
+    n.midAngle = midAngle(b);
+  }
+};
+
+/**
+ * The orientation. The default orientation is "left", which means that nodes
+ * will be positioned from left-to-right in the order they are specified in the
+ * <tt>nodes</tt> property. The following orientations are supported:<ul>
+ *
+ * <li>left - left-to-right.
+ * <li>right - right-to-left.
+ * <li>top - top-to-bottom.
+ * <li>bottom - bottom-to-top.
+ * <li>radial - radially, starting at 12 o'clock and proceeding clockwise.</ul>
+ *
+ * @type string
+ * @name pv.Layout.Arc.prototype.orient
+ */
+
+/**
+ * Whether this arc digram is directed (bidirectional); only applies to
+ * non-radial orientations. By default, arc digrams are undirected, such that
+ * all arcs appear on one side. If the arc digram is directed, then forward
+ * links are drawn on the conventional side (the same as as undirected
+ * links--right, left, bottom and top for left, right, top and bottom,
+ * respectively), while reverse links are drawn on the opposite side.
+ *
+ * @type boolean
+ * @name pv.Layout.Arc.prototype.directed
+ */
+/**
+ * Constructs a new, empty horizon layout. Layouts are not typically constructed
+ * directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a horizon layout, which is a variation of a single-series
+ * area chart where the area is folded into multiple bands. Color is used to
+ * encode band, allowing the size of the chart to be reduced significantly
+ * without impeding readability. This layout algorithm is based on the work of
+ * J. Heer, N. Kong and M. Agrawala in <a
+ * href="http://hci.stanford.edu/publications/2009/heer-horizon-chi09.pdf">"Sizing
+ * the Horizon: The Effects of Chart Size and Layering on the Graphical
+ * Perception of Time Series Visualizations"</a>, CHI 2009.
+ *
+ * <p>This layout exports a single <tt>band</tt> mark prototype, which is
+ * intended to be used with an area mark. The band mark is contained in a panel
+ * which is replicated per band (and for negative/positive bands). For example,
+ * to create a simple horizon graph given an array of numbers:
+ *
+ * <pre>vis.add(pv.Layout.Horizon)
+ *     .bands(n)
+ *   .band.add(pv.Area)
+ *     .data(data)
+ *     .left(function() this.index * 35)
+ *     .height(function(d) d * 40);</pre>
+ *
+ * The layout can be further customized by changing the number of bands, and
+ * toggling whether the negative bands are mirrored or offset. (See the
+ * above-referenced paper for guidance.)
+ *
+ * <p>The <tt>fillStyle</tt> of the area can be overridden, though typically it
+ * is easier to customize the layout's behavior through the custom
+ * <tt>backgroundStyle</tt>, <tt>positiveStyle</tt> and <tt>negativeStyle</tt>
+ * properties. By default, the background is white, positive bands are blue, and
+ * negative bands are red. For the most accurate presentation, use fully-opaque
+ * colors of equal intensity for the negative and positive bands.
+ *
+ * @extends pv.Layout
+ */
+pv.Layout.Horizon = function() {
+  pv.Layout.call(this);
+  var that = this,
+      bands, // cached bands
+      mode, // cached mode
+      size, // cached height
+      fill, // cached background style
+      red, // cached negative color (ramp)
+      blue, // cached positive color (ramp)
+      buildImplied = this.buildImplied;
+
+  /** @private Cache the layout state to optimize properties. */
+  this.buildImplied = function(s) {
+    buildImplied.call(this, s);
+    bands = s.bands;
+    mode = s.mode;
+    size = Math.round((mode == "color" ? .5 : 1) * s.height);
+    fill = s.backgroundStyle;
+    red = pv.ramp(fill, s.negativeStyle).domain(0, bands);
+    blue = pv.ramp(fill, s.positiveStyle).domain(0, bands);
+  };
+
+  var bands = new pv.Panel()
+      .data(function() { return pv.range(bands * 2); })
+      .overflow("hidden")
+      .height(function() { return size; })
+      .top(function(i) { return mode == "color" ? (i & 1) * size : 0; })
+      .fillStyle(function(i) { return i ? null : fill; });
+
+  /**
+   * The band prototype. This prototype is intended to be used with an Area
+   * mark to render the horizon bands.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Horizon.prototype.band
+   */
+  this.band = new pv.Mark()
+      .top(function(d, i) {
+          return mode == "mirror" && i & 1
+              ? (i + 1 >> 1) * size
+              : null;
+        })
+      .bottom(function(d, i) {
+          return mode == "mirror"
+              ? (i & 1 ? null : (i + 1 >> 1) * -size)
+              : ((i & 1 || -1) * (i + 1 >> 1) * size);
+        })
+      .fillStyle(function(d, i) {
+          return (i & 1 ? red : blue)((i >> 1) + 1);
+        });
+
+  this.band.add = function(type) {
+    return that.add(pv.Panel).extend(bands).add(type).extend(this);
+  };
+};
+
+pv.Layout.Horizon.prototype = pv.extend(pv.Layout)
+    .property("bands", Number)
+    .property("mode", String)
+    .property("backgroundStyle", pv.color)
+    .property("positiveStyle", pv.color)
+    .property("negativeStyle", pv.color);
+
+/**
+ * Default properties for horizon layouts. By default, there are two bands, the
+ * mode is "offset", the background style is "white", the positive style is
+ * blue, negative style is red.
+ *
+ * @type pv.Layout.Horizon
+ */
+pv.Layout.Horizon.prototype.defaults = new pv.Layout.Horizon()
+    .extend(pv.Layout.prototype.defaults)
+    .bands(2)
+    .mode("offset")
+    .backgroundStyle("white")
+    .positiveStyle("#1f77b4")
+    .negativeStyle("#d62728");
+
+/**
+ * The horizon mode: offset, mirror, or color. The default is "offset".
+ *
+ * @type string
+ * @name pv.Layout.Horizon.prototype.mode
+ */
+
+/**
+ * The number of bands. Must be at least one. The default value is two.
+ *
+ * @type number
+ * @name pv.Layout.Horizon.prototype.bands
+ */
+
+/**
+ * The positive band color; if non-null, the interior of positive bands are
+ * filled with the specified color. The default value of this property is blue.
+ * For accurate blending, this color should be fully opaque.
+ *
+ * @type pv.Color
+ * @name pv.Layout.Horizon.prototype.positiveStyle
+ */
+
+/**
+ * The negative band color; if non-null, the interior of negative bands are
+ * filled with the specified color. The default value of this property is red.
+ * For accurate blending, this color should be fully opaque.
+ *
+ * @type pv.Color
+ * @name pv.Layout.Horizon.prototype.negativeStyle
+ */
+
+/**
+ * The background color. The panel background is filled with the specified
+ * color, and the negative and positive bands are filled with an interpolated
+ * color between this color and the respective band color. The default value of
+ * this property is white. For accurate blending, this color should be fully
+ * opaque.
+ *
+ * @type pv.Color
+ * @name pv.Layout.Horizon.prototype.backgroundStyle
+ */
+/**
+ * Constructs a new, empty rollup network layout. Layouts are not typically
+ * constructed directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a network visualization using a node-link diagram where
+ * nodes are rolled up along two dimensions. This implementation is based on the
+ * "PivotGraph" designed by Martin Wattenberg:
+ *
+ * <blockquote>The method is designed for graphs that are "multivariate", i.e.,
+ * where each node is associated with several attributes. Unlike visualizations
+ * which emphasize global graph topology, PivotGraph uses a simple grid-based
+ * approach to focus on the relationship between node attributes &amp;
+ * connections.</blockquote>
+ *
+ * This layout requires two psuedo-properties to be specified, which assign node
+ * positions along the two dimensions {@link #x} and {@link #y}, corresponding
+ * to the left and top properties, respectively. Typically, these functions are
+ * specified using an {@link pv.Scale.ordinal}. Nodes that share the same
+ * position in <i>x</i> and <i>y</i> are "rolled up" into a meta-node, and
+ * similarly links are aggregated between meta-nodes. For example, to construct
+ * a rollup to analyze links by gender and affiliation, first define two ordinal
+ * scales:
+ *
+ * <pre>var x = pv.Scale.ordinal(nodes, function(d) d.gender).split(0, w),
+ *     y = pv.Scale.ordinal(nodes, function(d) d.aff).split(0, h);</pre>
+ *
+ * Next, define the position psuedo-properties:
+ *
+ * <pre>    .x(function(d) x(d.gender))
+ *     .y(function(d) y(d.aff))</pre>
+ *
+ * Linear and other quantitative scales can alternatively be used to position
+ * the nodes along either dimension. Note, however, that the rollup requires
+ * that the positions match exactly, and thus ordinal scales are recommended to
+ * avoid precision errors.
+ *
+ * <p>Note that because this layout provides a visualization of the rolled up
+ * graph, the data properties for the mark prototypes (<tt>node</tt>,
+ * <tt>link</tt> and <tt>label</tt>) are different from most other network
+ * layouts: they reference the rolled-up nodes and links, rather than the nodes
+ * and links of the full network. The underlying nodes and links for each
+ * rolled-up node and link can be accessed via the <tt>nodes</tt> and
+ * <tt>links</tt> attributes, respectively. The aggregated link values for
+ * rolled-up links can similarly be accessed via the <tt>linkValue</tt>
+ * attribute.
+ *
+ * <p>For undirected networks, links are duplicated in both directions. For
+ * directed networks, use <tt>directed(true)</tt>. The graph is assumed to be
+ * undirected by default.
+ *
+ * @extends pv.Layout.Network
+ * @see <a href="http://www.research.ibm.com/visual/papers/pivotgraph.pdf"
+ * >"Visual Exploration of Multivariate Graphs"</a> by M. Wattenberg, CHI 2006.
+ */
+pv.Layout.Rollup = function() {
+  pv.Layout.Network.call(this);
+  var that = this,
+      nodes, // cached rollup nodes
+      links, // cached rollup links
+      buildImplied = that.buildImplied;
+
+  /** @private Cache layout state to optimize properties. */
+  this.buildImplied = function(s) {
+    buildImplied.call(this, s);
+    nodes = s.$rollup.nodes;
+    links = s.$rollup.links;
+  };
+
+  /* Render rollup nodes. */
+  this.node
+      .data(function() { return nodes; })
+      .size(function(d) { return d.nodes.length * 20; });
+
+  /* Render rollup links. */
+  this.link
+      .interpolate("polar")
+      .eccentricity(.8);
+
+  this.link.add = function(type) {
+    return that.add(pv.Panel)
+        .data(function() { return links; })
+      .add(type)
+        .extend(this);
+  };
+};
+
+pv.Layout.Rollup.prototype = pv.extend(pv.Layout.Network)
+    .property("directed", Boolean);
+
+/**
+ * Whether the underlying network is directed. By default, the graph is assumed
+ * to be undirected, and links are rendered in both directions. If the network
+ * is directed, then forward links are drawn above the diagonal, while reverse
+ * links are drawn below.
+ *
+ * @type boolean
+ * @name pv.Layout.Rollup.prototype.directed
+ */
+
+/**
+ * Specifies the <i>x</i>-position function used to rollup nodes. The rolled up
+ * nodes are positioned horizontally using the return values from the given
+ * function. Typically the function is specified as an ordinal scale. For
+ * single-dimension rollups, a constant value can be specified.
+ *
+ * @param {function} f the <i>x</i>-position function.
+ * @returns {pv.Layout.Rollup} this.
+ * @see pv.Scale.ordinal
+ */
+pv.Layout.Rollup.prototype.x = function(f) {
+  this.$x = pv.functor(f);
+  return this;
+};
+
+/**
+ * Specifies the <i>y</i>-position function used to rollup nodes. The rolled up
+ * nodes are positioned vertically using the return values from the given
+ * function. Typically the function is specified as an ordinal scale. For
+ * single-dimension rollups, a constant value can be specified.
+ *
+ * @param {function} f the <i>y</i>-position function.
+ * @returns {pv.Layout.Rollup} this.
+ * @see pv.Scale.ordinal
+ */
+pv.Layout.Rollup.prototype.y = function(f) {
+  this.$y = pv.functor(f);
+  return this;
+};
+
+/** @private */
+pv.Layout.Rollup.prototype.buildImplied = function(s) {
+  if (pv.Layout.Network.prototype.buildImplied.call(this, s)) return;
+
+  var nodes = s.nodes,
+      links = s.links,
+      directed = s.directed,
+      n = nodes.length,
+      x = [],
+      y = [],
+      rnindex = 0,
+      rnodes = {},
+      rlinks = {};
+
+  /** @private */
+  function id(i) {
+    return x[i] + "," + y[i];
+  }
+
+  /* Iterate over the data, evaluating the x and y functions. */
+  var stack = pv.Mark.stack, o = {parent: this};
+  stack.unshift(null);
+  for (var i = 0; i < n; i++) {
+    o.index = i;
+    stack[0] = nodes[i];
+    x[i] = this.$x.apply(o, stack);
+    y[i] = this.$y.apply(o, stack);
+  }
+  stack.shift();
+
+  /* Compute rollup nodes. */
+  for (var i = 0; i < nodes.length; i++) {
+    var nodeId = id(i),
+        rn = rnodes[nodeId];
+    if (!rn) {
+      rn = rnodes[nodeId] = pv.extend(nodes[i]);
+      rn.index = rnindex++;
+      rn.x = x[i];
+      rn.y = y[i];
+      rn.nodes = [];
+    }
+    rn.nodes.push(nodes[i]);
+  }
+
+  /* Compute rollup links. */
+  for (var i = 0; i < links.length; i++) {
+    var source = links[i].sourceNode,
+        target = links[i].targetNode,
+        rsource = rnodes[id(source.index)],
+        rtarget = rnodes[id(target.index)],
+        reverse = !directed && rsource.index > rtarget.index,
+        linkId = reverse
+            ? rtarget.index + "," + rsource.index
+            : rsource.index + "," + rtarget.index,
+        rl = rlinks[linkId];
+    if (!rl) {
+      rl = rlinks[linkId] = {
+        sourceNode: rsource,
+        targetNode: rtarget,
+        linkValue: 0,
+        links: []
+      };
+    }
+    rl.links.push(links[i]);
+    rl.linkValue += links[i].linkValue;
+  }
+
+  /* Export the rolled up nodes and links to the scene. */
+  s.$rollup = {
+    nodes: pv.values(rnodes),
+    links: pv.values(rlinks)
+  };
+};
+/**
+ * Constructs a new, empty matrix network layout. Layouts are not typically
+ * constructed directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class Implements a network visualization using a matrix view. This is, in
+ * effect, a visualization of the graph's <i>adjacency matrix</i>: the cell at
+ * row <i>i</i>, column <i>j</i>, corresponds to the link from node <i>i</i> to
+ * node <i>j</i>. The fill color of each cell is binary by default, and
+ * corresponds to whether a link exists between the two nodes. If the underlying
+ * graph has links with variable values, the <tt>fillStyle</tt> property can be
+ * substited to use an appropriate color function, such as {@link pv.ramp}.
+ *
+ * <p>For undirected networks, the matrix is symmetric around the diagonal. For
+ * directed networks, links in opposite directions can be rendered on opposite
+ * sides of the diagonal using <tt>directed(true)</tt>. The graph is assumed to
+ * be undirected by default.
+ *
+ * <p>The mark prototypes for this network layout are slightly different than
+ * other implementations:<ul>
+ *
+ * <li><tt>node</tt> - unsupported; undefined. No mark is needed to visualize
+ * nodes directly, as the nodes are implicit in the location (rows and columns)
+ * of the links.
+ *
+ * <p><li><tt>link</tt> - for rendering links; typically a {@link pv.Bar}. The
+ * link mark is added directly to the layout, with the data property defined as
+ * all possible pairs of nodes. Each pair is represented as a
+ * {@link pv.Network.Layout.Link}, though the <tt>linkValue</tt> attribute may
+ * be 0 if no link exists in the graph.
+ *
+ * <p><li><tt>label</tt> - for rendering node labels; typically a
+ * {@link pv.Label}. The label mark is added directly to the layout, with the
+ * data property defined via the layout's <tt>nodes</tt> property; note,
+ * however, that the nodes are duplicated so as to provide a label across the
+ * top and down the side. Properties such as <tt>strokeStyle</tt> and
+ * <tt>fillStyle</tt> can be overridden to compute properties from node data
+ * dynamically.
+ *
+ * </ul>For more details on how to use this layout, see
+ * {@link pv.Layout.Network}.
+ *
+ * @extends pv.Layout.Network
+ */
+pv.Layout.Matrix = function() {
+  pv.Layout.Network.call(this);
+  var that = this,
+      n, // cached matrix size
+      dx, // cached cell width
+      dy, // cached cell height
+      labels, // cached labels (array of strings)
+      pairs, // cached pairs (array of links)
+      buildImplied = that.buildImplied;
+
+  /** @private Cache layout state to optimize properties. */
+  this.buildImplied = function(s) {
+    buildImplied.call(this, s);
+    n = s.nodes.length;
+    dx = s.width / n;
+    dy = s.height / n;
+    labels = s.$matrix.labels;
+    pairs = s.$matrix.pairs;
+  };
+
+  /* Links are all pairs of nodes. */
+  this.link
+      .data(function() { return pairs; })
+      .left(function() { return dx * (this.index % n); })
+      .top(function() { return dy * Math.floor(this.index / n); })
+      .width(function() { return dx; })
+      .height(function() { return dy; })
+      .lineWidth(1.5)
+      .strokeStyle("#fff")
+      .fillStyle(function(l) { return l.linkValue ? "#555" : "#eee"; })
+      .parent = this;
+
+  /* No special add for links! */
+  delete this.link.add;
+
+  /* Labels are duplicated for top & left. */
+  this.label
+      .data(function() { return labels; })
+      .left(function() { return this.index & 1 ? dx * ((this.index >> 1) + .5) : 0; })
+      .top(function() { return this.index & 1 ? 0 : dy * ((this.index >> 1) + .5); })
+      .textMargin(4)
+      .textAlign(function() { return this.index & 1 ? "left" : "right"; })
+      .textAngle(function() { return this.index & 1 ? -Math.PI / 2 : 0; });
+
+  /* The node mark is unused. */
+  delete this.node;
+};
+
+pv.Layout.Matrix.prototype = pv.extend(pv.Layout.Network)
+    .property("directed", Boolean);
+
+/**
+ * Whether this matrix visualization is directed (bidirectional). By default,
+ * the graph is assumed to be undirected, such that the visualization is
+ * symmetric across the matrix diagonal. If the network is directed, then
+ * forward links are drawn above the diagonal, while reverse links are drawn
+ * below.
+ *
+ * @type boolean
+ * @name pv.Layout.Matrix.prototype.directed
+ */
+
+/**
+ * Specifies an optional sort function. The sort function follows the same
+ * comparator contract required by {@link pv.Dom.Node#sort}. Specifying a sort
+ * function provides an alternative to sort the nodes as they are specified by
+ * the <tt>nodes</tt> property; the main advantage of doing this is that the
+ * comparator function can access implicit fields populated by the network
+ * layout, such as the <tt>linkDegree</tt>.
+ *
+ * <p>Note that matrix visualizations are particularly sensitive to order. This
+ * is referred to as the seriation problem, and many different techniques exist
+ * to find good node orders that emphasize clusters, such as spectral layout and
+ * simulated annealing.
+ *
+ * @param {function} f comparator function for nodes.
+ * @returns {pv.Layout.Matrix} this.
+ */
+pv.Layout.Matrix.prototype.sort = function(f) {
+  this.$sort = f;
+  return this;
+};
+
+/** @private */
+pv.Layout.Matrix.prototype.buildImplied = function(s) {
+  if (pv.Layout.Network.prototype.buildImplied.call(this, s)) return;
+
+  var nodes = s.nodes,
+      links = s.links,
+      sort = this.$sort,
+      n = nodes.length,
+      index = pv.range(n),
+      labels = [],
+      pairs = [],
+      map = {};
+
+  s.$matrix = {labels: labels, pairs: pairs};
+
+  /* Sort the nodes. */
+  if (sort) index.sort(function(a, b) { return sort(nodes[a], nodes[b]); });
+
+  /* Create pairs. */
+  for (var i = 0; i < n; i++) {
+    for (var j = 0; j < n; j++) {
+      var a = index[i],
+          b = index[j],
+          p = {
+            row: i,
+            col: j,
+            sourceNode: nodes[a],
+            targetNode: nodes[b],
+            linkValue: 0
+          };
+      pairs.push(map[a + "." + b] = p);
+    }
+  }
+
+  /* Create labels. */
+  for (var i = 0; i < n; i++) {
+    var a = index[i];
+    labels.push(nodes[a], nodes[a]);
+  }
+
+  /* Accumulate link values. */
+  for (var i = 0; i < links.length; i++) {
+    var l = links[i],
+        source = l.sourceNode.index,
+        target = l.targetNode.index,
+        value = l.linkValue;
+    map[source + "." + target].linkValue += value;
+    if (!s.directed) map[target + "." + source].linkValue += value;
+  }
+};
+// ranges (bad, satisfactory, good)
+// measures (actual, forecast)
+// markers (previous, goal)
+
+/*
+ * Chart design based on the recommendations of Stephen Few. Implementation
+ * based on the work of Clint Ivy, Jamie Love, and Jason Davies.
+ * http://projects.instantcognition.com/protovis/bulletchart/
+ */
+
+/**
+ * Constructs a new, empty bullet layout. Layouts are not typically constructed
+ * directly; instead, they are added to an existing panel via
+ * {@link pv.Mark#add}.
+ *
+ * @class
+ * @extends pv.Layout
+ */
+pv.Layout.Bullet = function() {
+  pv.Layout.call(this);
+  var that = this,
+      buildImplied = that.buildImplied,
+      scale = that.x = pv.Scale.linear(),
+      orient,
+      horizontal,
+      rangeColor,
+      measureColor,
+      x;
+
+  /** @private Cache layout state to optimize properties. */
+  this.buildImplied = function(s) {
+    buildImplied.call(this, x = s);
+    orient = s.orient;
+    horizontal = /^left|right$/.test(orient);
+    rangeColor = pv.ramp("#bbb", "#eee")
+        .domain(0, Math.max(1, x.ranges.length - 1));
+    measureColor = pv.ramp("steelblue", "lightsteelblue")
+        .domain(0, Math.max(1, x.measures.length - 1));
+  };
+
+  /**
+   * The range prototype.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Bullet.prototype.range
+   */
+  (this.range = new pv.Mark())
+      .data(function() { return x.ranges; })
+      .reverse(true)
+      .left(function() { return orient == "left" ? 0 : null; })
+      .top(function() { return orient == "top" ? 0 : null; })
+      .right(function() { return orient == "right" ? 0 : null; })
+      .bottom(function() { return orient == "bottom" ? 0 : null; })
+      .width(function(d) { return horizontal ? scale(d) : null; })
+      .height(function(d) { return horizontal ? null : scale(d); })
+      .fillStyle(function() { return rangeColor(this.index); })
+      .antialias(false)
+      .parent = that;
+
+  /**
+   * The measure prototype.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Bullet.prototype.measure
+   */
+  (this.measure = new pv.Mark())
+      .extend(this.range)
+      .data(function() { return x.measures; })
+      .left(function() { return orient == "left" ? 0 : horizontal ? null : this.parent.width() / 3.25; })
+      .top(function() { return orient == "top" ? 0 : horizontal ? this.parent.height() / 3.25 : null; })
+      .right(function() { return orient == "right" ? 0 : horizontal ? null : this.parent.width() / 3.25; })
+      .bottom(function() { return orient == "bottom" ? 0 : horizontal ? this.parent.height() / 3.25 : null; })
+      .fillStyle(function() { return measureColor(this.index); })
+      .parent = that;
+
+  /**
+   * The marker prototype.
+   *
+   * @type pv.Mark
+   * @name pv.Layout.Bullet.prototype.marker
+   */
+  (this.marker = new pv.Mark())
+      .data(function() { return x.markers; })
+      .left(function(d) { return orient == "left" ? scale(d) : horizontal ? null : this.parent.width() / 2; })
+      .top(function(d) { return orient == "top" ? scale(d) : horizontal ? this.parent.height() / 2 : null; })
+      .right(function(d) { return orient == "right" ? scale(d) : null; })
+      .bottom(function(d) { return orient == "bottom" ? scale(d) : null; })
+      .strokeStyle("black")
+      .shape("bar")
+      .angle(function() { return horizontal ? 0 : Math.PI / 2; })
+      .parent = that;
+
+  (this.tick = new pv.Mark())
+      .data(function() { return scale.ticks(7); })
+      .left(function(d) { return orient == "left" ? scale(d) : null; })
+      .top(function(d) { return orient == "top" ? scale(d) : null; })
+      .right(function(d) { return orient == "right" ? scale(d) : horizontal ? null : -6; })
+      .bottom(function(d) { return orient == "bottom" ? scale(d) : horizontal ? -8 : null; })
+      .height(function() { return horizontal ? 6 : null; })
+      .width(function() { return horizontal ? null : 6; })
+      .parent = that;
+};
+
+pv.Layout.Bullet.prototype = pv.extend(pv.Layout)
+    .property("orient", String) // left, right, top, bottom
+    .property("ranges")
+    .property("markers")
+    .property("measures")
+    .property("maximum", Number);
+
+/**
+ * Default properties for bullet layouts.
+ *
+ * @type pv.Layout.Bullet
+ */
+pv.Layout.Bullet.prototype.defaults = new pv.Layout.Bullet()
+    .extend(pv.Layout.prototype.defaults)
+    .orient("left")
+    .ranges([])
+    .markers([])
+    .measures([]);
+
+/**
+ * The orientation.
+ *
+ * @type string
+ * @name pv.Layout.Bullet.prototype.orient
+ */
+
+/**
+ * The array of range values.
+ *
+ * @type array
+ * @name pv.Layout.Bullet.prototype.ranges
+ */
+
+/**
+ * The array of marker values.
+ *
+ * @type array
+ * @name pv.Layout.Bullet.prototype.markers
+ */
+
+/**
+ * The array of measure values.
+ *
+ * @type array
+ * @name pv.Layout.Bullet.prototype.measures
+ */
+
+/**
+ * Optional; the maximum range value.
+ *
+ * @type number
+ * @name pv.Layout.Bullet.prototype.maximum
+ */
+
+/** @private */
+pv.Layout.Bullet.prototype.buildImplied = function(s) {
+  pv.Layout.prototype.buildImplied.call(this, s);
+  var size = this.parent[/^left|right$/.test(s.orient) ? "width" : "height"]();
+  s.maximum = s.maximum || pv.max([].concat(s.ranges, s.markers, s.measures));
+  this.x.domain(0, s.maximum).range(0, size);
+};
+/**
+ * Abstract; see an implementing class for details.
+ *
+ * @class Represents a reusable interaction; applies an interactive behavior to
+ * a given mark. Behaviors are themselves functions designed to be used as event
+ * handlers. For example, to add pan and zoom support to any panel, say:
+ *
+ * <pre>    .event("mousedown", pv.Behavior.pan())
+ *     .event("mousewheel", pv.Behavior.zoom())</pre>
+ *
+ * The behavior should be registered on the event that triggers the start of the
+ * behavior. Typically, the behavior will take care of registering for any
+ * additional events that are necessary. For example, dragging starts on
+ * mousedown, while the drag behavior automatically listens for mousemove and
+ * mouseup events on the window. By listening to the window, the behavior can
+ * continue to receive mouse events even if the mouse briefly leaves the mark
+ * being dragged, or even the root panel.
+ *
+ * <p>Each behavior implementation has specific requirements as to which events
+ * it supports, and how it should be used. For example, the drag behavior
+ * requires that the data associated with the mark be an object with <tt>x</tt>
+ * and <tt>y</tt> attributes, such as a {@link pv.Vector}, storing the mark's
+ * position. See an implementing class for details.
+ *
+ * @see pv.Behavior.drag
+ * @see pv.Behavior.pan
+ * @see pv.Behavior.point
+ * @see pv.Behavior.select
+ * @see pv.Behavior.zoom
+ * @extends function
+ */
+pv.Behavior = {};
+/**
+ * Returns a new drag behavior to be registered on mousedown events.
+ *
+ * @class Implements interactive dragging starting with mousedown events.
+ * Register this behavior on marks that should be draggable by the user, such as
+ * the selected region for brushing and linking. This behavior can be used in
+ * tandom with {@link pv.Behavior.select} to allow the selected region to be
+ * dragged interactively.
+ *
+ * <p>After the initial mousedown event is triggered, this behavior listens for
+ * mousemove and mouseup events on the window. This allows dragging to continue
+ * even if the mouse temporarily leaves the mark that is being dragged, or even
+ * the root panel.
+ *
+ * <p>This behavior requires that the data associated with the mark being
+ * dragged have <tt>x</tt> and <tt>y</tt> attributes that correspond to the
+ * mark's location in pixels. The mark's positional properties are not set
+ * directly by this behavior; instead, the positional properties should be
+ * defined as:
+ *
+ * <pre>    .left(function(d) d.x)
+ *     .top(function(d) d.y)</pre>
+ *
+ * Thus, the behavior does not move the mark directly, but instead updates the
+ * mark position by updating the underlying data. Note that if the positional
+ * properties are defined with bottom and right (rather than top and left), the
+ * drag behavior will be inverted, which will confuse users!
+ *
+ * <p>The drag behavior is bounded by the parent panel; the <tt>x</tt> and
+ * <tt>y</tt> attributes are clamped such that the mark being dragged does not
+ * extend outside the enclosing panel's bounds. To facilitate this, the drag
+ * behavior also queries for <tt>dx</tt> and <tt>dy</tt> attributes on the
+ * underlying data, to determine the dimensions of the bar being dragged. For
+ * non-rectangular marks, the drag behavior simply treats the mark as a point,
+ * which means that only the mark's center is bounded.
+ *
+ * <p>The mark being dragged is automatically re-rendered for each mouse event
+ * as part of the drag operation. In addition, a <tt>fix</tt> attribute is
+ * populated on the mark, which allows visual feedback for dragging. For
+ * example, to change the mark fill color while dragging:
+ *
+ * <pre>    .fillStyle(function(d) d.fix ? "#ff7f0e" : "#aec7e8")</pre>
+ *
+ * In some cases, such as with network layouts, dragging the mark may cause
+ * related marks to change, in which case additional marks may also need to be
+ * rendered. This can be accomplished by listening for the drag
+ * psuedo-events:<ul>
+ *
+ * <li>dragstart (on mousedown)
+ * <li>drag (on mousemove)
+ * <li>dragend (on mouseup)
+ *
+ * </ul>For example, to render the parent panel while dragging, thus
+ * re-rendering all sibling marks:
+ *
+ * <pre>    .event("mousedown", pv.Behavior.drag())
+ *     .event("drag", function() this.parent)</pre>
+ *
+ * This behavior may be enhanced in the future to allow more flexible
+ * configuration of drag behavior.
+ *
+ * @extends pv.Behavior
+ * @see pv.Behavior
+ * @see pv.Behavior.select
+ * @see pv.Layout.force
+ */
+pv.Behavior.drag = function() {
+  var scene, // scene context
+      index, // scene context
+      p, // particle being dragged
+      v1, // initial mouse-particle offset
+      max;
+
+  /** @private */
+  function mousedown(d) {
+    index = this.index;
+    scene = this.scene;
+    var m = this.mouse();
+    v1 = ((p = d).fix = pv.vector(d.x, d.y)).minus(m);
+    max = {
+      x: this.parent.width() - (d.dx || 0),
+      y: this.parent.height() - (d.dy || 0)
+    };
+    scene.mark.context(scene, index, function() { this.render(); });
+    pv.Mark.dispatch("dragstart", scene, index);
+  }
+
+  /** @private */
+  function mousemove() {
+    if (!scene) return;
+    scene.mark.context(scene, index, function() {
+        var m = this.mouse();
+        p.x = p.fix.x = Math.max(0, Math.min(v1.x + m.x, max.x));
+        p.y = p.fix.y = Math.max(0, Math.min(v1.y + m.y, max.y));
+        this.render();
+      });
+    pv.Mark.dispatch("drag", scene, index);
+  }
+
+  /** @private */
+  function mouseup() {
+    if (!scene) return;
+    p.fix = null;
+    scene.mark.context(scene, index, function() { this.render(); });
+    pv.Mark.dispatch("dragend", scene, index);
+    scene = null;
+  }
+
+  pv.listen(window, "mousemove", mousemove);
+  pv.listen(window, "mouseup", mouseup);
+  return mousedown;
+};
+/**
+ * Returns a new point behavior to be registered on mousemove events.
+ *
+ * @class Implements interactive fuzzy pointing, identifying marks that are in
+ * close proximity to the mouse cursor. This behavior is an alternative to the
+ * native mouseover and mouseout events, improving usability. Rather than
+ * requiring the user to mouseover a mark exactly, the mouse simply needs to
+ * move near the given mark and a "point" event is triggered. In addition, if
+ * multiple marks overlap, the point behavior can be used to identify the mark
+ * instance closest to the cursor, as opposed to the one that is rendered on
+ * top.
+ *
+ * <p>The point behavior can also identify the closest mark instance for marks
+ * that produce a continuous graphic primitive. The point behavior can thus be
+ * used to provide details-on-demand for both discrete marks (such as dots and
+ * bars), as well as continuous marks (such as lines and areas).
+ *
+ * <p>This behavior is implemented by finding the closest mark instance to the
+ * mouse cursor on every mousemove event. If this closest mark is within the
+ * given radius threshold, which defaults to 30 pixels, a "point" psuedo-event
+ * is dispatched to the given mark instance. If any mark were previously
+ * pointed, it would receive a corresponding "unpoint" event. These two
+ * psuedo-event types correspond to the native "mouseover" and "mouseout"
+ * events, respectively. To increase the radius at which the point behavior can
+ * be applied, specify an appropriate threshold to the constructor, up to
+ * <tt>Infinity</tt>.
+ *
+ * <p>By default, the standard Cartesian distance is computed. However, with
+ * some visualizations it is desirable to consider only a single dimension, such
+ * as the <i>x</i>-dimension for an independent variable. In this case, the
+ * collapse parameter can be set to collapse the <i>y</i> dimension:
+ *
+ * <pre>    .event("mousemove", pv.Behavior.point(Infinity).collapse("y"))</pre>
+ *
+ * <p>This behavior only listens to mousemove events on the assigned panel,
+ * which is typically the root panel. The behavior will search recursively for
+ * descendant marks to point. If the mouse leaves the assigned panel, the
+ * behavior no longer receives mousemove events; an unpoint psuedo-event is
+ * automatically dispatched to unpoint any pointed mark. Marks may be re-pointed
+ * when the mouse reenters the panel.
+ *
+ * <p>Panels have transparent fill styles by default; this means that panels may
+ * not receive the initial mousemove event to start pointing. To fix this
+ * problem, either given the panel a visible fill style (such as "white"), or
+ * set the <tt>events</tt> property to "all" such that the panel receives events
+ * despite its transparent fill.
+ *
+ * <p>Note: this behavior does not currently wedge marks.
+ *
+ * @extends pv.Behavior
+ *
+ * @param {number} [r] the fuzzy radius threshold in pixels
+ * @see <a href="http://www.tovigrossman.com/papers/chi2005bubblecursor.pdf"
+ * >"The Bubble Cursor: Enhancing Target Acquisition by Dynamic Resizing of the
+ * Cursor's Activation Area"</a> by T. Grossman &amp; R. Balakrishnan, CHI 2005.
+ */
+pv.Behavior.point = function(r) {
+  var unpoint, // the current pointer target
+      collapse = null, // dimensions to collapse
+      kx = 1, // x-dimension cost scale
+      ky = 1, // y-dimension cost scale
+      r2 = arguments.length ? r * r : 900; // fuzzy radius
+
+  /** @private Search for the mark closest to the mouse. */
+  function search(scene, index) {
+    var s = scene[index],
+        point = {cost: Infinity};
+    for (var i = 0, n = s.visible && s.children.length; i < n; i++) {
+      var child = s.children[i], mark = child.mark, p;
+      if (mark.type == "panel") {
+        mark.scene = child;
+        for (var j = 0, m = child.length; j < m; j++) {
+          mark.index = j;
+          p = search(child, j);
+          if (p.cost < point.cost) point = p;
+        }
+        delete mark.scene;
+        delete mark.index;
+      } else if (mark.$handlers.point) {
+        var v = mark.mouse();
+        for (var j = 0, m = child.length; j < m; j++) {
+          var c = child[j],
+              dx = v.x - c.left - (c.width || 0) / 2,
+              dy = v.y - c.top - (c.height || 0) / 2,
+              dd = kx * dx * dx + ky * dy * dy;
+          if (dd < point.cost) {
+            point.distance = dx * dx + dy * dy;
+            point.cost = dd;
+            point.scene = child;
+            point.index = j;
+          }
+        }
+      }
+    }
+    return point;
+  }
+
+  /** @private */
+  function mousemove() {
+    /* If the closest mark is far away, clear the current target. */
+    var point = search(this.scene, this.index);
+    if ((point.cost == Infinity) || (point.distance > r2)) point = null;
+
+    /* Unpoint the old target, if it's not the new target. */
+    if (unpoint) {
+      if (point
+          && (unpoint.scene == point.scene)
+          && (unpoint.index == point.index)) return;
+      pv.Mark.dispatch("unpoint", unpoint.scene, unpoint.index);
+    }
+
+    /* Point the new target, if there is one. */
+    if (unpoint = point) {
+      pv.Mark.dispatch("point", point.scene, point.index);
+
+      /* Unpoint when the mouse leaves the root panel. */
+      pv.listen(this.root.canvas(), "mouseout", mouseout);
+    }
+  }
+
+  /** @private */
+  function mouseout(e) {
+    if (unpoint && !pv.ancestor(this, e.relatedTarget)) {
+      pv.Mark.dispatch("unpoint", unpoint.scene, unpoint.index);
+      unpoint = null;
+    }
+  }
+
+  /**
+   * Sets or gets the collapse parameter. By default, the standard Cartesian
+   * distance is computed. However, with some visualizations it is desirable to
+   * consider only a single dimension, such as the <i>x</i>-dimension for an
+   * independent variable. In this case, the collapse parameter can be set to
+   * collapse the <i>y</i> dimension:
+   *
+   * <pre>    .event("mousemove", pv.Behavior.point(Infinity).collapse("y"))</pre>
+   *
+   * @function
+   * @returns {pv.Behavior.point} this, or the current collapse parameter.
+   * @name pv.Behavior.point.prototype.collapse
+   * @param {string} [x] the new collapse parameter
+   */
+  mousemove.collapse = function(x) {
+    if (arguments.length) {
+      collapse = String(x);
+      switch (collapse) {
+        case "y": kx = 1; ky = 0; break;
+        case "x": kx = 0; ky = 1; break;
+        default: kx = 1; ky = 1; break;
+      }
+      return mousemove;
+    }
+    return collapse;
+  };
+
+  return mousemove;
+};
+/**
+ * Returns a new select behavior to be registered on mousedown events.
+ *
+ * @class Implements interactive selecting starting with mousedown events.
+ * Register this behavior on panels that should be selectable by the user, such
+ * for brushing and linking. This behavior can be used in tandom with
+ * {@link pv.Behavior.drag} to allow the selected region to be dragged
+ * interactively.
+ *
+ * <p>After the initial mousedown event is triggered, this behavior listens for
+ * mousemove and mouseup events on the window. This allows selecting to continue
+ * even if the mouse temporarily leaves the assigned panel, or even the root
+ * panel.
+ *
+ * <p>This behavior requires that the data associated with the mark being
+ * dragged have <tt>x</tt>, <tt>y</tt>, <tt>dx</tt> and <tt>dy</tt> attributes
+ * that correspond to the mark's location and dimensions in pixels. The mark's
+ * positional properties are not set directly by this behavior; instead, the
+ * positional properties should be defined as:
+ *
+ * <pre>    .left(function(d) d.x)
+ *     .top(function(d) d.y)
+ *     .width(function(d) d.dx)
+ *     .height(function(d) d.dy)</pre>
+ *
+ * Thus, the behavior does not resize the mark directly, but instead updates the
+ * selection by updating the assigned panel's underlying data. Note that if the
+ * positional properties are defined with bottom and right (rather than top and
+ * left), the drag behavior will be inverted, which will confuse users!
+ *
+ * <p>The select behavior is bounded by the assigned panel; the positional
+ * attributes are clamped such that the selection does not extend outside the
+ * panel's bounds.
+ *
+ * <p>The panel being selected is automatically re-rendered for each mouse event
+ * as part of the drag operation. This behavior may be enhanced in the future to
+ * allow more flexible configuration of select behavior. In some cases, such as
+ * with parallel coordinates, making a selection may cause related marks to
+ * change, in which case additional marks may also need to be rendered. This can
+ * be accomplished by listening for the select psuedo-events:<ul>
+ *
+ * <li>selectstart (on mousedown)
+ * <li>select (on mousemove)
+ * <li>selectend (on mouseup)
+ *
+ * </ul>For example, to render the parent panel while selecting, thus
+ * re-rendering all sibling marks:
+ *
+ * <pre>    .event("mousedown", pv.Behavior.drag())
+ *     .event("select", function() this.parent)</pre>
+ *
+ * This behavior may be enhanced in the future to allow more flexible
+ * configuration of the selection behavior.
+ *
+ * @extends pv.Behavior
+ * @see pv.Behavior.drag
+ */
+pv.Behavior.select = function() {
+  var scene, // scene context
+      index, // scene context
+      r, // region being selected
+      m1; // initial mouse position
+
+  /** @private */
+  function mousedown(d) {
+    index = this.index;
+    scene = this.scene;
+    m1 = this.mouse();
+    r = d;
+    r.x = m1.x;
+    r.y = m1.y;
+    r.dx = r.dy = 0;
+    pv.Mark.dispatch("selectstart", scene, index);
+  }
+
+  /** @private */
+  function mousemove() {
+    if (!scene) return;
+    scene.mark.context(scene, index, function() {
+        var m2 = this.mouse();
+        r.x = Math.max(0, Math.min(m1.x, m2.x));
+        r.y = Math.max(0, Math.min(m1.y, m2.y));
+        r.dx = Math.min(this.width(), Math.max(m2.x, m1.x)) - r.x;
+        r.dy = Math.min(this.height(), Math.max(m2.y, m1.y)) - r.y;
+        this.render();
+      });
+    pv.Mark.dispatch("select", scene, index);
+  }
+
+  /** @private */
+  function mouseup() {
+    if (!scene) return;
+    pv.Mark.dispatch("selectend", scene, index);
+    scene = null;
+  }
+
+  pv.listen(window, "mousemove", mousemove);
+  pv.listen(window, "mouseup", mouseup);
+  return mousedown;
+};
+/**
+ * Returns a new resize behavior to be registered on mousedown events.
+ *
+ * @class Implements interactive resizing of a selection starting with mousedown
+ * events. Register this behavior on selection handles that should be resizeable
+ * by the user, such for brushing and linking. This behavior can be used in
+ * tandom with {@link pv.Behavior.select} and {@link pv.Behavior.drag} to allow
+ * the selected region to be selected and dragged interactively.
+ *
+ * <p>After the initial mousedown event is triggered, this behavior listens for
+ * mousemove and mouseup events on the window. This allows resizing to continue
+ * even if the mouse temporarily leaves the assigned panel, or even the root
+ * panel.
+ *
+ * <p>This behavior requires that the data associated with the mark being
+ * resized have <tt>x</tt>, <tt>y</tt>, <tt>dx</tt> and <tt>dy</tt> attributes
+ * that correspond to the mark's location and dimensions in pixels. The mark's
+ * positional properties are not set directly by this behavior; instead, the
+ * positional properties should be defined as:
+ *
+ * <pre>    .left(function(d) d.x)
+ *     .top(function(d) d.y)
+ *     .width(function(d) d.dx)
+ *     .height(function(d) d.dy)</pre>
+ *
+ * Thus, the behavior does not resize the mark directly, but instead updates the
+ * size by updating the assigned panel's underlying data. Note that if the
+ * positional properties are defined with bottom and right (rather than top and
+ * left), the resize behavior will be inverted, which will confuse users!
+ *
+ * <p>The resize behavior is bounded by the assigned mark's enclosing panel; the
+ * positional attributes are clamped such that the selection does not extend
+ * outside the panel's bounds.
+ *
+ * <p>The mark being resized is automatically re-rendered for each mouse event
+ * as part of the resize operation. This behavior may be enhanced in the future
+ * to allow more flexible configuration. In some cases, such as with parallel
+ * coordinates, resizing the selection may cause related marks to change, in
+ * which case additional marks may also need to be rendered. This can be
+ * accomplished by listening for the select psuedo-events:<ul>
+ *
+ * <li>resizestart (on mousedown)
+ * <li>resize (on mousemove)
+ * <li>resizeend (on mouseup)
+ *
+ * </ul>For example, to render the parent panel while resizing, thus
+ * re-rendering all sibling marks:
+ *
+ * <pre>    .event("mousedown", pv.Behavior.resize("left"))
+ *     .event("resize", function() this.parent)</pre>
+ *
+ * This behavior may be enhanced in the future to allow more flexible
+ * configuration of the selection behavior.
+ *
+ * @extends pv.Behavior
+ * @see pv.Behavior.select
+ * @see pv.Behavior.drag
+ */
+pv.Behavior.resize = function(side) {
+  var scene, // scene context
+      index, // scene context
+      r, // region being selected
+      m1; // initial mouse position
+
+  /** @private */
+  function mousedown(d) {
+    index = this.index;
+    scene = this.scene;
+    m1 = this.mouse();
+    r = d;
+    switch (side) {
+      case "left": m1.x = r.x + r.dx; break;
+      case "right": m1.x = r.x; break;
+      case "top": m1.y = r.y + r.dy; break;
+      case "bottom": m1.y = r.y; break;
+    }
+    pv.Mark.dispatch("resizestart", scene, index);
+  }
+
+  /** @private */
+  function mousemove() {
+    if (!scene) return;
+    scene.mark.context(scene, index, function() {
+        var m2 = this.mouse();
+        r.x = Math.max(0, Math.min(m1.x, m2.x));
+        r.y = Math.max(0, Math.min(m1.y, m2.y));
+        r.dx = Math.min(this.parent.width(), Math.max(m2.x, m1.x)) - r.x;
+        r.dy = Math.min(this.parent.height(), Math.max(m2.y, m1.y)) - r.y;
+        this.render();
+      });
+    pv.Mark.dispatch("resize", scene, index);
+  }
+
+  /** @private */
+  function mouseup() {
+    if (!scene) return;
+    pv.Mark.dispatch("resizeend", scene, index);
+    scene = null;
+  }
+
+  pv.listen(window, "mousemove", mousemove);
+  pv.listen(window, "mouseup", mouseup);
+  return mousedown;
+};
+/**
+ * Returns a new pan behavior to be registered on mousedown events.
+ *
+ * @class Implements interactive panning starting with mousedown events.
+ * Register this behavior on panels to allow panning. This behavior can be used
+ * in tandem with {@link pv.Behavior.zoom} to allow both panning and zooming:
+ *
+ * <pre>    .event("mousedown", pv.Behavior.pan())
+ *     .event("mousewheel", pv.Behavior.zoom())</pre>
+ *
+ * The pan behavior currently supports only mouse events; support for keyboard
+ * shortcuts to improve accessibility may be added in the future.
+ *
+ * <p>After the initial mousedown event is triggered, this behavior listens for
+ * mousemove and mouseup events on the window. This allows panning to continue
+ * even if the mouse temporarily leaves the panel that is being panned, or even
+ * the root panel.
+ *
+ * <p>The implementation of this behavior relies on the panel's
+ * <tt>transform</tt> property, which specifies a matrix transformation that is
+ * applied to child marks. Note that the transform property only affects the
+ * panel's children, but not the panel itself; therefore the panel's fill and
+ * stroke will not change when the contents are panned.
+ *
+ * <p>Panels have transparent fill styles by default; this means that panels may
+ * not receive the initial mousedown event to start panning. To fix this
+ * problem, either given the panel a visible fill style (such as "white"), or
+ * set the <tt>events</tt> property to "all" such that the panel receives events
+ * despite its transparent fill.
+ *
+ * <p>The pan behavior has optional support for bounding. If enabled, the user
+ * will not be able to pan the panel outside of the initial bounds. This feature
+ * is designed to work in conjunction with the zoom behavior; otherwise,
+ * bounding the panel effectively disables all panning.
+ *
+ * @extends pv.Behavior
+ * @see pv.Behavior.zoom
+ * @see pv.Panel#transform
+ */
+pv.Behavior.pan = function() {
+  var scene, // scene context
+      index, // scene context
+      m1, // transformation matrix at the start of panning
+      v1, // mouse location at the start of panning
+      k, // inverse scale
+      bound; // whether to bound to the panel
+
+  /** @private */
+  function mousedown() {
+    index = this.index;
+    scene = this.scene;
+    v1 = pv.vector(pv.event.pageX, pv.event.pageY);
+    m1 = this.transform();
+    k = 1 / (m1.k * this.scale);
+    if (bound) {
+      bound = {
+        x: (1 - m1.k) * this.width(),
+        y: (1 - m1.k) * this.height()
+      };
+    }
+  }
+
+  /** @private */
+  function mousemove() {
+    if (!scene) return;
+    scene.mark.context(scene, index, function() {
+        var x = (pv.event.pageX - v1.x) * k,
+            y = (pv.event.pageY - v1.y) * k,
+            m = m1.translate(x, y);
+        if (bound) {
+          m.x = Math.max(bound.x, Math.min(0, m.x));
+          m.y = Math.max(bound.y, Math.min(0, m.y));
+        }
+        this.transform(m).render();
+      });
+    pv.Mark.dispatch("pan", scene, index);
+  }
+
+  /** @private */
+  function mouseup() {
+    scene = null;
+  }
+
+  /**
+   * Sets or gets the bound parameter. If bounding is enabled, the user will not
+   * be able to pan outside the initial panel bounds; this typically applies
+   * only when the pan behavior is used in tandem with the zoom behavior.
+   * Bounding is not enabled by default.
+   *
+   * <p>Note: enabling bounding after panning has already occurred will not
+   * immediately reset the transform. Bounding should be enabled before the
+   * panning behavior is applied.
+   *
+   * @function
+   * @returns {pv.Behavior.pan} this, or the current bound parameter.
+   * @name pv.Behavior.pan.prototype.bound
+   * @param {boolean} [x] the new bound parameter.
+   */
+  mousedown.bound = function(x) {
+    if (arguments.length) {
+      bound = Boolean(x);
+      return this;
+    }
+    return Boolean(bound);
+  };
+
+  pv.listen(window, "mousemove", mousemove);
+  pv.listen(window, "mouseup", mouseup);
+  return mousedown;
+};
+/**
+ * Returns a new zoom behavior to be registered on mousewheel events.
+ *
+ * @class Implements interactive zooming using mousewheel events. Register this
+ * behavior on panels to allow zooming. This behavior can be used in tandem with
+ * {@link pv.Behavior.pan} to allow both panning and zooming:
+ *
+ * <pre>    .event("mousedown", pv.Behavior.pan())
+ *     .event("mousewheel", pv.Behavior.zoom())</pre>
+ *
+ * The zoom behavior currently supports only mousewheel events; support for
+ * keyboard shortcuts and gesture events to improve accessibility may be added
+ * in the future.
+ *
+ * <p>The implementation of this behavior relies on the panel's
+ * <tt>transform</tt> property, which specifies a matrix transformation that is
+ * applied to child marks. Note that the transform property only affects the
+ * panel's children, but not the panel itself; therefore the panel's fill and
+ * stroke will not change when the contents are zoomed. The built-in support for
+ * transforms only supports uniform scaling and translates, which is sufficient
+ * for panning and zooming.  Note that this is not a strict geometric
+ * transformation, as the <tt>lineWidth</tt> property is scale-aware: strokes
+ * are drawn at constant size independent of scale.
+ *
+ * <p>Panels have transparent fill styles by default; this means that panels may
+ * not receive mousewheel events to zoom. To fix this problem, either given the
+ * panel a visible fill style (such as "white"), or set the <tt>events</tt>
+ * property to "all" such that the panel receives events despite its transparent
+ * fill.
+ *
+ * <p>The zoom behavior has optional support for bounding. If enabled, the user
+ * will not be able to zoom out farther than the initial bounds. This feature is
+ * designed to work in conjunction with the pan behavior.
+ *
+ * @extends pv.Behavior
+ * @see pv.Panel#transform
+ * @see pv.Mark#scale
+ * @param {number} speed
+ */
+pv.Behavior.zoom = function(speed) {
+  var bound; // whether to bound to the panel
+
+  if (!arguments.length) speed = 1 / 48;
+
+  /** @private */
+  function mousewheel() {
+    var v = this.mouse(),
+        k = pv.event.wheel * speed,
+        m = this.transform().translate(v.x, v.y)
+            .scale((k < 0) ? (1e3 / (1e3 - k)) : ((1e3 + k) / 1e3))
+            .translate(-v.x, -v.y);
+    if (bound) {
+      m.k = Math.max(1, m.k);
+      m.x = Math.max((1 - m.k) * this.width(), Math.min(0, m.x));
+      m.y = Math.max((1 - m.k) * this.height(), Math.min(0, m.y));
+    }
+    this.transform(m).render();
+    pv.Mark.dispatch("zoom", this.scene, this.index);
+  }
+
+  /**
+   * Sets or gets the bound parameter. If bounding is enabled, the user will not
+   * be able to zoom out farther than the initial panel bounds. Bounding is not
+   * enabled by default. If this behavior is used in tandem with the pan
+   * behavior, both should use the same bound parameter.
+   *
+   * <p>Note: enabling bounding after zooming has already occurred will not
+   * immediately reset the transform. Bounding should be enabled before the zoom
+   * behavior is applied.
+   *
+   * @function
+   * @returns {pv.Behavior.zoom} this, or the current bound parameter.
+   * @name pv.Behavior.zoom.prototype.bound
+   * @param {boolean} [x] the new bound parameter.
+   */
+  mousewheel.bound = function(x) {
+    if (arguments.length) {
+      bound = Boolean(x);
+      return this;
+    }
+    return Boolean(bound);
+  };
+
+  return mousewheel;
+};
+/**
+ * @ignore
+ * @namespace
+ */
+pv.Geo = function() {};
+/**
+ * Abstract; not implemented. There is no explicit constructor; this class
+ * merely serves to document the representation used by {@link pv.Geo.scale}.
+ *
+ * @class Represents a pair of geographic coordinates.
+ *
+ * @name pv.Geo.LatLng
+ * @see pv.Geo.scale
+ */
+
+/**
+ * The <i>latitude</i> coordinate in degrees; positive is North.
+ *
+ * @type number
+ * @name pv.Geo.LatLng.prototype.lat
+ */
+
+/**
+ * The <i>longitude</i> coordinate in degrees; positive is East.
+ *
+ * @type number
+ * @name pv.Geo.LatLng.prototype.lng
+ */
+/**
+ * Abstract; not implemented. There is no explicit constructor; this class
+ * merely serves to document the representation used by {@link pv.Geo.scale}.
+ *
+ * @class Represents a geographic projection. This class provides the core
+ * implementation for {@link pv.Geo.scale}s, mapping between geographic
+ * coordinates (latitude and longitude) and normalized screen space in the range
+ * [-1,1]. The remaining mapping between normalized screen space and actual
+ * pixels is performed by <tt>pv.Geo.scale</tt>.
+ *
+ * <p>Many geographic projections have a point around which the projection is
+ * centered. Rather than have each implementation add support for a
+ * user-specified center point, the <tt>pv.Geo.scale</tt> translates the
+ * geographic coordinates relative to the center point for both the forward and
+ * inverse projection.
+ *
+ * <p>In general, this class should not be used directly, unless the desire is
+ * to implement a new geographic projection. Instead, use <tt>pv.Geo.scale</tt>.
+ * Implementations are not required to implement inverse projections, but are
+ * needed for some forms of interactivity. Also note that some inverse
+ * projections are ambiguous, such as the connecting points in Dymaxian maps.
+ *
+ * @name pv.Geo.Projection
+ * @see pv.Geo.scale
+ */
+
+/**
+ * The <i>forward</i> projection.
+ *
+ * @function
+ * @name pv.Geo.Projection.prototype.project
+ * @param {pv.Geo.LatLng} latlng the latitude and longitude to project.
+ * @returns {pv.Vector} the xy-coordinates of the given point.
+ */
+
+/**
+ * The <i>inverse</i> projection; optional.
+ *
+ * @function
+ * @name pv.Geo.Projection.prototype.invert
+ * @param {pv.Vector} xy the x- and y-coordinates to invert.
+ * @returns {pv.Geo.LatLng} the latitude and longitude of the given point.
+ */
+/**
+ * The built-in projections.
+ *
+ * @see pv.Geo.Projection
+ * @namespace
+ */
+pv.Geo.projections = {
+
+  /** @see http://en.wikipedia.org/wiki/Mercator_projection */
+  mercator: {
+    project: function(latlng) {
+      return {
+          x: latlng.lng / 180,
+          y: latlng.lat > 85 ? 1 : latlng.lat < -85 ? -1
+              : Math.log(Math.tan(Math.PI / 4
+              + pv.radians(latlng.lat) / 2)) / Math.PI
+        };
+    },
+    invert: function(xy) {
+      return {
+          lng: xy.x * 180,
+          lat: pv.degrees(2 * Math.atan(Math.exp(xy.y * Math.PI)) - Math.PI / 2)
+        };
+    }
+  },
+
+  /** @see http://en.wikipedia.org/wiki/Gall-Peters_projection */
+  "gall-peters": {
+    project: function(latlng) {
+      return {
+          x: latlng.lng / 180,
+          y: Math.sin(pv.radians(latlng.lat))
+        };
+    },
+    invert: function(xy) {
+      return {
+          lng: xy.x * 180,
+          lat: pv.degrees(Math.asin(xy.y))
+        };
+    }
+  },
+
+  /** @see http://en.wikipedia.org/wiki/Sinusoidal_projection */
+  sinusoidal: {
+    project: function(latlng) {
+      return {
+          x: pv.radians(latlng.lng) * Math.cos(pv.radians(latlng.lat)) / Math.PI,
+          y: latlng.lat / 90
+        };
+    },
+    invert: function(xy) {
+      return {
+          lng: pv.degrees((xy.x * Math.PI) / Math.cos(xy.y * Math.PI / 2)),
+          lat: xy.y * 90
+        };
+    }
+  },
+
+  /** @see http://en.wikipedia.org/wiki/Aitoff_projection */
+  aitoff: {
+    project: function(latlng) {
+      var l = pv.radians(latlng.lng),
+          f = pv.radians(latlng.lat),
+          a = Math.acos(Math.cos(f) * Math.cos(l / 2));
+      return {
+          x: 2 * (a ? (Math.cos(f) * Math.sin(l / 2) * a / Math.sin(a)) : 0) / Math.PI,
+          y: 2 * (a ? (Math.sin(f) * a / Math.sin(a)) : 0) / Math.PI
+        };
+    },
+    invert: function(xy) {
+      var x = xy.x * Math.PI / 2,
+          y = xy.y * Math.PI / 2;
+      return {
+          lng: pv.degrees(x / Math.cos(y)),
+          lat: pv.degrees(y)
+        };
+    }
+  },
+
+  /** @see http://en.wikipedia.org/wiki/Hammer_projection */
+  hammer: {
+    project: function(latlng) {
+      var l = pv.radians(latlng.lng),
+          f = pv.radians(latlng.lat),
+          c = Math.sqrt(1 + Math.cos(f) * Math.cos(l / 2));
+      return {
+          x: 2 * Math.SQRT2 * Math.cos(f) * Math.sin(l / 2) / c / 3,
+          y: Math.SQRT2 * Math.sin(f) / c / 1.5
+        };
+    },
+    invert: function(xy) {
+      var x = xy.x * 3,
+          y = xy.y * 1.5,
+          z = Math.sqrt(1 - x * x / 16 - y * y / 4);
+      return {
+          lng: pv.degrees(2 * Math.atan2(z * x, 2 * (2 * z * z - 1))),
+          lat: pv.degrees(Math.asin(z * y))
+        };
+    }
+  },
+
+  /** The identity or "none" projection. */
+  identity: {
+    project: function(latlng) {
+      return {
+          x: latlng.lng / 180,
+          y: latlng.lat / 90
+        };
+    },
+    invert: function(xy) {
+      return {
+          lng: xy.x * 180,
+          lat: xy.y * 90
+        };
+    }
+  }
+};
+/**
+ * Returns a geographic scale. The arguments to this constructor are optional,
+ * and equivalent to calling {@link #projection}.
+ *
+ * @class Represents a geographic scale; a mapping between latitude-longitude
+ * coordinates and screen pixel coordinates. By default, the domain is inferred
+ * from the geographic coordinates, so that the domain fills the output range.
+ *
+ * <p>Note that geographic scales are two-dimensional transformations, rather
+ * than the one-dimensional bidrectional mapping typical of other scales.
+ * Rather than mapping (for example) between a numeric domain and a numeric
+ * range, geographic scales map between two coordinate objects: {@link
+ * pv.Geo.LatLng} and {@link pv.Vector}.
+ *
+ * @param {pv.Geo.Projection} [p] optional projection.
+ * @see pv.Geo.scale#ticks
+ */
+pv.Geo.scale = function(p) {
+  var rmin = {x: 0, y: 0}, // default range minimum
+      rmax = {x: 1, y: 1}, // default range maximum
+      d = [], // default domain
+      j = pv.Geo.projections.identity, // domain <-> normalized range
+      x = pv.Scale.linear(-1, 1).range(0, 1), // normalized <-> range
+      y = pv.Scale.linear(-1, 1).range(1, 0), // normalized <-> range
+      c = {lng: 0, lat: 0}, // Center Point
+      lastLatLng, // cached latlng
+      lastPoint; // cached point
+
+  /** @private */
+  function scale(latlng) {
+    if (!lastLatLng
+        || (latlng.lng != lastLatLng.lng)
+        || (latlng.lat != lastLatLng.lat)) {
+      lastLatLng = latlng;
+      var p = project(latlng);
+      lastPoint = {x: x(p.x), y: y(p.y)};
+    }
+    return lastPoint;
+  }
+
+  /** @private */
+  function project(latlng) {
+    var offset = {lng: latlng.lng - c.lng, lat: latlng.lat};
+    return j.project(offset);
+  }
+
+  /** @private */
+  function invert(xy) {
+    var latlng = j.invert(xy);
+    latlng.lng += c.lng;
+    return latlng;
+  }
+
+  /** Returns the projected x-coordinate. */
+  scale.x = function(latlng) {
+    return scale(latlng).x;
+  };
+
+  /** Returns the projected y-coordinate. */
+  scale.y = function(latlng) {
+    return scale(latlng).y;
+  };
+
+  /**
+   * Abstract; this is a local namespace on a given geographic scale.
+   *
+   * @namespace Tick functions for geographic scales. Because geographic scales
+   * represent two-dimensional transformations (as opposed to one-dimensional
+   * transformations typical of other scales), the tick values are similarly
+   * represented as two-dimensional coordinates in the input domain, i.e.,
+   * {@link pv.Geo.LatLng} objects.
+   *
+   * <p>Also, note that non-rectilinear projections, such as sinsuoidal and
+   * aitoff, may not produce straight lines for constant longitude or constant
+   * latitude. Therefore the returned array of ticks is a two-dimensional array,
+   * sampling various latitudes as constant longitude, and vice versa.
+   *
+   * <p>The tick lines can therefore be approximated as polylines, either with
+   * "linear" or "cardinal" interpolation. This is not as accurate as drawing
+   * the true curve through the projection space, but is usually sufficient.
+   *
+   * @name pv.Geo.scale.prototype.ticks
+   * @see pv.Geo.scale
+   * @see pv.Geo.LatLng
+   * @see pv.Line#interpolate
+   */
+  scale.ticks = {
+
+    /**
+     * Returns longitude ticks.
+     *
+     * @function
+     * @param {number} [m] the desired number of ticks.
+     * @returns {array} a nested array of <tt>pv.Geo.LatLng</tt> ticks.
+     * @name pv.Geo.scale.prototype.ticks.prototype.lng
+     */
+    lng: function(m) {
+      var lat, lng;
+      if (d.length > 1) {
+        var s = pv.Scale.linear();
+        if (m == undefined) m = 10;
+        lat = s.domain(d, function(d) { return d.lat; }).ticks(m);
+        lng = s.domain(d, function(d) { return d.lng; }).ticks(m);
+      } else {
+        lat = pv.range(-80, 81, 10);
+        lng = pv.range(-180, 181, 10);
+      }
+      return lng.map(function(lng) {
+        return lat.map(function(lat) {
+          return {lat: lat, lng: lng};
+        });
+      });
+    },
+
+    /**
+     * Returns latitude ticks.
+     *
+     * @function
+     * @param {number} [m] the desired number of ticks.
+     * @returns {array} a nested array of <tt>pv.Geo.LatLng</tt> ticks.
+     * @name pv.Geo.scale.prototype.ticks.prototype.lat
+     */
+    lat: function(m) {
+      return pv.transpose(scale.ticks.lng(m));
+    }
+  };
+
+  /**
+   * Inverts the specified value in the output range, returning the
+   * corresponding value in the input domain. This is frequently used to convert
+   * the mouse location (see {@link pv.Mark#mouse}) to a value in the input
+   * domain. Inversion is only supported for numeric ranges, and not colors.
+   *
+   * <p>Note that this method does not do any rounding or bounds checking. If
+   * the input domain is discrete (e.g., an array index), the returned value
+   * should be rounded. If the specified <tt>y</tt> value is outside the range,
+   * the returned value may be equivalently outside the input domain.
+   *
+   * @function
+   * @name pv.Geo.scale.prototype.invert
+   * @param {number} y a value in the output range (a pixel location).
+   * @returns {number} a value in the input domain.
+   */
+  scale.invert = function(p) {
+    return invert({x: x.invert(p.x), y: y.invert(p.y)});
+  };
+
+  /**
+   * Sets or gets the input domain. Note that unlike quantitative scales, the
+   * domain cannot be reduced to a simple rectangle (i.e., minimum and maximum
+   * values for latitude and longitude). Instead, the domain values must be
+   * projected to normalized space, effectively finding the domain in normalized
+   * space rather than in terms of latitude and longitude. Thus, changing the
+   * projection requires recomputing the normalized domain.
+   *
+   * <p>This method can be invoked several ways:
+   *
+   * <p>1. <tt>domain(values...)</tt>
+   *
+   * <p>Specifying the domain as a series of {@link pv.Geo.LatLng}s is the most
+   * explicit and recommended approach. However, if the domain values are
+   * derived from data, you may find the second method more appropriate.
+   *
+   * <p>2. <tt>domain(array, f)</tt>
+   *
+   * <p>Rather than enumerating the domain explicitly, you can specify a single
+   * argument of an array. In addition, you can specify an optional accessor
+   * function to extract the domain values (as {@link pv.Geo.LatLng}s) from the
+   * array. If the specified array has fewer than two elements, this scale will
+   * default to the full normalized domain.
+   *
+   * <p>2. <tt>domain()</tt>
+   *
+   * <p>Invoking the <tt>domain</tt> method with no arguments returns the
+   * current domain as an array.
+   *
+   * @function
+   * @name pv.Geo.scale.prototype.domain
+   * @param {...} domain... domain values.
+   * @returns {pv.Geo.scale} <tt>this</tt>, or the current domain.
+   */
+  scale.domain = function(array, f) {
+    if (arguments.length) {
+      d = (array instanceof Array)
+          ? ((arguments.length > 1) ? pv.map(array, f) : array)
+          : Array.prototype.slice.call(arguments);
+      if (d.length > 1) {
+        var lngs = d.map(function(c) { return c.lng; });
+        var lats = d.map(function(c) { return c.lat; });
+        c = {
+          lng: (pv.max(lngs) + pv.min(lngs)) / 2,
+          lat: (pv.max(lats) + pv.min(lats)) / 2
+        };
+        var n = d.map(project); // normalized domain
+        x.domain(n, function(p) { return p.x; });
+        y.domain(n, function(p) { return p.y; });
+      } else {
+        c = {lng: 0, lat: 0};
+        x.domain(-1, 1);
+        y.domain(-1, 1);
+      }
+      lastLatLng = null; // invalidate the cache
+      return this;
+    }
+    return d;
+  };
+
+  /**
+   * Sets or gets the output range. This method can be invoked several ways:
+   *
+   * <p>1. <tt>range(min, max)</tt>
+   *
+   * <p>If two objects are specified, the arguments should be {@link pv.Vector}s
+   * which specify the minimum and maximum values of the x- and y-coordinates
+   * explicitly.
+   *
+   * <p>2. <tt>range(width, height)</tt>
+   *
+   * <p>If two numbers are specified, the arguments specify the maximum values
+   * of the x- and y-coordinates explicitly; the minimum values are implicitly
+   * zero.
+   *
+   * <p>3. <tt>range()</tt>
+   *
+   * <p>Invoking the <tt>range</tt> method with no arguments returns the current
+   * range as an array of two {@link pv.Vector}s: the minimum (top-left) and
+   * maximum (bottom-right) values.
+   *
+   * @function
+   * @name pv.Geo.scale.prototype.range
+   * @param {...} range... range values.
+   * @returns {pv.Geo.scale} <tt>this</tt>, or the current range.
+   */
+  scale.range = function(min, max) {
+    if (arguments.length) {
+      if (typeof min == "object") {
+        rmin = {x: Number(min.x), y: Number(min.y)};
+        rmax = {x: Number(max.x), y: Number(max.y)};
+      } else {
+        rmin = {x: 0, y: 0};
+        rmax = {x: Number(min), y: Number(max)};
+      }
+      x.range(rmin.x, rmax.x);
+      y.range(rmax.y, rmin.y); // XXX flipped?
+      lastLatLng = null; // invalidate the cache
+      return this;
+    }
+    return [rmin, rmax];
+  };
+
+  /**
+   * Sets or gets the projection. This method can be invoked several ways:
+   *
+   * <p>1. <tt>projection(string)</tt>
+   *
+   * <p>Specifying a string sets the projection to the given named projection in
+   * {@link pv.Geo.projections}. If no such projection is found, the identity
+   * projection is used.
+   *
+   * <p>2. <tt>projection(object)</tt>
+   *
+   * <p>Specifying an object sets the projection to the given custom projection,
+   * which must implement the <i>forward</i> and <i>inverse</i> methods per the
+   * {@link pv.Geo.Projection} interface.
+   *
+   * <p>3. <tt>projection()</tt>
+   *
+   * <p>Invoking the <tt>projection</tt> method with no arguments returns the
+   * current object that defined the projection.
+   *
+   * @function
+   * @name pv.Scale.geo.prototype.projection
+   * @param {...} range... range values.
+   * @returns {pv.Scale.geo} <tt>this</tt>, or the current range.
+   */
+  scale.projection = function(p) {
+    if (arguments.length) {
+      j = typeof p == "string"
+          ? pv.Geo.projections[p] || pv.Geo.projections.identity
+          : p;
+      return this.domain(d); // recompute normalized domain
+    }
+    return p;
+  };
+
+  /**
+   * Returns a view of this scale by the specified accessor function <tt>f</tt>.
+   * Given a scale <tt>g</tt>, <tt>g.by(function(d) d.foo)</tt> is equivalent to
+   * <tt>function(d) g(d.foo)</tt>. This method should be used judiciously; it
+   * is typically more clear to invoke the scale directly, passing in the value
+   * to be scaled.
+   *
+   * @function
+   * @name pv.Geo.scale.prototype.by
+   * @param {function} f an accessor function.
+   * @returns {pv.Geo.scale} a view of this scale by the specified accessor
+   * function.
+   */
+  scale.by = function(f) {
+    function by() { return scale(f.apply(this, arguments)); }
+    for (var method in scale) by[method] = scale[method];
+    return by;
+  };
+
+  if (arguments.length) scale.projection(p);
+  return scale;
+};
index e291e44f222c3964e3808e44626de71c85b5da74..23fcfe8fbe2ef6b3095de267485cb5403d554924 100644 (file)
@@ -1,6 +1,6 @@
-// script.aculo.us builder.js v1.8.3, Thu Oct 08 11:23:33 +0200 2009
+// script.aculo.us builder.js v1.9.0, Thu Dec 23 16:54:48 -0500 2010
 
-// Copyright (c) 2005-2009 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
+// Copyright (c) 2005-2010 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
 //
 // script.aculo.us is freely distributable under the terms of an MIT-style license.
 // For details, see the script.aculo.us web site: http://script.aculo.us/
@@ -134,11 +134,9 @@ var Builder = {
     });
   }
 };
+// script.aculo.us effects.js v1.9.0, Thu Dec 23 16:54:48 -0500 2010
 
-
-// script.aculo.us effects.js v1.8.3, Thu Oct 08 11:23:33 +0200 2009
-
-// Copyright (c) 2005-2009 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
+// Copyright (c) 2005-2010 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
 // Contributors:
 //  Justin Palmer (http://encytemedia.com/)
 //  Mark Pilgrim (http://diveintomark.org/)
@@ -1260,12 +1258,11 @@ $w('getInlineOpacity forceRerendering setContentZoom collectTextNodes collectTex
 
 Element.addMethods(Effect.Methods);
 
+// script.aculo.us controls.js v1.9.0, Thu Dec 23 16:54:48 -0500 2010
 
-// script.aculo.us controls.js v1.8.3, Thu Oct 08 11:23:33 +0200 2009
-
-// Copyright (c) 2005-2009 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
-//           (c) 2005-2009 Ivan Krstic (http://blogs.law.harvard.edu/ivan)
-//           (c) 2005-2009 Jon Tirsen (http://www.tirsen.com)
+// Copyright (c) 2005-2010 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
+//           (c) 2005-2010 Ivan Krstic (http://blogs.law.harvard.edu/ivan)
+//           (c) 2005-2010 Jon Tirsen (http://www.tirsen.com)
 // Contributors:
 //  Richard Livsey
 //  Rahul Bhargava
@@ -1506,10 +1503,10 @@ Autocompleter.Base = Class.create({
     var value = '';
     if (this.options.select) {
       var nodes = $(selectedElement).select('.' + this.options.select) || [];
-      if(nodes.length>0) value = Element.collectTextNodes(nodes[0], this.options.select);      
+      if(nodes.length>0) value = Element.collectTextNodes(nodes[0], this.options.select);
     } else
       value = Element.collectTextNodesIgnoreClass(selectedElement, 'informal');
-    
+
     var bounds = this.getTokenBounds();
     if (bounds[0] != -1) {
       var newValue = this.element.value.substr(0, bounds[0]);
@@ -1521,9 +1518,12 @@ Autocompleter.Base = Class.create({
       this.element.value = value;
     }
     this.oldElementValue = this.element.value;
-    // Following line was commented for SONAR-1688 because in our autosuggest text fields, we use
+
+    // SONAR
+       // Following line was commented for SONAR-1688 because in our autosuggest text fields, we use
     // the onfocus() method to reinitialize the value of the input field to ''.
     //this.element.focus();
+    // /SONAR
 
     if (this.options.afterUpdateElement)
       this.options.afterUpdateElement(this.element, selectedElement);
@@ -2228,11 +2228,9 @@ Form.Element.DelayedObserver = Class.create({
     this.callback(this.element, $F(this.element));
   }
 });
+// script.aculo.us dragdrop.js v1.9.0, Thu Dec 23 16:54:48 -0500 2010
 
-
-// script.aculo.us dragdrop.js v1.8.3, Thu Oct 08 11:23:33 +0200 2009
-
-// Copyright (c) 2005-2009 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
+// Copyright (c) 2005-2010 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
 //
 // script.aculo.us is freely distributable under the terms of an MIT-style license.
 // For details, see the script.aculo.us web site: http://script.aculo.us/
@@ -2606,7 +2604,7 @@ var Draggable = Class.create({
       if (this.options.scroll == window) {
         with(this._getWindowScroll(this.options.scroll)) { p = [ left, top, left+width, top+height ]; }
       } else {
-        p = Position.page(this.options.scroll);
+        p = Position.page(this.options.scroll).toArray();
         p[0] += this.options.scroll.scrollLeft + Position.deltaX;
         p[1] += this.options.scroll.scrollTop + Position.deltaY;
         p.push(p[0]+this.options.scroll.offsetWidth);