/*************************************************************************
    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.minX = this.maxX = x;
    this.minY = y;
    }
  else {
    this.maxX = this.trkWd - bar.offsetWidth - x; 
    this.minX = x;
    this.minY = this.maxY = 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) {
    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);
      bar.style.width = (scrollObj.wd > wn.offsetWidth) ? 
        barObj.trkWd / (scrollObj.wd / wn.offsetWidth) + "px": 
        barObj.trkWd - (2 * barObj.minX) + "px";
      barObj.maxX = barObj.trkWd - bar.offsetWidth - barObj.minX;
      }
    },
    
  // 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);
      }
    }
  }


