
/***********************************************************
 *
 * 		dw_event.js
 *
 ***********************************************************/

//  dw_event.js version date Apr 2008
//  basic event handling file from dyn-web.com
var pricingTypeEN=" MSRP";
var pricingTypeFR=" PDSF";
var dw_Event = {
  
    add: function(obj, etype, fp, cap) {
        cap = cap || false;
        if (obj.addEventListener) obj.addEventListener(etype, fp, cap);
        else if (obj.attachEvent) obj.attachEvent("on" + etype, fp);
    }, 

    remove: function(obj, etype, fp, cap) {
        cap = cap || false;
        if (obj.removeEventListener) obj.removeEventListener(etype, fp, cap);
        else if (obj.detachEvent) obj.detachEvent("on" + etype, fp);
    }, 
    
    DOMit: function(e) { 
        e = e? e: window.event; // e IS passed when using attachEvent though ...
        if (!e.target) e.target = e.srcElement;
        if (!e.preventDefault) e.preventDefault = function () { e.returnValue = false; return false; }
        if (!e.stopPropagation) e.stopPropagation = function () { e.cancelBubble = true; }
        return e;
    },
    
    getTarget: function(e) {
        e = dw_Event.DOMit(e); var tgt = e.target; 
        if (tgt.nodeType != 1) tgt = tgt.parentNode; // safari...
        return tgt;
    }
    
}

// Danny Goodman's version (DHTML def ref)
function addLoadEvent(func) {
    var oldQueue = window.onload? window.onload: function() {};
    window.onload = function() {
        oldQueue();
        func();
    }
}


/***********************************************************
 *
 * 		dw_scroll.js
 *
 ***********************************************************/

/*************************************************************************
    This code is from Dynamic Web Coding at dyn-web.com
    Copyright 2001-2008 by Sharon Paine 
    See Terms of Use at www.dyn-web.com/business/terms.php
    regarding conditions under which you may use this code.
    This notice must be retained in the code as is!
    
    version date: Aug 2008
*************************************************************************/

// horizId only needed for horizontal scrolling
function dw_scrollObj(wndoId, lyrId, horizId) {
    var wn = document.getElementById(wndoId);
    this.id = wndoId; dw_scrollObj.col[this.id] = this;
    this.animString = "dw_scrollObj.col." + this.id;
    this.load(lyrId, horizId);
    
    if (wn.addEventListener) {
        wn.addEventListener('DOMMouseScroll', dw_scrollObj.doOnMouseWheel, false);
    } 
    wn.onmousewheel = dw_scrollObj.doOnMouseWheel;
}

dw_scrollObj.isSupported = function () {
    if ( document.getElementById && document.getElementsByTagName 
         && document.addEventListener || document.attachEvent ) {
        return true;
    }
    return false;
}

dw_scrollObj.col = {}; // collect instances
dw_scrollObj.defaultSpeed = dw_scrollObj.prototype.speed = 100; // default for mouseover or mousedown scrolling
dw_scrollObj.defaultSlideDur = dw_scrollObj.prototype.slideDur = 500; // default duration of glide onclick

// pseudo events 
dw_scrollObj.prototype.on_load = function() {} // when dw_scrollObj initialized or new layer loaded
dw_scrollObj.prototype.on_scroll = function() {}
dw_scrollObj.prototype.on_scroll_start = function() {}
dw_scrollObj.prototype.on_scroll_stop = function() {} // when scrolling has ceased (mouseout/up)
dw_scrollObj.prototype.on_scroll_end = function() {} // reached end
dw_scrollObj.prototype.on_update = function() {} // called in updateDims

dw_scrollObj.prototype.on_glidescroll = function() {}
dw_scrollObj.prototype.on_glidescroll_start = function() {}
dw_scrollObj.prototype.on_glidescroll_stop = function() {} // destination (to/by) reached
dw_scrollObj.prototype.on_glidescroll_end = function() {} // reached end

dw_scrollObj.prototype.load = function(lyrId, horizId) {
    var wndo, lyr;
    if (this.lyrId) { // layer currently loaded?
        lyr = document.getElementById(this.lyrId);
        lyr.style.visibility = "hidden";
    }
    this.lyr = lyr = document.getElementById(lyrId); // hold this.lyr?
    this.lyr.style.position = 'absolute'; 
    this.lyrId = lyrId; // hold id of currently visible layer
    this.horizId = horizId || null; // hold horizId for update fn
    wndo = document.getElementById(this.id);
    this.y = 0; this.x = 0; this.shiftTo(0,0);
    this.getDims(wndo, lyr); 
    lyr.style.visibility = "visible";
    this.ready = true; this.on_load();
}

dw_scrollObj.prototype.shiftTo = function(x, y) {
    if (this.lyr) {
        this.lyr.style.left = (this.x = x) + "px"; 
        this.lyr.style.top = (this.y = y) + "px";
    }
}

dw_scrollObj.prototype.getX = function() { return this.x; }
dw_scrollObj.prototype.getY = function() { return this.y; }

dw_scrollObj.prototype.getDims = function(wndo, lyr) { 
    this.wd = this.horizId? document.getElementById( this.horizId ).offsetWidth: lyr.offsetWidth;
    this.maxX = (this.wd - wndo.offsetWidth > 0)? this.wd - wndo.offsetWidth: 0;
    this.maxY = (lyr.offsetHeight - wndo.offsetHeight > 0)? lyr.offsetHeight - wndo.offsetHeight: 0;
}

dw_scrollObj.prototype.updateDims = function() {
    var wndo = document.getElementById(this.id);
    var lyr = document.getElementById( this.lyrId );
    this.getDims(wndo, lyr);
    this.on_update();
}

// for mouseover/mousedown scrolling
dw_scrollObj.prototype.initScrollVals = function(deg, speed) {
    if (!this.ready) return; 
    if (this.timerId) {
        clearInterval(this.timerId); this.timerId = 0;
    }
    this.speed = speed || dw_scrollObj.defaultSpeed;
    this.fx = (deg == 0)? -1: (deg == 180)? 1: 0;
    this.fy = (deg == 90)? 1: (deg == 270)? -1: 0;
    this.endX = (deg == 90 || deg == 270)? this.x: (deg == 0)? -this.maxX: 0; 
    this.endY = (deg == 0 || deg == 180)? this.y: (deg == 90)? 0: -this.maxY;
    this.lyr = document.getElementById(this.lyrId);
    this.lastTime = new Date().getTime();
    this.on_scroll_start(this.x, this.y);  
    this.timerId = setInterval(this.animString + ".scroll()", 10);    
}

dw_scrollObj.prototype.scroll = function() {
    var now = new Date().getTime();
    var d = (now - this.lastTime)/1000 * this.speed;
    if (d > 0) { 
        var x = this.x + Math.round(this.fx * d); var y = this.y + Math.round(this.fy * d);
        if ( ( this.fx == -1 && x > -this.maxX ) || ( this.fx == 1 && x < 0 ) || 
                ( this.fy == -1 && y > -this.maxY ) || ( this.fy == 1 && y < 0 ) ) 
       {
            this.lastTime = now;
            this.shiftTo(x, y);
            this.on_scroll(x, y);
        } else {
            clearInterval(this.timerId); this.timerId = 0;
            this.shiftTo(this.endX, this.endY);
            this.on_scroll(this.endX, this.endY);
            this.on_scroll_end(this.endX, this.endY);
        }
    }
}

// when scrolling has ceased (mouseout/up)
dw_scrollObj.prototype.ceaseScroll = function() {
    if (!this.ready) return;
    if (this.timerId) {
        clearInterval(this.timerId); this.timerId = 0; 
    }
    this.on_scroll_stop(this.x, this.y); 
}

// glide onclick scrolling
dw_scrollObj.prototype.initScrollByVals = function(dx, dy, dur) {
    if ( !this.ready || this.sliding ) return;
    this.startX = this.x; this.startY = this.y;
    this.destX = this.destY = this.distX = this.distY = 0;
    if (dy < 0) {
        this.distY = (this.startY + dy >= -this.maxY)? dy: -(this.startY  + this.maxY);
    } else if (dy > 0) {
        this.distY = (this.startY + dy <= 0)? dy: -this.startY;
    }
    if (dx < 0) {
        this.distX = (this.startX + dx >= -this.maxX)? dx: -(this.startX + this.maxX);
    } else if (dx > 0) {
        this.distX = (this.startX + dx <= 0)? dx: -this.startX;
    }
    this.destX = this.startX + this.distX; this.destY = this.startY + this.distY;
    this.glideScrollPrep(this.destX, this.destY, dur);
}

dw_scrollObj.prototype.initScrollToVals = function(destX, destY, dur) {
    if ( !this.ready || this.sliding ) return;
    this.startX = this.x; this.startY = this.y;
    this.destX = -Math.max( Math.min(destX, this.maxX), 0);
    this.destY = -Math.max( Math.min(destY, this.maxY), 0);
    this.distY = this.destY - this.startY;
    this.distX = this.destX - this.startX;
    this.glideScrollPrep(this.destX, this.destY, dur);
}

dw_scrollObj.prototype.glideScrollPrep = function(destX, destY, dur) {
    this.slideDur = (typeof dur == 'number')? dur: dw_scrollObj.defaultSlideDur;
    this.per = Math.PI/(2 * this.slideDur); this.sliding = true;
    this.lyr = document.getElementById(this.lyrId); 
    this.startTime = new Date().getTime();
    this.timerId = setInterval(this.animString + ".doGlideScroll()",10);
    this.on_glidescroll_start(this.startX, this.startY);
}

dw_scrollObj.prototype.doGlideScroll = function() {
    var elapsed = new Date().getTime() - this.startTime;
    if (elapsed < this.slideDur) {
        var x = this.startX + Math.round( this.distX * Math.sin(this.per*elapsed) );
        var y = this.startY + Math.round( this.distY * Math.sin(this.per*elapsed) );
        this.shiftTo(x, y); 
        this.on_glidescroll(x, y);
    } else {	// if time's up
        clearInterval(this.timerId); this.timerId = 0; this.sliding = false;
        this.shiftTo(this.destX, this.destY);
        this.on_glidescroll(this.destX, this.destY);
        this.on_glidescroll_stop(this.destX, this.destY);
        // end of axis reached ? 
        if ( this.distX && (this.destX == 0 || this.destX == -this.maxX) 
          || this.distY && (this.destY == 0 || this.destY == -this.maxY) ) { 
            this.on_glidescroll_end(this.destX, this.destY);
        } 
    }
}

//  resource: http://adomas.org/javascript-mouse-wheel/
dw_scrollObj.handleMouseWheel = function(id, delta) {
    var wndo = dw_scrollObj.col[id];
    var x = wndo.x;
    var y = wndo.y;
    wndo.on_scroll_start(x,y);
    var ny;
    ny = 12  * delta + y
    ny = (ny < 0 && ny >= -wndo.maxY)? ny: (ny < -wndo.maxY)? -wndo.maxY: 0;
    wndo.shiftTo(x, ny);
    wndo.on_scroll(x, ny);
}

dw_scrollObj.doOnMouseWheel = function(e) {
    var delta = 0;
    if (!e) e = window.event;
    if (e.wheelDelta) { /* IE/Opera. */
        delta = e.wheelDelta/120;
        if (window.opera) delta = -delta;
    } else if (e.detail) { // Mozilla 
        delta = -e.detail/3;
    }
    if (delta) { // > 0 up, < 0 down
        dw_scrollObj.handleMouseWheel(this.id, delta);
    }
    if (e.preventDefault) e.preventDefault();
    e.returnValue = false;
}

dw_scrollObj.GeckoTableBugFix = function() {} // no longer need old bug fix

// Get position of el within layer (oCont) sOff: 'left' or 'top'
// Assumes el is within oCont
function dw_getLayerOffset(el, oCont, sOff) {
    var off = "offset" + sOff.charAt(0).toUpperCase() + sOff.slice(1);
    var val = el[off];
    while ( (el = el.offsetParent) != oCont ) 
        val += el[off];
    var clientOff = off.replace("offset", "client");
    if ( el[clientOff] ) val += el[clientOff];
    return val;
}


/***********************************************************
 *
 * 		dw_scrollbar.js
 *
 ***********************************************************/

/*************************************************************************
    This code is from Dynamic Web Coding at dyn-web.com
    Copyright 2001-2008 by Sharon Paine 
    See Terms of Use at www.dyn-web.com/business/terms.php
    regarding conditions under which you may use this code.
    This notice must be retained in the code as is!
    
    for use with dw_scroll.js - provides scrollbar functionality
    version date: Aug 2008
        bug fixed in .setBarSize
*************************************************************************/

function dw_Slidebar(barId, trackId, axis, x, y) {
    var bar = document.getElementById(barId);
    var track = document.getElementById(trackId);
    this.barId = barId; this.trackId = trackId;
    this.axis = axis; this.x = x || 0; this.y = y || 0;
    dw_Slidebar.col[this.barId] = this;
    this.bar = bar;  this.shiftTo(x, y);
    
    // hold for setBarSize  
    this.trkHt = track.offsetHeight; 
    this.trkWd = track.offsetWidth; 

    if (axis == 'v') {
        this.maxY = this.trkHt - bar.offsetHeight - y; 
        this.maxX = x; this.minX = x; this.minY = y;
    } else {
        this.maxX = this.trkWd - bar.offsetWidth - x; 
        this.minX = x; this.maxY = y; this.minY = y;
    }
    
    this.on_drag_start =  this.on_drag =   this.on_drag_end = 
    this.on_slide_start = this.on_slide =  this.on_slide_end = function() {}
    
    bar.onmousedown = dw_Slidebar.prepDrag; 
    // pass barId to obtain instance from dw_Slidebar.col
    track.onmousedown = function(e) { dw_Slidebar.prepSlide(barId, e); }
    this.bar = bar = null; track = null; 
}

dw_Slidebar.col = {}; // hold instances for global access
dw_Slidebar.current = null; // hold current instance

dw_Slidebar.prototype.slideDur = 500;

// track received onmousedown event
dw_Slidebar.prepSlide = function(barId, e) {
    var _this = dw_Slidebar.col[barId];
    dw_Slidebar.current = _this;
    var bar = _this.bar = document.getElementById(barId);
    
    if ( _this.timer ) { clearInterval(_this.timer); _this.timer = 0; }
    e = e? e: window.event;
    
    e.offX = (typeof e.layerX != "undefined")? e.layerX: e.offsetX;
    e.offY = (typeof e.layerY != "undefined")? e.layerY: e.offsetY;
    _this.startX = parseInt(bar.style.left); _this.startY = parseInt(bar.style.top);

    if (_this.axis == "v") {
        _this.destX = _this.startX;
        _this.destY = (e.offY < _this.startY)? e.offY: e.offY - bar.offsetHeight;
        _this.destY = Math.min( Math.max(_this.destY, _this.minY), _this.maxY );
    } else {
        _this.destX = (e.offX < _this.startX)? e.offX: e.offX - bar.offsetWidth;
        _this.destX = Math.min( Math.max(_this.destX, _this.minX), _this.maxX );
        _this.destY = _this.startY;
    }
    _this.distX = _this.destX - _this.startX; _this.distY = _this.destY - _this.startY;
    _this.per = Math.PI/(2 * _this.slideDur);
    _this.slideStartTime = new Date().getTime();
    _this.on_slide_start(_this.startX, _this.startY);
    _this.timer = setInterval("dw_Slidebar.doSlide()", 10);
}

dw_Slidebar.doSlide = function() {
    var _this = dw_Slidebar.current;
    var elapsed = new Date().getTime() - _this.slideStartTime;
    if (elapsed < _this.slideDur) {
        var x = _this.startX + _this.distX * Math.sin(_this.per*elapsed);
        var y = _this.startY + _this.distY * Math.sin(_this.per*elapsed);
        _this.shiftTo(x,y);
        _this.on_slide(x, y);
    } else {	// if time's up
        clearInterval(_this.timer);
        _this.shiftTo(_this.destX,  _this.destY);
        _this.on_slide(_this.destX,  _this.destY);
        _this.on_slide_end(_this.destX, _this.destY);
        dw_Slidebar.current = null;
    }    
}

dw_Slidebar.prepDrag = function (e) { 
    var bar = this; // bar received onmousedown event
    var barId = this.id; // id of element mousedown event assigned to
    var _this = dw_Slidebar.col[barId]; // Slidebar instance
    dw_Slidebar.current = _this;
    _this.bar = bar;
    e = dw_Event.DOMit(e);
    if ( _this.timer ) { clearInterval(_this.timer); _this.timer = 0; }
    _this.downX = e.clientX; _this.downY = e.clientY;
    _this.startX = parseInt(bar.style.left);
    _this.startY = parseInt(bar.style.top);
    _this.on_drag_start(_this.startX, _this.startY);
    dw_Event.add( document, "mousemove", dw_Slidebar.doDrag, true );
    dw_Event.add( document, "mouseup",   dw_Slidebar.endDrag,  true );
    e.stopPropagation(); e.preventDefault();
}

dw_Slidebar.doDrag = function(e) {
    if ( !dw_Slidebar.current ) return; // avoid errors in ie if inappropriate selections
    var _this = dw_Slidebar.current;
    var bar = _this.bar;
    e = dw_Event.DOMit(e);
    var nx = _this.startX + e.clientX - _this.downX;
    var ny = _this.startY + e.clientY - _this.downY;
    nx = Math.min( Math.max( _this.minX, nx ), _this.maxX);
    ny = Math.min( Math.max( _this.minY, ny ), _this.maxY);
    _this.shiftTo(nx, ny);
    _this.on_drag(nx, ny);
    e.preventDefault(); e.stopPropagation();
}

dw_Slidebar.endDrag = function() {
    if ( !dw_Slidebar.current ) return; // avoid errors in ie if inappropriate selections
    var _this = dw_Slidebar.current;
    var bar = _this.bar;
    dw_Event.remove( document, "mousemove", dw_Slidebar.doDrag, true );
    dw_Event.remove( document, "mouseup",   dw_Slidebar.endDrag,  true );
    _this.on_drag_end( parseInt(bar.style.left), parseInt(bar.style.top) );
    dw_Slidebar.current = null;
}

dw_Slidebar.prototype.shiftTo = function(x, y) {
    if ( this.bar ) {
        this.bar.style.left = x + "px";
        this.bar.style.top = y + "px";
    }
}

/////////////////////////////////////////////////////////////////////
//  connect slidebar with scrollObj
dw_scrollObj.prototype.setUpScrollbar = function(barId, trkId, axis, offx, offy, bSize) {
    var scrollbar = new dw_Slidebar(barId, trkId, axis, offx, offy);
    if (axis == "v") {
        this.vBarId = barId; 
    } else {
        this.hBarId = barId;
    }
    scrollbar.wndoId = this.id;
    scrollbar.bSizeDragBar = (bSize == false)? false: true; 
    if (scrollbar.bSizeDragBar) {
		//alert('451 dw_scrollObj.prototype.setUpScrollbar');
        dw_Scrollbar_Co.setBarSize(this, scrollbar);
		
    }
    dw_Scrollbar_Co.setEvents(this, scrollbar);
}

// Coordinates slidebar and scrollObj 
dw_Scrollbar_Co = {
    
    // This function is called for each scrollbar attached to a scroll area (change from previous version)
    setBarSize: function(scrollObj, barObj) {
        var lyr = document.getElementById(scrollObj.lyrId);
        var wn = document.getElementById(scrollObj.id);
		
        if ( barObj.axis == 'v' ) {
            var bar = document.getElementById(scrollObj.vBarId);
            bar.style.height = (lyr.offsetHeight > wn.offsetHeight)? 
                barObj.trkHt / ( lyr.offsetHeight / wn.offsetHeight ) + "px":             
                barObj.trkHt - ( 2 * barObj.minY ) + "px";
            barObj.maxY = barObj.trkHt - bar.offsetHeight - barObj.minY; 
        } else if ( barObj.axis == 'h' ) {
            var bar = document.getElementById(scrollObj.hBarId);
			if ( scrollObj.wd > wn.offsetWidth){
				bar.style.width = barObj.trkWd / ( scrollObj.wd / wn.offsetWidth ) + "px";				
			}
			else{
				bar.style.width = barObj.trkWd - ( 2 * barObj.minX ) + "px";
			}
            barObj.maxX = barObj.trkWd - bar.offsetWidth - barObj.minX;
			//alert(bar.style.width);
			//console.log("scrollObj.wd = " + scrollObj.wd);
			//console.log("wn.offsetWidth = " + wn.offsetWidth);
			//console.log("bar.style.width " + bar.style.width);
        }
    },
    
    // Find bars associated with this scrollObj. if they have bSizeDragBar set true reevaluate size and reset position to top 
    resetBars: function(scrollObj) {
		
        var barObj, bar;
        if (scrollObj.vBarId) {
            barObj = dw_Slidebar.col[scrollObj.vBarId];
            bar = document.getElementById(scrollObj.vBarId);
            bar.style.left = barObj.minX + "px"; bar.style.top = barObj.minY + "px";
            if (barObj.bSizeDragBar) {
                dw_Scrollbar_Co.setBarSize(scrollObj, barObj);
            }
        }
        if (scrollObj.hBarId) {
            barObj = dw_Slidebar.col[scrollObj.hBarId];
            bar = document.getElementById(scrollObj.hBarId);
            bar.style.left = barObj.minX + "px"; bar.style.top = barObj.minY + "px";
            if (barObj.bSizeDragBar) {				
                dw_Scrollbar_Co.setBarSize(scrollObj, barObj);
            }
        }
    },
    
    setEvents: function(scrollObj, barObj) {
        // scrollObj
        this.addEvent(scrollObj, 'on_load', function() { dw_Scrollbar_Co.resetBars(scrollObj); } );
        this.addEvent(scrollObj, 'on_scroll_start', function() { dw_Scrollbar_Co.getBarRefs(scrollObj) } );
        this.addEvent(scrollObj, 'on_glidescroll_start', function() { dw_Scrollbar_Co.getBarRefs(scrollObj) } );
        this.addEvent(scrollObj, 'on_scroll', function(x,y) { dw_Scrollbar_Co.updateScrollbar(scrollObj, x, y) } );
        this.addEvent(scrollObj, 'on_glidescroll', function(x,y) { dw_Scrollbar_Co.updateScrollbar(scrollObj, x, y) } );
        this.addEvent(scrollObj, 'on_scroll_stop', function(x,y) { dw_Scrollbar_Co.updateScrollbar(scrollObj, x, y); } );
        this.addEvent(scrollObj, 'on_glidescroll_stop', function(x,y) { dw_Scrollbar_Co.updateScrollbar(scrollObj, x, y); } );
        this.addEvent(scrollObj, 'on_scroll_end', function(x,y) { dw_Scrollbar_Co.updateScrollbar(scrollObj, x, y); } );
        this.addEvent(scrollObj, 'on_glidescroll_end', function(x,y) { dw_Scrollbar_Co.updateScrollbar(scrollObj, x, y); } );
            
        // barObj 
        this.addEvent(barObj, 'on_slide_start', function() { dw_Scrollbar_Co.getWndoLyrRef(barObj) } );
        this.addEvent(barObj, 'on_drag_start', function() { dw_Scrollbar_Co.getWndoLyrRef(barObj) } );
        this.addEvent(barObj, 'on_slide', function(x,y) { dw_Scrollbar_Co.updateScrollPosition(barObj, x, y) } );
        this.addEvent(barObj, 'on_drag', function(x,y) { dw_Scrollbar_Co.updateScrollPosition(barObj, x, y) } );
        this.addEvent(barObj, 'on_slide_end', function(x,y) { dw_Scrollbar_Co.updateScrollPosition(barObj, x, y); } );
        this.addEvent(barObj, 'on_drag_end', function(x,y) { dw_Scrollbar_Co.updateScrollPosition(barObj, x, y); } );
    
    },
    
    // Provide means to add functions to be invoked on pseudo events (on_load, on_scroll, etc) 
    // without overwriting any others that may already be set
    // by Mark Wubben (see http://simonwillison.net/2004/May/26/addLoadEvent/)
    addEvent: function(o, ev, fp) {
        var oldEv = o[ev];
        if ( typeof oldEv != 'function' ) {
            //o[ev] = fp;
            // almost all the functions (on_scroll, on_drag, etc.) pass x,y
            o[ev] = function (x,y) { fp(x,y); }
        } else {
            o[ev] = function (x,y) {
                  oldEv(x,y );
                  fp(x,y);
            }
        }
    },

    // Keep position of dragBar in sync with position of layer when scrolled by other means (mouseover, etc.)
    updateScrollbar: function(scrollObj, x, y) { // 
        var nx, ny;
        if ( scrollObj.vBar && scrollObj.maxY ) { 
            var vBar = scrollObj.vBar;
            ny = -( y * ( (vBar.maxY - vBar.minY) / scrollObj.maxY ) - vBar.minY );
            ny = Math.min( Math.max(ny, vBar.minY), vBar.maxY);  
            if (vBar.bar) { // ref to bar el
                nx = parseInt(vBar.bar.style.left);
                vBar.shiftTo(nx, ny);
            }
        }
        if ( scrollObj.hBar && scrollObj.maxX ) {
            var hBar = scrollObj.hBar;
            nx = -( x * ( (hBar.maxX - hBar.minX) / scrollObj.maxX ) - hBar.minX );
            nx = Math.min( Math.max(nx, hBar.minX), hBar.maxX);
            if (hBar.bar) {
                ny = parseInt(hBar.bar.style.top);
                hBar.shiftTo(nx, ny);
            }
        }
    },

    updateScrollPosition: function(barObj, x, y) { // on scrollbar movement
        var nx, ny; var wndo = barObj.wndo; 
        if ( !wndo.lyr ) {
            wndo.lyr = document.getElementById(wndo.lyrId);
        }
        if (barObj.axis == "v") {
            nx = wndo.x; // floating point values for loaded layer's position held in shiftTo method
            ny = -(y - barObj.minY) * ( wndo.maxY / (barObj.maxY - barObj.minY) ) || 0;
        } else {
            ny = wndo.y;
            nx = -(x - barObj.minX) * ( wndo.maxX / (barObj.maxX - barObj.minX) ) || 0;
        }
        wndo.shiftTo(nx, ny);
    },
    
    // Scroll area may have both vertical and horizontal bars 
    getBarRefs: function(scrollObj) { // References to Slidebar instance and dom element 
        if ( scrollObj.vBarId ) {
            scrollObj.vBar = dw_Slidebar.col[scrollObj.vBarId];
            scrollObj.vBar.bar = document.getElementById(scrollObj.vBarId);
        }
        if ( scrollObj.hBarId ) {
            scrollObj.hBar = dw_Slidebar.col[scrollObj.hBarId];
            scrollObj.hBar.bar = document.getElementById(scrollObj.hBarId);
        }
    },
    
    getWndoLyrRef: function(barObj) {
        var wndo = barObj.wndo = dw_scrollObj.col[ barObj.wndoId ];
        if ( wndo && !wndo.lyr ) {
            wndo.lyr = document.getElementById(wndo.lyrId);
        }
    }

}


/***********************************************************
 *
 * 		html_att_ev.js
 *
 ***********************************************************/
/*
    intermediary functions to prevent errors before page loaded 
    when using html element event handler attributes
    Provided for backwards compatibility with updated dw_scroll.js 
*/

dw_scrollObj.loadLayer = function(wndoId, lyrId, horizId) {
    if ( dw_scrollObj.col[wndoId] ) dw_scrollObj.col[wndoId].load(lyrId, horizId);
}

// Support for mouseover/down scrolling at any angle has been removed 
dw_scrollObj.initScroll = function(wndoId, dir, speed) {
    var deg = dir == 'up'? 90: dir == 'down'? 270: dir == 'left'? 180: dir == 'right'? 0: dir;
    if ( deg != null && dw_scrollObj.col[wndoId] ) {
        dw_scrollObj.col[wndoId].initScrollVals(deg, speed);
    }
}

dw_scrollObj.stopScroll = function(wndoId) {
    if ( dw_scrollObj.col[wndoId] ) dw_scrollObj.col[wndoId].ceaseScroll();
}

// increase speed onmousedown of scroll links (for mouseover scrolling)
dw_scrollObj.doubleSpeed = function(wndoId) {
    if ( dw_scrollObj.col[wndoId] ) dw_scrollObj.col[wndoId].speed *= 2;
}

dw_scrollObj.resetSpeed = function(wndoId) {
    if ( dw_scrollObj.col[wndoId] ) dw_scrollObj.col[wndoId].speed /= 2;
}

// for glide onclick scrolling 
dw_scrollObj.scrollBy = function(wndoId, x, y, dur) {
    if ( dw_scrollObj.col[wndoId] ) dw_scrollObj.col[wndoId].initScrollByVals(x, y, dur);
}

dw_scrollObj.scrollTo = function(wndoId, x, y, dur) {
    if ( dw_scrollObj.col[wndoId] ) dw_scrollObj.col[wndoId].initScrollToVals(x, y, dur);
}


/***********************************************************
 *
 * 		scoll_controls.js
 *
 ***********************************************************/

/*************************************************************************
    This code is from Dynamic Web Coding at dyn-web.com
    Copyright 2008 by Sharon Paine 
    See Terms of Use at www.dyn-web.com/business/terms.php
    regarding conditions under which you may use this code.
    This notice must be retained in the code as is!

    unobtrusive event handling for use with dw_scroll.js
    version date: Aug 2008
*************************************************************************/

/////////////////////////////////////////////////////////////////////
// two ways to add style sheet for capable browsers

dw_writeStyleSheet = function(file) {
    document.write('<link rel="stylesheet" href="' + file + '" media="screen" />');
}

function dw_addLinkCSS(file) {
    if ( !document.createElement ) return;
    var el = document.createElement("link");
    el.setAttribute("rel", "stylesheet");
    el.setAttribute("type", "text/css");
    el.setAttribute("media", "screen");
    el.setAttribute("href", file);
    document.getElementsByTagName('head')[0].appendChild(el);
}
/////////////////////////////////////////////////////////////////////

// Example class names: load_wn_lyr1, load_wn_lyr2_t2
dw_scrollObj.prototype.setUpLoadLinks = function(controlsId) {
    var wndoId = this.id; var el = document.getElementById(controlsId); 
    var links = el.getElementsByTagName('a');
    var cls, parts;
    for (var i=0; links[i]; i++) {
        cls = dw_scrollObj.get_DelimitedClass( links[i].className );
        parts = cls.split('_');
        if ( parts[0] == 'load' && parts[1] == wndoId && parts.length > 2 ) {
            // no checks on lyrId, horizId
            var lyrId = parts[2]; var horizId = parts[3]? parts[3]: null;
            dw_Event.add( links[i], 'click', function (wndoId, lyrId, horizId) {
                return function (e) {
                    dw_scrollObj.col[wndoId].load(lyrId, horizId);
                    if (e && e.preventDefault) e.preventDefault();
                    return false;
                }
            }(wndoId, lyrId, horizId) ); // see Crockford js good parts pg 39
        }
    }
}

dw_scrollObj.prototype.setUpScrollControls = function(controlsId, autoHide, axis) {
    var wndoId = this.id; var el = document.getElementById(controlsId); 
    if ( autoHide && axis == 'v' || axis == 'h' ) {
        dw_scrollObj.handleControlVis(controlsId, wndoId, axis);
        dw_Scrollbar_Co.addEvent( this, 'on_load', function() { dw_scrollObj.handleControlVis(controlsId, wndoId, axis); } );
        dw_Scrollbar_Co.addEvent( this, 'on_update', function() { dw_scrollObj.handleControlVis(controlsId, wndoId, axis); } );
    }
    
    var links = el.getElementsByTagName('a'), cls, eType;
    for (var i=0; links[i]; i++) { 
        cls = dw_scrollObj.get_DelimitedClass( links[i].className );
        eType = dw_scrollObj.getEv_FnType( cls.slice(0, cls.indexOf('_') ) );
        switch ( eType ) {
            case 'mouseover' :
            case 'mousedown' :
                dw_scrollObj.handleMouseOverDownLinks(links[i], wndoId, cls);
                break;
            case 'scrollToId': 
                dw_scrollObj.handleScrollToId(links[i], wndoId, cls);
                break;
            case 'scrollTo' :
            case 'scrollBy':
            case 'click': 
                dw_scrollObj.handleClick(links[i], wndoId, cls) ;
                break;
        }
    }
}

dw_scrollObj.handleMouseOverDownLinks = function (linkEl, wndoId, cls) {
    var parts = cls.split('_'); var eType = parts[0];
    var re = /^(mouseover|mousedown)_(up|down|left|right)(_[\d]+)?$/;
                
    if ( re.test(cls) ) { 
        var eAlt = (eType == 'mouseover')? 'mouseout': 'mouseup';
        var dir = parts[1];  var speed = parts[2] || null; 
        var deg = (dir == 'up')? 90: (dir == 'down')? 270: (dir == 'left')? 180: 0;
        
        dw_Event.add(linkEl, eType, function (e) { dw_scrollObj.col[wndoId].initScrollVals(deg, speed); } );
        dw_Event.add(linkEl, eAlt, function (e) { dw_scrollObj.col[wndoId].ceaseScroll(); } );
            
        if ( eType == 'mouseover') {
            dw_Event.add( linkEl, 'mousedown', function (e) { dw_scrollObj.col[wndoId].speed *= 3; } );
            dw_Event.add( linkEl, 'mouseup', function (e) { 
                dw_scrollObj.col[wndoId].speed = dw_scrollObj.prototype.speed; } ); 
        }
        dw_Event.add( linkEl, 'click', function(e) { if (e && e.preventDefault) e.preventDefault(); return false; } );
    }
}

// scrollToId_smile, scrollToId_smile_100, scrollToId_smile_lyr1_100    
dw_scrollObj.handleScrollToId = function (linkEl, wndoId, cls) {
    var parts = cls.split('_'); var id = parts[1], lyrId, dur;
    if ( parts[2] ) {
        if ( isNaN( parseInt(parts[2]) ) ) { 
            lyrId = parts[2];
            dur = ( parts[3] && !isNaN( parseInt(parts[3]) ) )? parseInt(parts[3]): null;
        } else {
            dur = parseInt( parts[2] );
        }
    }
    dw_Event.add( linkEl, 'click', function (e) {
            dw_scrollObj.scrollToId(wndoId, id, lyrId, dur);
            if (e && e.preventDefault) e.preventDefault();
            return false;
        } );
}

// doesn't checks if lyrId in wndo, el in lyrId
dw_scrollObj.scrollToId = function(wndoId, id, lyrId, dur) {
    var wndo = dw_scrollObj.col[wndoId];
    var el = document.getElementById(id);
    if (el) {
        if ( lyrId ) {
            if ( document.getElementById(lyrId) && wndo.lyrId != lyrId ) {
                wndo.load(lyrId);
            }
        }
        var lyr = document.getElementById(wndo.lyrId);
        var x = dw_getLayerOffset(el, lyr, 'left');
        var y = dw_getLayerOffset(el, lyr, 'top');
        wndo.initScrollToVals(x, y, dur);
    }
}

dw_scrollObj.handleClick = function (linkEl, wndoId, cls) {
    var wndo = dw_scrollObj.col[wndoId];
    var parts = cls.split('_'); var eType = parts[0]; 
    var dur_re = /^([\d]+)$/; var fn, re, x, y, dur;
    
    switch (eType) {
        case 'scrollTo' :
            fn = 'scrollTo';  re = /^(null|end|[\d]+)$/;
            x = re.test( parts[1] )? parts[1]: '';
            y = re.test( parts[2] )? parts[2]: '';
            dur = ( parts[3] && dur_re.test(parts[3]) )? parts[3]: null;
            break;
        case 'scrollBy': // scrollBy_m30_m40, scrollBy_null_m100, scrollBy_100_null
            fn = 'scrollBy';  re = /^(([m]?[\d]+)|null)$/;
            x = re.test( parts[1] )? parts[1]: '';
            y = re.test( parts[2] )? parts[2]: '';
            
            // negate numbers (m not - but vice versa) 
            if ( !isNaN( parseInt(x) ) ) {
                x = -parseInt(x);
            } else if ( typeof x == 'string' ) {
                x = x.indexOf('m') !=-1 ? x.replace('m', ''): x;
            }
            if ( !isNaN( parseInt(y) ) ) {
                y = -parseInt(y);
            } else if ( typeof y == 'string' ) {
                y = y.indexOf('m') !=-1 ? y.replace('m', ''): y;
            }
            
            dur = ( parts[3] && dur_re.test(parts[3]) )? parts[3]: null;
            break;
        
        case 'click': 
            var o = dw_scrollObj.getClickParts(cls);
            fn = o.fn; x = o.x; y = o.y; dur = o.dur;
            break;
    }
    if ( x !== '' && y !== '' ) {
        if (x == 'end') { x = wndo.maxX; }
        if (y == 'end') { y = wndo.maxY; }
        if (x === 'null' || x === null) { x = wndo.x; }
        if (y === 'null' || y === null) { y = wndo.y; }
        
        x = parseInt(x); y = parseInt(y);  
        dur = !isNaN( parseInt(dur) )? parseInt(dur): null;
        
        if (fn == 'scrollBy') {
            dw_Event.add( linkEl, 'click', function (e) {
                    dw_scrollObj.col[wndoId].initScrollByVals(x, y, dur);
                    if (e && e.preventDefault) e.preventDefault();
                    return false;
                } );
        } else if (fn == 'scrollTo') {
            dw_Event.add( linkEl, 'click', function (e) {
                    dw_scrollObj.col[wndoId].initScrollToVals(x, y, dur);
                    if (e && e.preventDefault) e.preventDefault();
                    return false;
                } );
        }
    }
}

// get info from className (e.g., click_down_by_100)
dw_scrollObj.getClickParts = function(cls) {
    var parts = cls.split('_');
    var re = /^(up|down|left|right)$/;
    var dir, fn = '', dur, ar, val, x = '', y = '';
    
    if ( parts.length >= 4 ) {
        ar = parts[1].match(re);
        dir = ar? ar[1]: null;
            
        re = /^(to|by)$/; 
        ar = parts[2].match(re);
        if (ar) {
            fn = (ar[0] == 'to')? 'scrollTo': 'scrollBy';
        } 
    
        val = parts[3]; // value on x or y axis
        re = /^([\d]+)$/;
        dur = ( parts[4] && re.test(parts[4]) )? parts[4]: null;
    
        switch (fn) {
            case 'scrollBy' :
                if ( !re.test( val ) ) {
                    x = ''; y = ''; break;
                }
                switch (dir) { // 0 for unspecified axis 
                    case 'up' : x = 0; y = val; break;
                    case 'down' : x = 0; y = -val; break;
                    case 'left' : x = val; y = 0; break;
                    case 'right' : x = -val; y = 0;
                 }
                break;
            case 'scrollTo' :
                re = /^(end|[\d]+)$/;
                if ( !re.test( val ) ) {
                    x = ''; y = ''; break;
                }
                switch (dir) { // null for unspecified axis 
                    case 'up' : x = null; y = val; break;
                    case 'down' : x = null; y = (val == 'end')? val: -val; break;
                    case 'left' : x = val; y = null; break;
                    case 'right' : x = (val == 'end')? val: -val; y = null;
                 } 
                break;
         }
    }
    return { fn: fn, x: x, y: y, dur: dur }
}

dw_scrollObj.getEv_FnType = function(str) {
    var re = /^(mouseover|mousedown|scrollBy|scrollTo|scrollToId|click)$/;
    if (re.test(str) ) {
        return str;
    }
    return '';
}

// return class name with underscores in it 
dw_scrollObj.get_DelimitedClass = function(cls) {
    if ( cls.indexOf('_') == -1 ) {
        return '';
    }
    var whitespace = /\s+/;
    if ( !whitespace.test(cls) ) {
        return cls;
    } else {
        var classes = cls.split(whitespace); 
        for(var i = 0; classes[i]; i++) { 
            if ( classes[i].indexOf('_') != -1 ) {
                return classes[i];
            }
        }
    }
}

dw_scrollObj.handleControlVis = function(controlsId, wndoId, axis) {
    var wndo = dw_scrollObj.col[wndoId];
    var el = document.getElementById(controlsId);
    if ( ( axis == 'v' && wndo.maxY > 0 ) || ( axis == 'h' && wndo.maxX > 0 ) ) {
        el.style.visibility = 'visible';
    } else {
        el.style.visibility = 'hidden';
    }
}


/***********************************************************
 *
 * 		scroll_init.js
 *
 ***********************************************************/

function init_dw_Scroll() {
    var wndo = new dw_scrollObj('wn', 'veh_jellybeans');
    wndo.setUpScrollbar("dragBar", "track", "h", 10, 1, true);
    wndo.setUpScrollControls('scrollbar',true,'h');
    //wndo.setUpLoadLinks('view_controls');
}

// if necessary objects exists link in the style sheet and call the init function onload
if ( document.getElementById && document.getElementsByTagName ) {
    //dw_writeStyleSheet('css/scroll.css');
    addLoadEvent(init_dw_Scroll);
}


/***********************************************************
 *
 * 		scroller.js
 *
 ***********************************************************/
var xmlInitial, xml;
var lang = (location.href.indexOf("french") != -1) ? "fr" : "en";
// var make (defined in head_common.jsp)

var make_arr = new Array();
make_arr['pontiac']		= "make=pontiac";
make_arr['buick']			= "make=buick";
make_arr['gmc']				= "make=GMC";
make_arr['chevrolet']	= "make=chevrolet";
make_arr['cadillac']	= "make=cadillac";
make_arr['hummer']		= "make=hummer";

var filter_arr = new Array();
filter_arr['ftr_all'] = "";
filter_arr['ftr_hyb'] = "hybrid=true";
filter_arr['ftr_car'] = "bodyStyles=small,full,midsize,luxury,sports";
filter_arr['ftr_suv'] = "bodyStyles=SUV";
filter_arr['ftr_trk'] = "bodyStyles=pickup";
filter_arr['ftr_van'] = "bodyStyles=van";
filter_arr['ftr_awd'] = "driveTypeAllWheel=true";
filter_arr['ftr_upc'] = "type=upcoming";

var sorter_arr = new Array();
sorter_arr['srt_msrp'] = "sortBy=price";
sorter_arr['srt_fuel'] = "sortBy=fuel";
sorter_arr['srt_seat'] = "sortBy=seating";
sorter_arr['srt_hpow'] = "sortBy=power";
sorter_arr['srt_towi'] = "sortBy=tow";
sorter_arr['asc'] = "sortOrder=asc";
sorter_arr['dsc'] = "sortOrder=desc";


$(document).ready(function(){

	/*============= onLoad ==============*/
	// Load XML data and populate scroller
	getInitialVehicles("showroomId=2&makes=" + make + "&" + make_arr[make] + "&" + filter_arr['ftr_all'] + "&" + sorter_arr['srt_msrp'] + "&" + sorter_arr['asc'])
	
	// Load flash defined in <make>_<lang>_CA.jsp
	loadFlash();
	
	// Since using divs as target, make sure cursor looks correct
	$("#veh_sorter").css("cursor", "default");

	//Automatically close sorter tab, only if hockey card is not showing
	t=setTimeout(function(){
		if ($("#veh_hcard").css("display") == 'none') moveSorter();
	}, 2500);
	
	//Check for PC cookie & Set Cash Credit copy within sorter
	$("#cda").html(checkPostal());

	/*============= onChange ==============*/
	// Hockey card mouse on/off
	$("#veh_hcard").hover(function(){
		$('#flashContent').hide();
		$('#veh_crunch').fadeIn();
	},function(){
		$('#flashContent').show();
		$('#veh_hcard').hide();
		//$("#jtable div").css({opacity:"1"});
	});
	
	$("#scrollbar, #tab").mouseover(function(){
		$('#veh_hcard').hide();
	});
	

	// Open/Close Panel button behaviour
	$("#tab").css("cursor","pointer").hover(function(){
		var myBot = $("#veh_sorter").css("bottom");
		if (myBot >="0px")
			var mySrc = "-15px 0";
		else
			var mySrc = "-45px 0";
		$('#tab div').css("background-position",mySrc);
		$('#tab').attr("class", "on");
	},function(){
		var myBot = $("#veh_sorter").css("bottom");
		if (myBot >="0px")
			var mySrc = "0 0";
		else
			var mySrc = "-30px 0";
		$('#tab div').css("background-position",mySrc);
		$('#tab').attr("class", "off");
	}).click(function(){
		clearTimeout(t);
		//close global nav if open
		if($("#GlobalNavTab").css("top") == "35px") toggleTab();
		moveSorter();
	});

	
}); // End Doc Ready

//Hide/Show sorter
function moveSorter() {
	//alert($("#veh_sorter").css("bottom"));
	var myBot = $("#veh_sorter").css("bottom");
	var moveTo;
	if (myBot >="0px") { // must be open, so close it
		switch(make) {
			case "buick":
				moveTo = "-175px"
				break;
			case "cadillac":
				moveTo = "-155px"
				break;
			case "hummer":
				moveTo = "-150px"
				break;
			default:
				moveTo = "-185px"
				break;
		}

		$('#veh_sorter').animate({
			bottom: moveTo
		},300, "linear", function() { 
			if (lang=="fr") $('#tab h3').html("CLIQUER ICI POUR CHOISIR UN V&#201;HICULE "+make.toUpperCase());
			else $('#tab h3').text("CLICK HERE TO SELECT A "+make.toUpperCase());
			if ($('#tab').attr("class") == "on")
				$('#tab div').css("background-position", "-45px 0");
			else
				$('#tab div').css("background-position", "-30px 0"); 
			} 
		);
		$('#veh_hcard,#veh_crunch').fadeOut();

	} else { // drawer closed, so open it
		$('#veh_sorter').animate({
			bottom:"0px"//,opacity:"1"
		},600, "linear", function() { 
			if (lang=="fr") $('#tab h3').text("CLIQUER ICI POUR FERMER LE PANNEAU");
			else $('#tab h3').text("CLICK HERE TO CLOSE PANEL");
			
			if ($('#tab').attr("class") == "on")
				$('#tab div').css("background-position", "-15px 0");
			else
				$('#tab div').css("background-position", "0 0"); 
			} 
		);
	}
}

// Loads XML feed based on sorter option passed in queryString
// creates the xmlInitial obj for use later
function getInitialVehicles(queryString) {
	var myLang = (lang=='fr')? 'french':'english';
	$.ajax({
		//url: '/xml/divisional/'+make+'_'+lang+'_CA.xml',
		url: "/gm/" + myLang + "/showroom/search",
		data: queryString,
		cache: false,
			type: 'GET',
		dataType: ($.browser.msie) ? "text" : "xml",
		//dataType: "xml",
		async:false,
			timeout: 3000,
		beforeSend: function(){
			loadingAnim();
		},
		error: function(){
				//alert('Error loading XML document');
		},
		success: function(xml){
		//work around for IE
		 var tempXML;
		 if (typeof xml == "string") {	
			 tempXML = new ActiveXObject("Microsoft.XMLDOM");
			 tempXML.async = false;
			 tempXML.loadXML(xml);
			 xml = tempXML; 
		 } 
			
			
			
			// Get the pricing Type MSRP or Purchase price and append the correct content to location.
	 $(xml).find("response:has(pricingType)").each(function(){																   
          var pricingTypeTemp = $(this).children("pricingType").text();
				if (pricingTypeTemp == "msrp") {
					pricingTypeEN ="MSRP*";	
					pricingTypeFR ="PDSF*";		
				}	
				else {
					pricingTypeEN ="&#8734;";
					pricingTypeFR ="&#8734;";
					$("#veh_hcard dd.msrp").html($("#veh_hcard dd.msrp").html().replace(/MSRP\*/,"&#8734;"));
					$("#veh_hcard dd.msrp").html($("#veh_hcard dd.msrp").html().replace(/PDSF\*/,"&#8734;"));
				}	
			
			});	
			createJtable(xml);
			xmlInitial = xml;
		}
	});	
}

// Loading graphic before AJAX
function loadingAnim(){
	var loadingMsg = (lang=="fr") ? "chargement..." : "loading...";
	$("#jtable tr").html('<td class="loading"><img src="/images/divisional/'+make.toLowerCase()+'/ajax-loader.gif" alt="'+ loadingMsg +'" /><\/td>');
	//dw_scrollObj.loadLayer('wn','veh_jellybeans');
}

// Populate jellybean table
function createJtable(xml) {
	$("#jtable tr").empty(); // clear table to start
	dw_scrollObj.loadLayer('wn','veh_jellybeans');
	if( $(xml).find('recordCount').text() == 0) {
		if (lang=="fr") {
			$("#jtable tr").html('<td class="loading">Non vehicules<\/td>');
		} else {
			$("#jtable tr").html('<td class="loading">Sorry, no vehicles.<\/td>');
		}
		dw_scrollObj.loadLayer('wn','veh_jellybeans');
		return false;
	}
	var tableTDs = '';
	$(xml).find('vehicle').each(function(){
		tableTDs += '<td><div id="div_'+ $("modelYearCode",this).text() +'">';
		//html += '<img src="/images/divisional/chevrolet/vehicles/jbean_2009_chev_aveo.jpg" alt="" />';
		// Unfortunately, since H2 SUT is the only vehicle jellybean which requires localization, we need to hard-code it
		if(make == "hummer" && lang=="fr" && $("modelName",this).text() == "h2sut") {
			tableTDs += '<a href="'+$("overviewLink",this).text()+'"><img src="/images/divisional/hummer/vehicles/jbean_2009_humm_h2cus.jpg" alt="H2 CUS" /></a>';
		} else {
			tableTDs += '<a href="'+$("overviewLink",this).text()+'"><img src="/images/divisional/' + make.toLowerCase() + '/vehicles/jbean_' + 	$("modelYearCode",this).text() + '.jpg" alt="' + $("modelNameDesc",this).text() + '" /></a>';
		}

		tableTDs += '<dl>';
		if($("modelYear",this).text() == "2011" && $("modelName",this).text() == "cts") {
			tableTDs += '<dt>' + $("modelYearDesc",this).text() + '<\/dt>';
		}
		else {
			tableTDs += '<dt>' + $("modelNameDesc",this).text() + '<\/dt>';
		}

		// If not upcoming vehicle AND there is a numeric value for MSRP
		var isUpcoming = ($("upcoming",this).text() == "true") ? true : false;
		var msrp = $("priceLow",this).text().replace(/\D/g, "");
		if(!isUpcoming && msrp > 0){
			if (lang=="fr") {
				//alert($("modelName",this).text() + " upcoming=" + $("upcoming",this).text());
				
				tableTDs += '<dd class="msrp">&#xC0; partir de ' + $("priceLow",this).text() +  pricingTypeFR +'<\/dd>';
			} else {
				
				tableTDs += '<dd class="msrp">FROM ' + $("priceLow",this).text() +  pricingTypeEN +'<\/dd>';
			}
		}
		var hasCC = ($("hasCashCredit",this).text() == 'true') ? true : false;
		if (hasCC) tableTDs += '<dd class="extra">' + $("cashCredit",this).text(); + '<\/dd>'
		
		tableTDs += '<\/dl><\/div><\/td>';
	}); // end Each vehicle

	$("#jtable").show();
	$("#jtable tr").html(tableTDs);

	//var numJbeans = $("#jtable td").length;
	//if (numJbeans < 6 && make != "hummer" && make != "buick") {centerJtable(numJbeans);}
	setTimeout ( "scrollBarDelay()", 300 );
	
	initCard(xml);

} // end createJtable

function scrollBarDelay(){
	//dw_scrollObj.loadLayer('wn','veh_jellybeans');
	// center again here after delay in case it failed above.
	var numJbeans = $("#jtable td").length;
	if (numJbeans < 6 && make != "hummer" && make != "buick") {centerJtable(numJbeans);}
}

function centerJtable(num) {
		switch(num) {
		case 1:
				$("#veh_jellybeans").css("left","412px");
				break;
		case 2:
				$("#veh_jellybeans").css("left","330px");
				break;
		case 3:
				$("#veh_jellybeans").css("left","247px");
				break;
		case 4:
				$("#veh_jellybeans").css("left","165px");
				break;
		case 5:
				$("#veh_jellybeans").css("left","82px");
				break;
		default:
				break;
		}
}

// We can only bind the hover event after the AJAX returns the data
// and the browser renders the full html
function initCard(xml) {
	$("#jtable td div").hover(function(){
		
		// Everytime a jellybean table div is hovered on, we need to:
		// - find which vehicle we are hovering over
		// - get all data from the xml to populate card
		// - position card
		// - show card
		//var $this = $(this);
		//var myc = $this.attr("id").replace("div_","");
		//var myVeh =$(xml).find("modelYearCode:contains("+myc+")").parent("vehicle");
	
		var myParent = $(this).parent("td");
		var index = $(myParent).parent("tr").find('> *').index(myParent);

		//setHcardValues(xml,index);

	//For some reason, :nth-child with XML doesn't work in IE
	//var myVeh = $(xml).find("vehicle:nth-child("+index+")");
	var myResource = $(xml).find("response").eq(index);
	var myVeh = $(xml).find("vehicle").eq(index);
	var modelName 	= $(myVeh).children("modelNameDesc").text();
	var crunchImg 	= "/images/divisional/" + make.toLowerCase() + "/vehicles/crunch_" + $(myVeh).children("modelYearCode").text() + "_" + lang + "_CA.jpg";
	var hcardImg 		= "/images/divisional/" + make.toLowerCase() + "/vehicles/hcard_" + $(myVeh).children("modelYearCode").text() + ".jpg";
	// Unfortunately, since H2 SUT is the only vehicle jellybean which requires localization, we need to hard-code it
	if(make == "hummer" && lang=="fr" && $(myVeh).children("modelNameDesc").text()  == "H2 SUT") {
		hcardImg 	= "/images/divisional/hummer/vehicles/hcard_2009_humm_h2cus.jpg";
	}
	
	//Set Img sources
	$("#veh_crunch img").attr("src",crunchImg).attr("alt",modelName);
	$("#hcardImg img").attr("src",hcardImg);//.attr("alt",modelName);
	
	var modelCode 	= $(myVeh).children("modelYearCode").text();
	var learnMore 	= $(myVeh).children("overviewLink").text();
	var gallery			= learnMore.replace("overview","gallery-photos-exterior01");	// gallery is simply overview replaced
	var compare			= $(myVeh).children("compareLink").text();
	var build				= $(myVeh).children("buildLink").text();
	var msrp 				= $(myVeh).children("priceLow").text();
	var hasCC 			= ($(myVeh).children("hasCashCredit").text() == 'true') ? true : false;
	var CC					=	$(myVeh).children("cashCredit").text();


																	

	//Set Links
	$("#veh_hcard ul li").eq(0).children("a").attr("href",learnMore);
	//Do not show links if Upcoming vehicle
	var isUpcoming = ($(myVeh).children("upcoming").text() == "true") ? true : false;
	if (isUpcoming) {
		$("#veh_hcard ul li:gt(0)").hide();
	} else {
		$("#veh_hcard ul li").eq(1).show().children("a").attr("href",gallery);
		$("#veh_hcard ul li").eq(2).show().children("a").attr("href",compare);
		$("#veh_hcard ul li").eq(3).show().children("a").attr("href",build);
	}
	$("#veh_hcard dt a, #veh_hcard > a").attr("href",learnMore);
	$("#veh_hcard dt a").html(modelName);
	if (isUpcoming) {
		if (msrp =="N/A" || msrp =="S.O.") $("#veh_hcard dd.msrp").hide();
	} else {
		$("#veh_hcard dd.msrp").show().children("span").html(msrp);
	}
	// Cash Incentives
	if (hasCC) { $("#veh_hcard dd.extra").show().html(CC);	}	
	else { $("#veh_hcard dd.extra").hide();}

		var myBot = $("#veh_sorter").css("bottom");
		if(myBot =="0px"){ //sorter is fully open

			$('#veh_crunch').show();

			var browser = $("body").innerWidth();
			var gutter = (browser - 990)/2;
			var xpos = $(this).offset().left;
			var jbeanWidth = $(this).innerWidth();
			var hcardWidth = $("#veh_hcard").innerWidth();
			var hcard_xpos = xpos + (jbeanWidth - hcardWidth)/2;


			// Make sure jtable thumbnail isn't partially obstructed
			var edgeRight = xpos + jbeanWidth;
			//console.log("xpos: "+xpos +" gutter: "+gutter+" right side: "+ edgeRight + " jbeanWidth: "+jbeanWidth);
			if (xpos < gutter){
				dw_scrollObj.scrollBy('wn', gutter - xpos, 0);
				hcard_xpos = gutter + (jbeanWidth - hcardWidth)/2;
			} else if (edgeRight > gutter + 990 ){
				dw_scrollObj.scrollBy('wn',gutter + 990 - edgeRight, 0);
				hcard_xpos = gutter + (jbeanWidth - hcardWidth)/2 + 990 - jbeanWidth;
			}
			
	//Set Img sources
	$("#veh_crunch img").attr("src",crunchImg).attr("alt",modelName);
	//Show hcard only after hcardImg has loaded (random courtesy of IE caching bug)
	$("#hcardImg img").attr("src",hcardImg + "?random=" + new Date().getTime()).load(function() {
			$('#veh_hcard').css({left:hcard_xpos}).show();//.dropShadow();
	});
			

		}
	},function(){
		// alert('out');
	});
}


/*
function setHcardValues(xml,index) {
	//For some reason, :nth-child with XML doesn't work in IE
	//var myVeh = $(xml).find("vehicle:nth-child("+index+")");
	var myVeh = $(xml).find("vehicle").eq(index);
	var modelName 	= $(myVeh).children("modelNameDesc").text();
	var crunchImg 	= "/images/divisional/" + make.toLowerCase() + "/vehicles/crunch_" + $(myVeh).children("modelYearCode").text() + "_" + lang + "_CA.jpg";
	var hcardImg 		= "/images/divisional/" + make.toLowerCase() + "/vehicles/hcard_" + $(myVeh).children("modelYearCode").text() + ".jpg";
	// Unfortunately, since H2 SUT is the only vehicle jellybean which requires localization, we need to hard-code it
	if(make == "hummer" && lang=="fr" && $(myVeh).children("modelNameDesc").text()  == "H2 SUT") {
		hcardImg 	= "/images/divisional/hummer/vehicles/hcard_2009_humm_h2cus.jpg";
	}
	//Set Img sources
	$("#veh_crunch img").attr("src",crunchImg).attr("alt",modelName);
	$("#hcardImg img").attr("src",hcardImg);//.attr("alt",modelName);
	
	var modelCode 	= $(myVeh).children("modelYearCode").text();
	var learnMore 	= $(myVeh).children("overviewLink").text();
	var gallery			= learnMore.replace("overview","gallery-photos-exterior01");	// gallery is simply overview replaced
	var compare			= $(myVeh).children("compareLink").text();
	var build				= $(myVeh).children("buildLink").text();
	var msrp 				= $(myVeh).children("priceLow").text();
	var hasCC 			= ($(myVeh).children("hasCashCredit").text() == 'true') ? true : false;
	var CC					=	$(myVeh).children("cashCredit").text();

	//Set Links
	$("#veh_hcard ul li").eq(0).children("a").attr("href",learnMore);
	//Do not show links if Upcoming vehicle
	var isUpcoming = ($(myVeh).children("upcoming").text() == "true") ? true : false;
	if (isUpcoming) {
		$("#veh_hcard ul li:gt(0)").hide();
	} else {
		$("#veh_hcard ul li").eq(1).show().children("a").attr("href",gallery);
		$("#veh_hcard ul li").eq(2).show().children("a").attr("href",compare);
		$("#veh_hcard ul li").eq(3).show().children("a").attr("href",build);
	}
	$("#veh_hcard dt a, #veh_hcard > a").attr("href",learnMore);
	$("#veh_hcard dt a").html(modelName);
	if (isUpcoming) {
		if (msrp =="N/A" || msrp =="S.O.") $("#veh_hcard dd.msrp").hide();
	} else {
		$("#veh_hcard dd.msrp").show().children("span").html(msrp);
	}
	if (hasCC) $("#veh_hcard dd.msrp").html(CC);
	
	return true;
}
*/

// Initiate scrolling layer after new data comes in
/*
dw_scrollObj.loadLayer = function(wndoId, lyrId, horizId) {
	if ( dw_scrollObj.col[wndoId] ) {
		dw_scrollObj.col[wndoId].load(lyrId, horizId);
	}
}
*/

// Depeding on whether the PC cookie exists, return the appropriate copy
function checkPostal() {
	var html;
	var myPC = getCookie("pc");
	
	$('#pc').val(myPC);

	
	//alert (myPC)
	
	//alert(escape(ParseQueryString("pc")));
	if(myPC) {// pc cookie exists
		html = (lang == "fr") ? 'Ajustements de prix dans votre r&#233;gion '+ myPC + ' <a href="javascript:checkJQueryUIDialogloaded()">(changer)</a>':'Price Adjustments shown for area '+ myPC + ' <a href="javascript:checkJQueryUIDialogloaded()">(change)</a>' ;
	} else {	// null
		html = (lang == "fr") ? '<a href="javascript:checkJQueryUIDialogloaded()">Inscrivez votre code postal</a> pour obtenir les ajustements de prix dans votre r&#233;gion. ':'For available Price Adjustments in your area please  <a href="javascript:checkJQueryUIDialogloaded()">provide your postal code</a>' ;
	}
	return html;
}


/***********************************************************
 *
 * 		features.js
 *
 ***********************************************************/

$(document).ready(function(){
	
	// pc cookie functionality
	
	var myPC = getCookie("pc");
	if(myPC) {
	//alert("pc cookie exists!")
	}
	
	else {
	var thisURL = location.toString();
	
	if (thisURL.indexOf('french') != -1){
			$('#pc').val("CODE POSTAL");
	}
		else {
				$('#pc').val("POSTAL CODE");
			}
	}
	
	//alert(document.cookie);
	
	
	if (( $('#pc').val()) != "POSTAL CODE" && ( $('#pc').val()) != "CODE POSTAL" && ( $('#pc').val()) != "")  { /*alert ("we have a cookie!")*/ }
	
	else {getVal();}
	
	
	
	//window.alert( $('#foo').val() );
	$("#pc").change(getVal);
	$("#pc").blur(getVal);
	
	function getVal() { 
	
			//alert("the value is " + ( $('#pc').val()) );
	
			if(validatePC($('#pc').val() )) {
			//alert($(this).val());
			//$.cookie("pc", $(this).val(), {path: '/'} );
			
			// Attempting to fix weird bug by using the cookie algo from the gmcl.js
			// - appeared to be 2 different cookies with same name, but different HOST values, not sure why/how
			
			var today = new Date();
			today.setTime(today.getTime());
			
			//Get 1 day in milliseconds
			var one_day=1000*60*60*24

			
			var cookieExpiresDate = new Date (today.getTime() + (30 * one_day));
			
			//alert("attempting to set cookie " + ( $('#pc').val()) + " until " + cookieExpiresDate);
			
			$('#pc').val($('#pc').val().replace(/ /,"").toUpperCase());
			var homeCook =  $('#pc').val();
			//alert ("the home cookie= " + homeCook);
			
			setCookie("pc",homeCook, cookieExpiresDate, "/");
			
			$(this).removeClass("error");
		 	} else {
		  	//alert('fail');
			$(this).addClass("error");
		 	}
	};
	// Check Postal Code format
	$("#pc").change(getVal);
	
	
	
	
	// Validate Postal Code
	function validatePC(el) {
	 if (( $('#pc').val()) != "POSTAL CODE" && ( $('#pc').val()) != "CODE POSTAL" && ( $('#pc').val()) != "") {
	//alert("attempting to validate " + el)
	
	//var pc = $(el).val();
	var pcRegExp = /(^\D{1}\d{1}\D{1}\s?\d{1}\D{1}\d{1}$)/;
	if (pcRegExp.test(el)) {
		// Save value to cookie
		//$.cookie("pc",pc, {path: '/'});
		// Attempting to fix weird bug by using the cookie algo from the gmcl.js
		// - appeared to be 2 different cookies with same name, but different HOST values, not sure why/how
		/*
		var today = new Date();
		today.setTime(today.getTime());
		var cookieExpiresDate = new Date (today.getTime() + (30 * 24 * 60));
		setCookie("pc",pc, cookieExpiresDate, "/");
		*/
		//alert ("valid!")
		return true;
		
	} else {
		if(location.href.indexOf("french")!= -1) {
			alert("Le code postal n'est pas valide. Veuillez inscrire un code postal canadien valide.");
		} else {
			alert("Postal code invalid. Please enter a valid Canadian Postal code.");
		}
		return false;
		};
	}
}
	
	//
	
	
	$("#pc").click(function(){
		//alert ("ouch!")
		getText = $("#pc").val();
		if ((getText == "POSTAL CODE") || (getText == "CODE POSTAL")){
			$("#pc").val("");
			 $("#pc").attr('maxlength',6);			 
		}
	});
	
	// Create Features Carousel
	$("#highlights").jCarouselLite({
		visible:1, 
		circular: false,
	    btnNext: "#ft_next",
	    btnPrev: "#ft_prev",
		btnGo: [".ft01", ".ft02", ".ft03", ".ft04", ".ft05"],
		tracking: true
	});	

	// On states for feature tabs (other than highlights, which is taken care of inside carousel)
	$("#feattabs li:not(.sub) a").click(function(){
		//close global nav if open
		if($("#GlobalNavTab").css("top") == "35px") toggleTab();

		$("#feattabs li:not(.sub) a").removeClass("on").parent("li").removeClass("on");
		$(this).addClass("on").parent("li").addClass("on");
		//$("#feattabs li.sub a").animate({opacity:"0.1"},400);
		$("#feattabs li.sub a").addClass("disabled");//.parent().hide();
		$("#feattabs li#first").siblings("li.sub");
		var myIndex = $("#feattabs li:not(.sub)").index($(this).parent("li"));
		$("#featContent > div").hide();
		$("#featContent > div").eq(myIndex).show();
		return false;
	})
	
	$("#feattabs li.sub a, #feattabs li#first a ").click(function(){
		//close global nav if open
		if($("#GlobalNavTab").css("top") == "35px") toggleTab();
	
		//$("#feattabs li.sub a").animate({opacity:"1"},100);
		$("#feattabs li:not(.sub)").removeClass("on").children("a").removeClass("on");
		$("#feattabs li.sub a").removeClass("disabled");//.parent().show();
		$("#feattabs li#first a").addClass("on").parent("li").addClass("on");
		$("#featContent > div").hide();
		$("#featContent > div").eq(0).show();
	})
})


/***********************************************************
 *
 * 		end of consolidated divisional pages js file
 *
 ***********************************************************/


