
/******************************************************************************\
*  geoxml.js		                               by Lance Dyas          *
*  A Google Maps API Extension  GeoXml parser                                 *
*  GeoXml Parser based on my maps kml parser by Mike Williams called egeoxml  *
*  Additions include:   GML/WFS/GeoRSS/GPX expanded GE KML style support      *                                            
\******************************************************************************/
// Modified by Rakshith for www.motorcycleroads.us Maps

// Constructor
function KMLObj(title,desc) {
	this.title = title;
  	this.description = escape(desc);
  	this.marks = [];
	this.folders = [];
	this.groundOverlays = [];
	}

function GeoXml(myvar, map, url, opts) {
  // store the parameters
  this.myvar = myvar;
  this.opts = opts || {};
  this.mb = new MessageBox(map,this,"mb",opts.messagebox);
  this.map = map;
  this.url = url;
  if (typeof url == "string") {
    this.urls = [url];
  } else {
    this.urls = url;
  }
 
  this.mb.style = this.opts.messagestyle || { backgroundColor: "silver"};

  // infowindow styles
  this.titlestyle = this.opts.titlestyle || 'style = "font-family: arial, sans-serif;font-size: medium;font-weight:bold;font-size: 100%;"';
  this.descstyle = this.opts.descstyle || 'style = "font-family: arial, sans-serif;font-size: small;padding-bottom:.7em;"';
  this.directionstyle = this.opts.directionstyle || 'style="font-family: arial, sans-serif;font-size: small;padding-left: 1px;padding-top: 1px;padding-right: 4px;"';
  // sidebar
  this.sidebarfn = this.opts.sidebarfn || GeoXml.addSidebar;
  this.dropboxfn = this.opts.dropboxfn || GeoXml.addDropdown;
  // elabel options 
  this.pointlabelopacity = this.opts.pointlabelopacity || 100;
  this.polylabelopacity = this.opts.polylabelopacity || 100;
   // other useful "global" stuff
  this.hilite  = this.opts.hilite || { color:"#aaffff",opacity: 0.3, textcolor:"#000000" };
  this.latestsidebar = "";
  this.forcefoldersopen = false;
  if(typeof this.opts.allfoldersopen !="undefined"){ this.forcefoldersopen = this.opts.allfoldersopen;}
  this.clickablepolys = true;
  this.clickablelines = true;
  if(typeof this.opts.clickablepolys !="undefined"){ this.clickablepolys = this.opts.clickablepolys;}
  if(typeof this.opts.clickablelines !="undefined"){ this.clickablelines = this.opts.clickablelines;}
  this.hideall = false;
  if(typeof proxy!="undefined")this.proxy = proxy;
  if(typeof getcapproxy !="undefined")this.proxy = getcapproxy+"url=";
  if(this.opts.hideall){ this.hideall = this.opts.hideall; }
  if(top.publishdirectory){ this.publishdirectory = top.publishdirectory; }
  else {this.publishdirectory = "http://www.microimages.com/ogc/ogcclient/";}
  if(top.standalone){this.publishdirectory = "";}
  this.kmlicon =  this.publishdirectory +"images/ge.png";
  this.docicon = this.publishdirectory +"images/ge.png";
  this.foldericon = this.publishdirectory + "images/folder.png";
  this.gmlicon = this.publishdirectory + "images/geo.gif";
  this.rssicon = this.publishdirectory + "images/rssb.png";
  this.globalicon = this.publishdirectory + "images/geo.gif"; 
  this.WMSICON = "<img src=\""+this.publishdirectory+"images/geo.gif\" style=\"border:none\" />";
  GeoXml.WMSICON = this.WMSICON;
  this.baseLayers = [];
  this.bounds = new GLatLngBounds();
  this.style = {width:2,opacity:0.75,fillopacity:0.4};
  this.style.color = this.randomColor();
  this.style.fillcolor = this.randomColor();
  this.iwwidth = this.opts.iwwidth || 250;
  this.lastmarker = {};   
  this.verySmall = 0.0000001;
  this.progress = 0;
  this.ZoomFactor = 2;
  this.NumLevels = 18;
  this.maxtitlewidth = 0;
  this.styles = []; // associative array
  this.kml = [new KMLObj("GeoXML","")];
  this.jsdocs = [];
  this.jsonmarks = [];
  this.polyset = []; /* used while rendering */
  this.polygons = []; /*stores indexes to multi-polygons */ 
  this.polylines = []; /*stores indexes to multi-line */ 
  this.multibounds = []; /*stores extents of multi elements */
  this.overlayman = new Clusterer(map, this);
  this.overlayman.rowHeight = 20;
  if(this.opts.sidebarid){ this.basesidebar = this.opts.sidebarid; }
  this.overlayman.folders.push([]);
  this.overlayman.subfolders.push([]);
  this.overlayman.folderhtml.push([]);
  this.overlayman.folderhtmlast.push(0);
  this.overlayman.folderBounds.push(new GLatLngBounds()); 
  this.wmscount = 0;
  this.labels = new GTileLayerOverlay(G_HYBRID_MAP.getTileLayers()[1]);
  }
function $(mid){ return document.getElementById(mid);}

GeoXml.prototype.clear = function(idx) {
	for(var m=0;m<this.overlayman.markers.length;m++){
		this.overlayman.RemoveMarker(this.overlayman.markers[m]);
		}
	this.kml = [new KMLObj("GeoXML","")];
 	this.maxtitlewidth = 0;
  	this.styles = []; // associative array
  	this.jsdocs = [];
  	this.jsonmarks = [];
  	this.polyset = []; /* used while rendering */
  	this.polylines = [];
  	this.multibounds = []; 
	this.bounds = new GLatLngBounds();
  	this.overlayman = new Clusterer(map, this);
  	this.overlayman.rowHeight = 20;
	$(this.basesidebar).innerHTML = "";
  	this.overlayman.folders.push([]);
  	this.overlayman.subfolders.push([]);
  	this.overlayman.folderhtml.push([]);
  	this.overlayman.folderhtmlast.push(0);
  	this.overlayman.folderBounds.push(new GLatLngBounds()); 
 	this.wmscount = 0;
	}

 
// Create Marker
GeoXml.prototype.createMarkerJSON = function(item,idx) {
	var that = this;
	var style = that.makeIcon(style, item.href);
 	var point = new GLatLng(item.y,item.x);
	that.overlayman.folderBounds[idx].extend(point);
	that.bounds.extend(point);

	if(item.shadow){ style.shadow = item.shadow; }
		else{ style.shadow = null; }
	if (!!that.opts.createmarker) {
          	that.opts.createmarker(point, item.title, unescape(item.description), null, idx, style, item.visibility);
        	} 
	else {
          	that.createMarker(point, item.title, unescape(item.description), null, idx, style, item.visibility);
        	}
	};
 
GeoXml.prototype.createMarker = function(point,name,desc,style,idx,instyle,visible) {
   var myvar=this.myvar;
   var icon;
   var bicon = new GIcon();
   bicon.infoWindowAnchor=new GPoint(16,0);
   bicon.iconSize=new GSize(32,32);
   bicon.shadowSize=new GSize(56,32);
   bicon.iconAnchor=new GPoint(16,32);
  if(this.opts.iconFromDescription){
	var text = desc;
	var pattern = /<\s*img/ig;
	var result;
	var pattern2 = /src\s*=\s*[\'\"]/;
	var pattern3 = /[\'\"]/;
    	while ((result = pattern.exec(text)) != null) {
		var stuff = text.substr(result.index);
		var result2 = pattern2.exec(stuff);
      		if (result2 != null) {
        		stuff = stuff.substr(result2.index+result2[0].length);
        		var result3 = pattern3.exec(stuff);
        		if (result3 != null) {
          			var imageUrl = stuff.substr(0,result3.index);
	 			 href = imageUrl;
          			}
       			}
     		}
	var shadow = null;
	if(!href){
		href = "http://maps.google.com/mapfiles/kml/pal3/icon40.png";
		}
	icon = new GIcon(bicon, href, null, shadow);
	}
  else {
  var href;
    if(typeof instyle =="undefined"){
  	href = "http://maps.google.com/mapfiles/kml/pal3/icon40";
	shadow = href + "s.png";
        icon = new GIcon(bicon, href+".png", null, shadow );
	href += ".png";
	}
   else { 
	href = instyle.href;
	shadow = instyle.shadow
	icon = new GIcon(bicon, instyle.href, null, shadow);
	}
  }
  var iwoptions = this.opts.iwoptions || {};
  var markeroptions = this.opts.markeroptions || {};
  var icontype = this.opts.icontype || "style";
  if (icontype == "style") {
    if (!!this.styles[style]) {
      icon = this.styles[style];
      href = icon.href;
    }
  }
  if (!markeroptions.icon) {
    markeroptions.icon = icon;
  }
  markeroptions.title = name;
  var m = new GMarker(point, markeroptions);
  m.title = name;
  var obj = {"type":"point","title":name,"description":escape(desc),"href":href,"shadow": shadow,"visibility":visible,"x":point.x,"y": point.y};
  this.kml[idx].marks.push(obj);

  if (this.opts.pointlabelclass) {
    var l = new ELabel(point, name, this.opts.pointlabelclass, this.opts.pointlabeloffset, this.pointlabelopacity, true);
    this.map.addOverlay(l);
  }

  var html = "<div style = 'width:"+this.iwwidth+"px'>"
               + "<h1 "+this.titlestyle+">"+name+"</h1>"
               +"<div "+this.descstyle+">"+desc+"</div>";

  if (this.opts.directions) {
    var html1 = html + '<div '+this.directionstyle+'>'
                     + 'Get Directions: <a href="#" onclick="GEvent.trigger(' + this.myvar +'.lastmarker,\'click2\');return false;">To Here</a> - ' 
                     + '<a href="#" onclick="GEvent.trigger(' + this.myvar +'.lastmarker,\'click3\');return false;">From Here</a><br>'
                     + '<a href="#" onclick="GEvent.trigger(' + this.myvar +'.lastmarker,\'click4\');return false;">Search nearby</a></div>';
    var html2 = html + '<div '+this.directionstyle+'>'
                     + 'Get Directions: To here - '
                     + '<a href="#" onclick="GEvent.trigger(' + this.myvar +'.lastmarker,\'click3\');return false;">From Here</a><br>'
                     + 'Start address:<form action="http://maps.google.com/maps" method="get" target="_blank">'
                     + '<input type="text" SIZE=35 MAXLENGTH=80 name="saddr" id="saddr" value="" />'
                     + '<INPUT value="Go" TYPE="SUBMIT">'
                     + '<input type="hidden" name="daddr" value="' + point.lat() + ',' + point.lng() + "(" + name + ")" + '"/>'
                     + '<br><a href="# onclick="GEvent.trigger(' + this.myvar +'.lastmarker,\'click\');return false;">&#171; Back</a></div>';
    var html3 = html + '<div '+this.directionstyle+'>'
                     + 'Get Directions: <a href="#" onclick="GEvent.trigger(' + this.myvar +'.lastmarker,\'click2\');return false;">To Here</a> - ' 
                     + 'From Here<br>'
                     + 'End address:<form action="http://maps.google.com/maps" method="get"" target="_blank">'
                     + '<input type="text" SIZE=35 MAXLENGTH=80 name="daddr" id="daddr" value="" />'
                     + '<INPUT value="Go" TYPE="SUBMIT">'
                     + '<input type="hidden" name="saddr" value="' + point.lat() + ',' + point.lng() +  "(" + name + ")" + '"/>'
                     + '<br><a href="#" onclick="GEvent.trigger(' + this.myvar +'.lastmarker,\'click\');return false;">&#171; Back</a></div>';
    var html4 = html + '<div '+this.directionstyle+'>'
                     + 'Search nearby: e.g. "pizza"<br>'
                     + '<form action="http://maps.google.com/maps" method="get"" target="_blank">'
                     + '<input type="text" SIZE=35 MAXLENGTH=80 name="q" id="q" value="" />'
                     + '<INPUT value="Go" TYPE="SUBMIT">'
                     + '<input type="hidden" name="near" value="' + name + ' @' + point.lat() + ',' + point.lng() + '"/>'
                   //  + '<input type="hidden" name="near" value="' +  point.lat() + ',' + point.lng() +  "(" + name + ")" + '"/>';
                     + '<br><a href="# onclick="GEvent.trigger(' + this.myvar +'.lastmarker,\'click\');return false;">&#171; Back</a></div>';
    GEvent.addListener(m, "click2", function() {
      m.openInfoWindowHtml(html2 + "</div>",iwoptions);
    });
    GEvent.addListener(m, "click3", function() {
      m.openInfoWindowHtml(html3 + "</div>",iwoptions);
    });
    GEvent.addListener(m, "click4", function() {
      m.openInfoWindowHtml(html4 + "</div>",iwoptions);
    });
  } else {
    var html1 = html;
  }
  

  GEvent.addListener(m, "click", function() {
    eval(myvar+".lastmarker = m");
    m.openInfoWindowHtml(html1 + "</div>",iwoptions);
  });

  if(this.opts.domouseover){
	m.mess = html1+"</div>";
	m.geoxml = this;
  	GEvent.addListener(m,"mouseover", function(point) {if(!point){ point=m.getPoint(); } m.geoxml.mb.showMess(m.mess,5000); } );
	}
  var nhtml = "";
  var parm;
  if (this.opts.sidebarid) {
    	var folderid = this.myvar+"_folder"+idx;    
    	var n = this.overlayman.markers.length; 
	var blob = "&nbsp;<img style=\"vertical-align:text-top;padding:0;margin:0\" height=\"16\" border=\"0\" src=\""+href+"\">&nbsp;";
 	parm =  this.myvar+"$$$" +name + "$$$marker$$$" + n +"$$$" + blob + "$$$" +visible+"$$$null"; 
		
	m.sidebarid = this.myvar+"sb"+n;
	m.hilite = this.hilite;
	m.geoxml = this;
//	GEvent.addListener(m,"mouseover", function() {
//		$(this.sidebarid).style.backgroundColor = this.hilite.color;
//		$(this.sidebarid).style.color = this.hilite.textcolor;
//		} 
//		);
//	GEvent.addListener(m,"mouseout", function() {
//			$(this.sidebarid).style.background = "none"; 
//			$(this.sidebarid).style.color = "";
//			});

  	} 

  if (!!this.opts.addmarker) {
    this.opts.addmarker(m, name,idx, parm, visible);
  } else {
     this.overlayman.AddMarker(m, name,idx, parm, visible);
  }

};

// Create Polyline



GeoXml.prototype.processLine = function (pnum, lnum, idx){
	var op = this.polylines[pnum];
	var line = op.lines[lnum];
	if(!line)return;
        var thismap = this.map;
	var iwoptions = this.opts.iwoptions || {};
	var polylineEncoder = new PolylineEncoder(this.NumLevels,this.ZoomFactor,this.verySmall,true);
	if(line.length >2) {
		var result = polylineEncoder.dpEncode(line);
		obj = { points: result.encodedPoints,
			levels: result.encodedLevels,
			color: op.color,
			weight: op.width,
			opacity: op.opacity,
			zoomFactor: this.ZoomFactor,
			numLevels: this.NumLevels,
			type: "polyline" 
	 		};
		var p = new GPolyline.fromEncoded(obj);	
		}
	else {
 		obj = { points:line, color:op.color, weight:op.width, opacity:op.opacity, type:"line" };
		var p = new GPolyline(line,op.color,op.width,op.opacity);
		}
	p.bounds = op.pbounds;
	var nhtml = "";
	var n = this.overlayman.markers.length;
	this.polylines[pnum].lineidx.push(n);
	var parm;
	var html = "<div style='font-weight: bold; font-size: medium; margin-bottom: 0em;'>"+op.name;
  	html += "</div>"+"<div style='font-family: Arial, sans-serif;font-size: small;width:"+this.iwwidth+"px'>"+op.description+"</div>";

	if(lnum ==0){
	 	if(this.opts.sidebarid) {
    			var blob = '&nbsp;&nbsp;<span style=";border-left:'+op.width+'px solid '+op.color+';">&nbsp;</span> ';
			parm =  this.myvar+"$$$" +op.name + "$$$polyline$$$" + n +"$$$" + blob + "$$$" +op.visible+"$$$"+pnum+"$$$";
			this.latestsidebar = this.myvar +"sb"+n;
 			}
		}

	if(lnum < line.length){
		setTimeout(this.myvar+".processLine("+pnum+","+(lnum+1)+",'"+idx+"');",15);
		if(this.opts.sidebarid) { p.sidebar = this.latestsidebar; }
		}
		
	if(this.opts.domouseover){
		p.mess = html;
		}
  	p.title = op.name;
        p.geoxml = this;
        p.strokeColor = op.color;
        p.strokeWeight = op.width;
	p.strokeOpacity = op.opacity;
	p.hilite = this.hilite;
	p.mytitle = p.title;
	p.map = this.map;
	p.idx = pnum;
  	p.onOver = function(){
		var pline = this.geoxml.polylines[this.idx];
		for(var l=0;l<pline.lineidx.length;l++){
			var mark = this.geoxml.overlayman.markers[pline.lineidx[l]];
			mark.realColor = mark.strokeColor; 
			mark.setStrokeColor(this.hilite.color);
			}
		$(this.sidebar).style.backgroundColor = this.hilite.color;
		$(this.sidebar).style.color = this.hilite.textcolor;
		if(this.mess) { this.geoxml.mb.showMess(this.mess,5000); } else { this.title = "Click for more information about "+this.mytitle; }
		};
  	p.onOut = function(){ 
		var pline = this.geoxml.polylines[this.idx];
		for(var l=0; l < pline.lineidx.length; l++){
			var mark = this.geoxml.overlayman.markers[pline.lineidx[l]];
			mark.setStrokeColor(mark.realColor);
			}
		this.geoxml.mb.hideMess();
		$(this.sidebar).style.background = "none";
		$(this.sidebar).style.color = "";
		};

  	GEvent.addListener(p,"click", function(point) {var doit=false; if(!point) { point = this.getPoint();doit=true; } if(this.geoxml.clickablelines|doit)this.map.openInfoWindowHtml(point, html, iwoptions); });
	obj.name = op.name;
        obj.description = escape(op.description);
	obj.visibility = op.visibility;
 	this.kml[idx].marks.push(obj);
 	this.overlayman.AddMarker(p, op.name, idx, parm, op.visibility);
};

GeoXml.prototype.createPolyline = function(lines,color,width,opacity,pbounds,name,desc,idx, visible) {
    	var p = {};
   	if(!color){p.color = this.randomColor();}
  	else { p.color = color; }
  	if(!opacity){p.opacity= 0.45;}
		else p.opacity = opacity;
  	if(!width){p.width = 2;}
 		 else{  p.width = width; }
  	p.idx = idx;
	p.visibility = visible;
	if(this.hideall)p.visibility = false;
	p.name = name;
	p.description = desc;
 	p.lines = lines;
        p.lineidx = [];
 	this.polylines.push(p);
	setTimeout(this.myvar+".processLine("+(this.polylines.length-1)+",0,'"+idx+"');",15);
	};

// Create Polygon

GeoXml.prototype.processPLine = function(pnum,linenum,idx) {
        
	var p = this.polyset[pnum];
	var line = p.lines[linenum];
	var obj = {};
	
	if(line && line.length){
		var polylineEncoder = new PolylineEncoder(18,2,0.00001,true);
		var result = polylineEncoder.dpEncode(line);
		obj = {};
		obj.points = result.encodedPoints;
		obj.levels = result.encodedLevels;
		obj.color = p.color;
		obj.weight = p.weight;
		obj.numLevels = 18;
		obj.zoomFactor = 2;
		obj.opacity = p.opacity;
		p.obj.polylines.push(obj);
		}
	if(linenum == p.lines.length-1){	
		this.finishPolygon(p.obj,idx);
		}
	else {
	    setTimeout(this.myvar+".processPLine("+pnum+","+(linenum+1)+",'"+idx+"');",5);
	    }
	};	

GeoXml.prototype.finishPolygon = function(op,idx) {
  op.type = "polygon";
  this.finishPolygonJSON(op,idx,false);
   };

GeoXml.prototype.finishPolygonJSON = function(op,idx,updatebound,lastpoly) {
  var that = this;
  var iwoptions = that.opts.iwoptions || {};
  if(typeof op.visibility == "undefined")op.visibility=true;
  if(that.hideall)op.visibility = false;
  var desc = unescape(op.description);
  var p = new GPolygon.fromEncoded(op);
  var html = "<div style='font-weight: bold; font-size: medium; margin-bottom: 0em;'>"+op.name+"</div>"
             +"<div style='font-family: Arial, sans-serif;font-size: small;width:"+this.iwwidth+"px'>"+desc+"</div>";
 var newgeom = (lastpoly != "p_"+op.name);
  if(newgeom){
	this.latestsidebar = that.myvar +"sb"+  this.overlayman.markers.length;
	}
  if(that.opts.domouseover){
  	p.mess = html;
	}
  p.strokeColor = op.polylines[0].color;
  p.mb = that.mb;
  p.hilite = that.hilite;
  p.strokeOpacity = op.polylines[0].opacity;
  p.fillopacity = op.opacity;
  if(that.domouseover){
	p.mess = html;
	}
  p.geoxml = that;
  p.title = op.name;
 var n = this.overlayman.markers.length;
  if(newgeom){
	that.multibounds.push(new GLatLngBounds());
 	that.polygons.push([])
	}
  var len = that.multibounds.length-1;
  that.multibounds[len].extend(p.getBounds().getSouthWest());
  that.multibounds[len].extend(p.getBounds().getNorthEast()); 
  that.polygons[that.polygons.length-1].push(n);
  p.polyindex = that.polygons.length-1;
  p.geomindex = len;
  p.sidebarid = this.latestsidebar;
  p.onOver = function(){ 
		$(this.sidebarid).style.backgroundColor = this.hilite.color;
		$(this.sidebarid).style.color = this.hilite.textcolor;
	//	if(this.geoxml.clickablepolys){
		var poly = this.geoxml.polygons[this.polyindex];
		for (var pg =0;pg < poly.length;pg++) {
			var mark = this.geoxml.overlayman.markers[poly[pg]];
			mark.realColor= mark.fillColor;
			mark.setFillColor(this.hilite.color);
			}
	
	//		}

		if(this.mess)p.geoxml.mb.showMess(this.mess,5000); 
		};
  p.onOut = function(){ 
		$(this.sidebarid).style.background= "none";
		$(this.sidebarid).style.color = "";
	//	if(this.geoxml.clickablepolys) 
		var poly = this.geoxml.polygons[this.polyindex];
		for (var pg =0;pg < poly.length;pg++) {
			var mark = this.geoxml.overlayman.markers[poly[pg]];
			mark.setFillColor(this.realColor);
			}
	//	}
		};
  op.description = escape(desc);
  this.kml[idx].marks.push(op);
  p.map = this.map;
 
  GEvent.addListener(p,"click", function(point, overlay) {
	if(!point){ 
		this.geoxml.mb.showMess("Zooming to "+p.title,3000);
		var bounds = this.geoxml.multibounds[this.geomindex];  
		this.map.setZoom(this.map.getBoundsZoomLevel(bounds));
    		this.map.panTo(bounds.getCenter());
		}
	else { 
	 	if(this.geoxml.clickablepolys)this.map.openInfoWindowHtml(point,html,iwoptions);} 
		}
	);

if(this.opts.polylabelclass && newgeom ) {
 	var epoint =  p.getBounds().getCenter();
        var off = this.opts.polylabeloffset;
	if(!off)off= new GSize(0,0);
	off.x = -(op.name.length * 6);
 	var l = new ELabel(epoint, " "+op.name+" ", this.opts.polylabelclass, off, this.polylabelopacity, true);
	this.map.addOverlay(l);
	}

  var nhtml ="";
  var parm;
 
  if (this.basesidebar &&  newgeom) {
   
    var folderid = this.myvar+"_folder"+idx;
    var blob = '<span style="background-color:' + op.color + ';border:2px solid '+op.fillcolor+';">&nbsp;&nbsp;&nbsp;&nbsp;</span> ';
    parm =  this.myvar+"$$$" +op.name + "$$$polygon$$$" + n +"$$$" + blob + "$$$" +op.visibility+"$$$null"; 
    }
   if(updatebound) {
  	var ne = p.getBounds().getNorthEast();
   	var sw = p.getBounds().getSouthWest()
   	this.bounds.extend(ne);
   	this.bounds.extend(sw);
   	this.overlayman.folderBounds[idx].extend(sw);
   	this.overlayman.folderBounds[idx].extend(ne);
	}
   this.overlayman.AddMarker(p,op.name,idx, parm, op.visibility);
   return op.name;
   };

GeoXml.prototype.finishLineJSON = function(po, idx, lastlinename){
	var m;
	var that = this;
	var thismap = this.map;
	if(po.type == "line"){ m = new GPolyline(po.points,po.color,po.weight,po.opacity); }
	else { m = new GPolyline.fromEncoded(po); }
	m.mytitle = po.name;
	m.title = po.name;
        m.geoxml = this;
        m.strokeColor = po.color;
        m.strokeWeight = po.weight;
	m.strokeOpacity = po.opacity;
        m.hilite = this.hilite;
	var n = that.overlayman.markers.length;
	var lineisnew = false;
	if(("l_"+po.name) != lastlinename){
		lineisnew = true;
		that.polylines.push(po);
		var pnum = that.polylines.length-1;
		that.polylines[pnum].lineidx = [];
		that.latestsidebar = that.myvar +"sb"+n;
		}
	else {
		var pnum = that.polylines.length-1;
		that.polylines[pnum].lineidx.push(n);
		}
	m.sidebarid = that.latestsidebar;

  	m.onOver = function(){
			$(this.sidebarid).style.backgroundColor = this.hilite.color;
			this.realColor = this.strokeColor; this.setStrokeColor(this.hilite.color);
			 if(this.mess) { this.geoxml.mb.showMess(this.mess,5000); } else { this.title = "Click for more information about "+this.mytitle; }
			};
  	m.onOut = function(){ 	
			$(this.sidebarid).style.background = "none";
			this.setStrokeColor(this.realColor);
			if(this.mess)this.geoxml.mb.hideMess();
			};

	var parm = "";
	that.kml[idx].marks.push(po);
	var desc = unescape(po.description);
	var html = "<div style='font-weight: bold; font-size: medium; margin-bottom: 0em;'>"+po.name;
  	html += "</div>"+"<div style='font-family: Arial, sans-serif;font-size: small;width:"+that.iwwidth+"px'>"+desc+"</div>";
	m.map = this.map;
	if(this.clickablelines){
  		GEvent.addListener(m,"click", function(point) {if(!point){ point=m.getPoint(); } this.map.openInfoWindowHtml(point,html,that.opts.iwoptions);} );
		}

	if(that.basesidebar && lineisnew) {
    		var blob = '&nbsp;&nbsp;<span style=";border-left:'+po.weight+'px solid '+po.color+';">&nbsp;</span> ';
		if(typeof po.visibility == "undefined")po.visibility = true;
		parm =  that.myvar+"$$$" +po.name + "$$$polyline$$$" + n +"$$$" + blob + "$$$" +po.visibility+"$$$"+(that.polylines.length-1)+"$$$";
 		}	
	
	var ne = m.getBounds().getNorthEast();
	var sw = m.getBounds().getSouthWest()
	that.bounds.extend(ne);
	that.bounds.extend(sw);
	that.overlayman.folderBounds[idx].extend(sw);
	that.overlayman.folderBounds[idx].extend(ne);

    	that.overlayman.AddMarker(m, po.name, idx, parm, po.visibility);	
	return(po.name);	
	};
	
GeoXml.prototype.handlePlaceObj = function(num, max, idx, lastlinename, depth){
	var that = this;
	var po = that.jsonmarks[num];
	var lastlinename;
	var name = po.name;
	if(po.title)name = po.title;
	if(name.length+depth > that.maxtitlewidth)that.maxtitlewidth = name.length+depth;
	switch (po.type) {
			case "polygon" :
				lastlinename = "p_"+ that.finishPolygonJSON(po,idx,true,lastlinename);
				break;
			case "line" :  
			case "polyline" :
				lastlinename = "l_"+ that.finishLineJSON(po,idx,lastlinename);		
				break;
			case "point":
          			that.createMarkerJSON(po,idx);
				lastlinename = "";
				break;
		 	}
	if (num < max-1){
		setTimeout(that.myvar+".handlePlaceObj("+(num+1)+","+max+","+idx+",'"+lastlinename+"',"+depth+");",1);
		}
	else {
		lastlinename = "";		
		if(num == that.jsonmarks.length-1){
			that.progress--;
    			if (that.progress <= 0) {
      		 	// Shall we zoom to the bounds?
      				if (!that.opts.nozoom) {
        				that.map.setZoom(that.map.getBoundsZoomLevel(that.bounds));
        				that.map.setCenter(that.bounds.getCenter());
      					}
      				GEvent.trigger(that,"parsed");
      				if(!that.opts.sidebarid){
					that.mb.showMess("Finished Parsing",1000);
					}
				}
	 		}
		}
	};

GeoXml.prototype.parseJSON  = function (doc, title, latlon, desc, sbid){
	var that = this;
 	that.overlayman.miStart = new Date();
	that.jsdocs = eval('(' + doc + ')');
	var bar = $(that.basesidebar);
	if(bar)bar.style.display="";
	that.recurseJSON(that.jsdocs[0], title, desc, that.basesidebar, 0);
	};

GeoXml.prototype.recurseJSON = function (doc, title, desc, sbid, depth){
	var that = this;
	var polys = doc.marks;
	var name = doc.title;
	if(!sbid)sbid = 0;
	var description = unescape(doc.description);
	if(!description && desc)description = desc;
	var open = that.forcefoldersopen;
	if(doc.open) open = true;
	var visible = true;
	if(typeof doc.visibility!="undefined" && doc.visibility == false)visible = false;
       	var snippet = doc.snippet;
	var idx = that.overlayman.folders.length;
	if(!description)description = name;
	var folderid;
	var idx = this.overlayman.folders.length;
	var icon;
	that.overlayman.folders.push([]);
	that.overlayman.subfolders.push([]);
    	that.overlayman.folderhtml.push([]);
    	that.overlayman.folderhtmlast.push(0);
	that.overlayman.folderBounds.push(new GLatLngBounds());
	that.kml.push(new KMLObj(title,description));
	if((depth != 0 && doc.folders.length >1) || doc.marks.length){
		if(depth < 2 || doc.marks.length < 1) { icon = that.globalicon; }
		else { icon = that.foldericon;}
		folderid = that.createFolder(idx, name, sbid, icon, description, snippet, open, visible);
		} 
	else {
		folderid = sbid;
		}
	var parm, blob;
	var nhtml ="";
	var html;
	var m;
	num = that.jsonmarks.length;
	max = num + polys.length;
 	for(var p =0;p<polys.length;p++){
		var po = polys[p];
		that.jsonmarks.push(po);
		var desc = unescape(po.description);
		var m;
 		if(desc && desc.match(/<(\s)*img/i)){
			var preload = document.createElement("span");
     			preload.style.visibility = "visible";
			preload.style.position = "absolute";
			preload.style.left = "-1200px";
			preload.style.top = "-1200px";
			preload.style.zIndex = this.overlayman.markers.length; 
     			document.body.appendChild(preload);
			preload.innerHTML = desc;
			}	 
		}	

	if(that.groundOverlays){
		


		}

	if(polys.length)that.handlePlaceObj(num,max,idx,null,depth);
	var fc = 0;
	var fid = 0;
	if(typeof doc.folders!="undefined"){
		fc = doc.folders.lenth;
		for(var f=0;f<doc.folders.length;++f){
			var nextdoc = that.jsdocs[doc.folders[f]];
			fid = that.recurseJSON(nextdoc, nextdoc.title, nextdoc.description, folderid, (depth+1));
			that.overlayman.subfolders[idx].push(fid);
			that.overlayman.folderBounds[idx].extend(that.overlayman.folderBounds[fid].getSouthWest());
			that.overlayman.folderBounds[idx].extend(that.overlayman.folderBounds[fid].getNorthEast());
			if(fid != idx){ that.kml[idx].folders.push(fid); }
			}
		}

        if(fc || polys.length ){
		that.bounds.extend(that.overlayman.folderBounds[idx].getSouthWest());
		that.bounds.extend(that.overlayman.folderBounds[idx].getNorthEast());
		}

	return idx;
	};

GeoXml.prototype.createPolygon = function(lines,color,width,opacity,fillcolor,fillopacity, pbounds, name, desc, folderid, visible,fill,outline) {
  var thismap = this.map;
  var p = {};	
  p.obj = {"description":desc,"name":name };
  p.obj.polylines = []; 
  p.obj.visibility = visible;
  p.obj.fill = fill;
  p.obj.outline = outline; 
  p.fillcolor = fillcolor;
  p.obj.strokecolor = color; 
  if(!color){p.color = this.style.color;}
  else { p.color = color; }

  if(!fillcolor){ p.obj.color = this.randomColor(); }
  else {p.obj.color = fillcolor;}

  if(!opacity){p.obj.opacity= this.style.opacity;}
	else{ p.obj.opacity = opacity; }

  if(!fillopacity){p.obj.fillopacity = this.style.fillopacity}
   else { p.obj.fillopacity = fillopacity;}

  if(!width){p.weight = this.style.width;}
  else{  p.weight = width; }

  p.bounds = pbounds;
  p.lines = lines;
  p.sidebarid = this.opts.sidebarid;
  this.polyset.push(p);
  setTimeout(this.myvar+".processPLine("+(this.polyset.length-1)+",0,'"+folderid+"')",5);
};


GeoXml.prototype.toggleFolder = function(i){
	var f = $(this.myvar+"_folder"+i);
	var tb = $(this.myvar+"TB"+i);
	if(f.style.display=="none"){
			f.style.display="";
			if(tb)tb.style.fontWeight = "normal";
			}
		else{ 
			f.style.display ="none"; 
			if(tb)tb.style.fontWeight = "bold";
			}
	}

GeoXml.prototype.saveJSON = function(){

	if(top.standalone){
		var fpath = browseForSave("Select a directory to place your json file","JSON Data Files (*.js)|*.js|All Files (*.*)|*.*","JSON-DATA");

 		if(typeof fpath!="undefined")saveLocalFile (fpath+".js", this.kml.toJSONString());
		return;
		}

	if(this.kml.toJSONString){
		if(typeof serverBlessJSON!="undefined"){
			serverBlessJSON(escape(this.kml.toJSONString()),"MyKJSON"); 
			}
		else {
			showIt(this.kml.toJSONString());
			}
		}
	else {
		alert("No JSON methods currently available");
		}
	}

function showIt (str, h, w) {
	var features = "status=yes,resizable=yes,toolbar=0,height=" + h + ",width=" + h + ",scrollbars=yes";
	var myWin;
 
	if (top.widget) {
		alert(str);
		showMess(str);
		}
	else {
		myWin = window.open("", "_blank", features);
		myWin.document.open("text/xml");
		myWin.document.write(str);
		myWin.document.close();
		}
	}

GeoXml.prototype.toggleContents = function(i,show){
 	var f = this.overlayman.folders[i];
	var cb;
	if(show){
	for (var j=0;j<f.length;j++){
			var m = this.overlayman.markers[f[j]];
		        m.hidden = false;	
			cb = $(this.myvar+''+f[j]+'CB');
			if(cb)cb.checked = true;
			if(true) { m.show(); }
				else { this.map.addOverlay(m); }
			}
		}
	else {
	   for (var j=0;j<f.length;j++){
			var m = this.overlayman.markers[f[j]];
			m.hidden = true;
			cb = $(this.myvar+''+f[j]+'CB');
			if(cb)cb.checked = false;
			if(true) { m.hide(); }
				else { this.map.removeOverlay(m); }
			}
		}

	var sf = this.overlayman.subfolders[i];
	if(typeof sf!="undefined"){
 		for (var j=0;j<sf.length;j++){
			if(sf[j]!=i){
	 			cb = $(this.myvar+''+sf[j]+'FCB');
				if(cb)cb.checked = (!!show);
				this.toggleContents(sf[j],show);
				}
			}
		 }
	};
GeoXml.prototype.showHide = function(a,show, p){
 	if(a !== null){
		
		if(show){ 
			this.overlayman.markers[a].show(); 
			this.overlayman.markers[a].hidden = false; }	
		else  { this.overlayman.markers[a].hide(); 
			this.overlayman.markers[a].hidden = true; }
		}
	else {
		var ms = this.polylines[p];
		if(show){
			for(var i=0;i<ms.lineidx.length;i++){
				var m = this.overlayman.markers[ms.lineidx[i]];
				m.hidden = false;
				if(true) { m.show(); }
				else { this.map.addOverlay(m); }
				}
		    }
		else {
			for(var i=0;i<ms.lineidx.length;i++){
				var m = this.overlayman.markers[ms.lineidx[i]]; 
				m.hidden = true;	
				if(true) { m.hide(); }
					else { this.map.removeOverlay(m); }
				}
		    }
	    }
	};

// Sidebar factory method One - adds an entry to the sidebar
GeoXml.addSidebar = function(myvar, name, type, e, graphic, ckd, i) {
   
   var check = "checked";
   if(ckd=="false")check = "";
    var h="";
    var mid = myvar+'sb'+e;
   switch(type) {
   case  "marker" :  h = '<nobr><a href="#" onclick="GEvent.trigger(' + myvar+ '.overlayman.markers['+e+'],\'click\');return false;">' + name + '</a></nobr><br>';
   break;
  case  "polyline" :  h = '<nobr><span style="margin-top:6px;"><a href="#" style="text-decoration: none;font-size: 8pt" onclick="GEvent.trigger(' + myvar+ '.overlayman.markers['+e+'],\'click\');return false;">&nbsp;' + graphic + name + '</a></span></nobr><br>';
  break;
  case "polygon": h = '<li id="'+mid+'"  onmouseout="'+myvar+ '.overlayman.markers['+e+'].onOut();" onmouseover="'+myvar+ '.overlayman.markers['+e+'].onOver();" ><input id="'+myvar+''+e+'CB" type="checkbox" '+check+' onclick="'+myvar+'.showHide('+e+',this.checked)"><span style="margin-top:6px;"><a href="#" onclick="GEvent.trigger(' + myvar+ '.overlayman.markers['+e+'],\'click\');return false;">&nbsp;' + graphic + name + '</a></span></nobr></li>';
  break;
 case "groundoverlay": h = '<li id="'+mid+'"><input id="'+myvar+''+e+'CB" type="checkbox" '+check+' onclick="'+myvar+'.showHide('+e+',this.checked)"><span style="margin-top:6px;"><a href="#" onclick="GEvent.trigger(' + myvar+ '.overlayman.markers['+e+'],\'zoomto\');return false;">&nbsp;' + graphic + name + '</a></span></li>';
   break;
case "tiledoverlay": h = '<li id="'+mid+'"><nobr><input id="'+myvar+''+e+'CB" type="checkbox" '+check+' onclick="'+myvar+'.showHide('+e+',this.checked)"><span style="margin-top:6px;"><a href="#" oncontextMenu="'+myvar+'.upgradeLayer('+i+');return false;" onclick="GEvent.trigger(' + myvar+ '.overlayman.markers['+e+'],\'zoomto\');return false;">'+GeoXml.WMSICON +'&nbsp;'+ name + '</a><br />'+ graphic +'</span></li>';
   break;
}
return h;
};

// Dropdown factory method
GeoXml.addDropdown = function(myvar,name,type,i,graphic) {
    return '<option value="' + i + '">' + name +'</option>';
};

// Request to Parse an XML file

GeoXml.prototype.parse = function(titles) {
 var that = this;
 var names =[];
 if(typeof titles !="undefined"){
 if(typeof titles!= "string") {
 	names = titles;
	}
 else {
	names = titles.split(",");
	}
}
 that.progress += that.urls.length;
 for (u=0; u<that.urls.length; u++) {
   var title = names[u];
  if(typeof title =="undefined" || !title || title =="null" ){
  	var segs = that.urls[u].split("/");
	title = segs[segs.length-1];
	}
   that.mb.showMess("Loading "+title);
   var re = /\.js$/i;
   if(that.urls[u].search(re) != -1){
	that.loadJSONUrl(this.urls[u], title);
	}
   else {
 	that.loadXMLUrl(this.urls[u], title);	}
 }
};

GeoXml.prototype.parseString = function(doc,titles,latlon) {
  var names =[];
 if(titles) {
 	names = titles.split(",");
	}
  if (typeof doc == "string") {
    this.docs = [doc];
  } else {
    this.docs = doc;
  }
  this.progress += this.docs.length;
  for (u=0; u<this.docs.length; u++) {
    this.mb.showMess("Processing "+names[u]);
    this.processing(GXml.parse(this.docs[u]),names[u],latlon);
  }
};

GeoXml.prototype.parseXML = function(doc,titles,latlon) {
 var names =[];
 if(typeof titles !="undefined"){
 	if(typeof titles == "string") {
 		names = titles.split(",");
		}
	 else names = titles;
	}

  if(typeof doc == "array"){
	this.docs = doc;
	}
  else {
 	this.docs = [doc];
	}
  this.progress += this.docs.length;
  for (u=0; u<this.docs.length; u++) {
    this.mb.showMess("Processing "+names[u]);
    this.processing(this.docs[u],names[u],latlon);
  }
};
var currdeschead = "";
GeoXml.prototype.makeDescription = function(elem, title, depth) {
         var d = ""; 
	 var len = elem.childNodes.length;
	 var ln = 0;
	 while (len--) {
		var subelem = elem.childNodes.item(ln);
		var nn = subelem.nodeName;
		var sec = nn.split(":");
		var base = "";
		if(sec.length>1){ 
			base = sec[1];
			}
		else { base = nn;}
 	
		if(base.match(/^(boundedBy|StyleMap|styleUrl|visibility|posList|coordinates|Style|Polygon|LineString|Point|LookAt|Envelope|Box|MultiPolygon)/)){
 			currdeschead = ""; 
			}
		else {
		
			if(base.match(/#text|the_geom/)){}
			else {	
				currdeschead = "<b>"+base+"</b> :";
				}
			val = subelem.nodeValue;
			if(base.match(/(\S)*(name|title)(\S)*/i)){
			 	if(!val)val = GXml.value(subelem);
				title = val;
				if(val && typeof title!="undefined" && title.length > this.maxtitlewidth){
					this.maxtitlewidth = title.length;
					}
				currdeschead="";
				}
			else {
				 if(val && val.match(/(\S)+/)){		
					if (val.match(/^http:\/\/|^https:\/\//i)) {
        	    				val = '<a href="' + val + '">' + val + '</a>';
      		    				}
					else {
						if(!title || title==""){
							title = val;	
							if(val && typeof title!="undefined" && title.length > this.maxtitlewidth){
								this.maxtitlewidth = title.length;
								}
							}
						}
				
					}
			   if(val) { d += currdeschead + "<span>"+val+"</span><br />"; currdeschead = ""}
			
				if(subelem.childNodes.length){
		 			var con = this.makeDescription(subelem, title);
					if(con){
						d += con.desc;
						if(typeof con.title!="undefined" && con.title){
						 	title = con.title;
							if(title.length > this.maxtitlewidth){
								this.maxtitlewidth = title.length + depth;
								}
							}
						}
					}
				}

			}
		
		ln++;
		}
	var dc = {};
	dc.desc = d;
	dc.title = title;
	return dc;
	};

GeoXml.prototype.randomColor = function(){ 
	var hex="0123456789abcdef";
	var color="#";
	for (i=0;i<6;i++){
		var idx = parseInt(Math.random()*16);
		color += hex.substring(idx,idx+1);
		}
	return color;
	};

GeoXml.prototype.handleGeomark = function (mark, idx, trans) {
     var that = this;
     var desc, title, name, style;
     title = "";
     desc = "";
     var styleid = 0;
     var lat, lon;
     var visible = true;
     var fill = true;
	var outline = true;

        var coordset=mark.getElementsByTagName("coordinates");
	if(coordset.length <1){
	    coordset=mark.getElementsByTagName("gml:coordinates");
	    }
	if(coordset.length <1){
	   	coordset = [];
	    	var poslist =mark.getElementsByTagName("gml:posList");
		if(poslist.length <1) poslist = mark.getElementsByTagName("posList");
		for(var l =0;l<poslist.length;l++){
			var coords = " ";
			var cor = GXml.value(poslist.item(l)).split(' ');
			for(var cc=0;cc<(cor.length-1);cc++){
					if(cor[cc] && cor[cc]!=" " && !isNaN(parseFloat(cor[cc]))){
						coords += ""+parseFloat(cor[cc])+","+parseFloat(cor[cc+1]);
						coords += " ";
						}
					cc++;
					}
			if(coords){
 				if(poslist.item(l).parentNode && (poslist.item(l).parentNode.nodeName == "gml:LineString") )line_count++;
					else poly_count++;
				var cm = "<coordinates>"+coords+"</coordinates>";
				var node = GXml.parse(cm);
				if(coordset.push)coordset.push(node); 
				}
			}

		var pos = mark.getElementsByTagName("gml:pos");
		if(pos.length <1)pos = mark.getElementsByTagName("gml:pos");
		if(pos.length){
			for(var p=0;p<pos.length;p++){
				var nv = GXml.value(pos.item(p));
				var cor = nv.split(' ');
				var node = GXml.parse("<coordinates>"+cor[0]+","+cor[1]+"</coordinates>");
				if(coordset.push)coordset.push(node); 
				}
			}
	    }

	var newcoords = false;
	var coords = "";
	var point_count =0;
	var box_count=0;
	var line_count=0;
	var poly_count=0;

     
	var dc = that.makeDescription(mark,"");
//	desc = "<ul>"+dc.desc+"</ul>";
	if(!name && dc.title){
		name = dc.title;
		if(name.length > this.maxtitlewidth){
			this.maxtitlewidth = name.length;
			}
		}
	     
    
     if(newcoords && typeof lat!="undefined"){
		coordset.push(""+lon+","+lat);
		}
    
     var lines = [];
     for(var c=0;c<coordset.length;c++){
      var skiprender =false;
     if(coordset[c].parentNode && (coordset[c].parentNode.nodeName == "gml:Box" || coordset[c].parentNode.nodeName == "gml:Envelope" )){
	skiprender = true;
	};

      var coords = GXml.value(coordset[c]);
      coords += " ";
      coords=coords.replace(/\s+/g," "); // tidy the whitespace
      coords=coords.replace(/^ /,"");    // remove possible leading whitespace
      coords=coords.replace(/, /,",");   // tidy the commas
      var path = coords.split(" ");
      // Is this a polyline/polygon?
      if (path.length == 1 || path[1] =="") {
        var bits = path[0].split(",");
        var point = new GLatLng(parseFloat(bits[1])/trans.ys-trans.y,parseFloat(bits[0])/trans.xs-trans.x);
        that.bounds.extend(point);
        // Does the user have their own createmarker function?
	if(!skiprender){
		if(!name)name="un-named place";
        	if (!!that.opts.createmarker) {
          		that.opts.createmarker(point, name, desc, styleid, idx, null, visible);
        		} 
		else {
          		that.createMarker(point, name, desc, styleid, idx, null, visible);
        		}
		}
	}
      else {
        // Build the list of points
        var points = [];
        var pbounds = new GLatLngBounds();
       		for (var p=0; p<path.length-1; p++) {
         	 var bits = path[p].split(",");
         	 var point = new GLatLng(parseFloat(bits[1])/trans.ys-trans.y,parseFloat(bits[0])/trans.xs-trans.x);
         	 points.push(point);
         	 pbounds.extend(point);
         	 }
	 	that.bounds.extend(pbounds.getNorthEast());
	 	that.bounds.extend(pbounds.getSouthWest());
		if(!skiprender) lines.push(points);
	     }
	}
 	if(!lines || lines.length <1) return;
        var linestring=mark.getElementsByTagName("LineString");
	if(linestring.length <1){
		linestring=mark.getElementsByTagName("gml:LineString");
		}
        if (linestring.length || line_count>0) {
          // its a polyline grab the info from the style
          if (!!style) {
            var width = style.width; 
            var color = style.color; 
            var opacity = style.opacity; 
          } else {
            var width = this.style.width;
            var color = this.style.color;
            var opacity = this.style.opacity;
          }
          // Does the user have their own createpolyline function?
	if(!name)name="un-named path";
          if (!!that.opts.createpolyline) {
            that.opts.createpolyline(lines,color,width,opacity,pbounds,name,desc,idx,visible);
          } else {
            that.createPolyline(lines,color,width,opacity,pbounds,name,desc,idx,visible);
          }
        }
        var polygons=mark.getElementsByTagName("Polygon");
	if(polygons.length <1){
		polygons=mark.getElementsByTagName("gml:Polygon");
		}
        if (polygons.length || poly_count>0) {
          // its a polygon grab the info from the style
          if (!!style) {
            var width = style.width; 
            var color = style.color; 
            var opacity = style.opacity; 
            var fillopacity = style.fillopacity; 
            var fillcolor = style.fillcolor; 
            fill = style.file;
	    outline = style.outline;
          } 
	if(!name)name="un-named area";

 	if (!!that.opts.createpolygon) {
            that.opts.createpolygon(lines,color,width,opacity,fillcolor,fillopacity,pbounds,name,desc,idx,visible,fill,outline);
          } else {
            that.createPolygon(lines,color,width,opacity,fillcolor,fillopacity,pbounds,name,desc,idx,visible,fill,outline);
          }
      }  
    };

GeoXml.prototype.handlePlacemark = function (mark, idx, depth, fullstyle) {
     var that = this;
     var desc, title, name, style;
     title = "";
     desc = "";
     var styleid = 0;
     var lat, lon;
     var visible = true;
     var newcoords = false;
     var l = mark.getAttribute("lat");
     if(typeof l!="undefined")lat = l;
     l = mark.getAttribute("lon");
     if(typeof l!="undefined"){
		newcoords = true;
		lon = l;
		}

     var coordset=mark.getElementsByTagName("coordinates");
	if(coordset.length <1){
	    coordset=mark.getElementsByTagName("gml:coordinates");
	    }
	if(coordset.length <1){
	   	coordset = [];
	    	var poslist =mark.getElementsByTagName("gml:posList");
		for(var l =0;l<poslist.length;l++){
			var coords = " ";
			var plitem = GXml.value(poslist.item(l)) + " ";
			var cor = plitem.split(' ');
			for(var cc=0;cc<(cor.length-1);cc++){
					if(cor[cc] && cor[cc]!=" " && !isNaN(parseFloat(cor[cc]))){
						coords += ""+parseFloat(cor[cc])+","+parseFloat(cor[cc+1]);
						coords += " ";
						}
					cc++;
					}
			if(coords){
 				if(poslist.item(l).parentNode && (poslist.item(l).parentNode.nodeName == "gml:LineString") )line_count++;
					else poly_count++;
				var cm = "<coordinates>"+coords+"</coordinates>";
				var node = GXml.parse(cm);
				if(coordset.push)coordset.push(node); 
				}
			}

		var pos = mark.getElementsByTagName("gml:pos");
		if(pos.length <1)pos = mark.getElementsByTagName("gml:pos");
		if(pos.length){
			for(var p=0;p<pos.length;p++){
				var nv = GXml.value(pos.item(p))+" ";
				var cor = nv.split(' ');
				var node = GXml.parse("<coordinates>"+cor[0]+","+cor[1]+"</coordinates>");
				if(coordset.push)coordset.push(node); 
				}
			}
	    }


	var coords = "";
	var point_count =0;
	var box_count=0;
	var line_count=0;
	var poly_count=0;
 	for (var ln = 0; ln < mark.childNodes.length; ln++) {
		var nn = mark.childNodes.item(ln).nodeName;
		var nv = GXml.value(mark.childNodes.item(ln));
		var ns = nn.split(":");
		var base;
		if(ns.length>1)base = ns[1].toLowerCase();
			else base = ns[0].toLowerCase();	
		
		processme = false;
		switch(base){
			case "name": 
				name = nv;
				if(name.length+depth > this.maxtitlewidth)this.maxtitlewidth = name.length+depth;
				break;
			case "title":
				title = nv;
				if(title.length+depth > this.maxtitlewidth)this.maxtitlewidth = title.length+depth;
				break;
			case "desc":
			case "description":
				desc += nv;
				if(desc.match(/<(\s)*img/i)){
					var preload = document.createElement("span");
     					preload.style.visibility = "visible";
					preload.style.position = "absolute";
					preload.style.left = "-1200px";
					preload.style.top = "-1200px";
					preload.style.zIndex = this.overlayman.markers.length; 
     					document.body.appendChild(preload);
					preload.innerHTML = desc;
					}	
				break;
			case "visibility":
 				if(nv == "0")visible = false;
				break;
			case "href":
			case "link":
				desc += "<p><a target='_blank' href='"+nv+"'>link</a></p>";
				break;
			case "author":
				desc += "<p><b>author:</b>"+nv+"</p>";
				break;
			case "time":
				desc += "<p><b>time:</b>"+nv+"</p>";
				break;
			case "lat":
				lat=nv; 
				break; 
			case "long":
				lon=nv; 
				newcoords = true;
				break;
			case "point":
				point_count++;
                                processme = true;
				break;
			case "line":
				line_count++;processme = true;break;
			case "box":
				box_count++;processme = true;break;
			case "polygon":
				poly_count++;processme = true;break;
				break;
			case "styleurl":
				styleid = nv;
			 	break;
			case "stylemap" :
				var found = false;
				var node = mark.childNodes.item(ln);
				for(var j=0;(j<node.childNodes.length && !found);j++){ 
					var pair = node.childNodes[j];
					for(var k =0;(k<pair.childNodes.length && !found);k++){
						var pn = pair.childNodes[k].nodeName;
						if(pn == "Style"){
							style = this.handleStyle(pair.childNodes[k]);
							found = true;
							}
						}
					}
				break;

			case "style":
				style = this.handleStyle(mark.childNodes.item(ln));
				break;
			}
			if(processme){
				var cor = nv.split(' ');
				coords = "";
				for(var cc=0;cc<(cor.length-1);cc++){
					if(cor[cc] && cor[cc]!=" " && !isNaN(parseFloat(cor[cc]))){
						coords += ""+parseFloat(cor[cc+1])+","+parseFloat(cor[cc]);
						if(cc>1 && cc<(cor.length-2)) coords += " ";
						}
					cc++;
					}
				if(coords !=""){
					var node = GXml.parse("<coordinates>"+coords+"</coordinates>");
					if(coordset.push)coordset.push(node); 
					}
				}

		}

      if(!name && title)name = title; 

      if(fullstyle){
		style = fullstyle;
		}
      if(styleid){
		style = this.styles[styleid];
		}

      if(typeof desc == "undefined" || !desc  ){
	    var dc = that.makeDescription(mark,"");
//	    desc = "<ul>"+dc.desc+"</ul>";
	    if(!name && dc.title){
			name = dc.title;
			if((name.length +depth) > this.maxtitlewidth){
				this.maxtitlewidth = name.length + depth;
				}
			}
	    }
      
     if(newcoords && typeof lat!="undefined"){
        if(lat){
		    var cs = ""+lon+","+lat+" ";
		    var node = GXml.parse("<coordinates>"+cs+"</coordinates>");
		    coordset.push(node);
		    }
		}
    
     var lines = [];
     for(var c=0;c<coordset.length;c++){
      var skiprender =false;
     if(coordset[c].parentNode && (coordset[c].parentNode.nodeName.match(/^(gml:Box|gml:Envelope)/i))){
	skiprender = true;
	};

      var coords = GXml.value(coordset[c]);
      coords += " ";
      coords=coords.replace(/\s+/g," "); // tidy the whitespace
      coords=coords.replace(/^ /,"");    // remove possible leading whitespace
      //coords=coords +" "; //ensure trailing space
      coords=coords.replace(/, /,",");   // tidy the commas
      var path = coords.split(" ");
      // Is this a polyline/polygon?
      
      if (path.length == 1 || path[1]== "") {
        var bits = path[0].split(",");
        var point = new GLatLng(parseFloat(bits[1]),parseFloat(bits[0]));
        this.overlayman.folderBounds[idx].extend(point);
        // Does the user have their own createmarker function?
	if(!skiprender){
		if(!name)name="un-named place";
        	if (!!that.opts.createmarker) {
          		that.opts.createmarker(point, name, desc, styleid, idx, style, visible);
        		} 
		else {
          		that.createMarker(point, name, desc, styleid, idx, style, visible);
        		}
		}
	}
      else {
        // Build the list of points
        var points = [];
        var pbounds = new GLatLngBounds();
       		for (var p=0; p<path.length-1; p++) {
         	 var bits = path[p].split(",");
         	 var point = new GLatLng(parseFloat(bits[1]),parseFloat(bits[0]));
         	 points.push(point);
         	 pbounds.extend(point);
         	 }
		this.overlayman.folderBounds[idx].extend(pbounds.getSouthWest());
	 	this.overlayman.folderBounds[idx].extend(pbounds.getNorthEast());
		if(!skiprender) lines.push(points);
	    }
	}
 	if(!lines || lines.length <1) return;
        var linestring=mark.getElementsByTagName("LineString");
	if(linestring.length <1){
		linestring=mark.getElementsByTagName("gml:LineString");
		}
        if (linestring.length || line_count>0) {
          // its a polyline grab the info from the style
          if (!!style) {
            var width = style.width; 
            var color = style.color; 
            var opacity = style.opacity; 
          } else {
            var width = this.style.width;
            var color = this.style.color;
            var opacity = this.style.opacity;
          }
          // Does the user have their own createmarker function?
	if(!name)name="un-named path";
          if (!!that.opts.createpolyline) {
            that.opts.createpolyline(lines,color,width,opacity,pbounds,name,desc,idx,visible);
          } else {
            that.createPolyline(lines,color,width,opacity,pbounds,name,desc,idx,visible);
          }
        }
        var polygons=mark.getElementsByTagName("Polygon");
	if(polygons.length <1){
		polygons=mark.getElementsByTagName("gml:Polygon");
		}
        if (polygons.length || poly_count>0) {
          // its a polygon grab the info from the style
          if (!!style) {
            var width = style.width; 
            var color = style.color; 
            var opacity = style.opacity; 
            var fillopacity = style.fillopacity; 
            var fillcolor = style.fillcolor;
	    var fill = style.fill;
	    var outline = style.outline; 
          } 
	if(!name)name="un-named area";
 	if (!!that.opts.createpolygon) {
            that.opts.createpolygon(lines,color,width,opacity,fillcolor,fillopacity,pbounds,name,desc,idx,visible,fill,outline);
          } else {
            that.createPolygon(lines,color,width,opacity,fillcolor,fillopacity,pbounds,name,desc,idx,visible,fill,outline);
          }
      }  
    };

GeoXml.prototype.makeIcon = function(tempstyle, href){
	if (!!href) {
          if (!!this.opts.baseicon) {
           tempstyle = new GIcon(this.opts.baseicon,href);
	   tempstyle.href = href;
          } else {
            tempstyle = new GIcon(G_DEFAULT_ICON,href);
            tempstyle.iconSize = new GSize(32,32);
            tempstyle.shadowSize = new GSize(59,32);
            tempstyle.dragCrossAnchor = new GPoint(2,8);
            tempstyle.iconAnchor = new GPoint(16,32);
  	    tempstyle.href = href;
            if (this.opts.printgif) {
              var bits = href.split("/");
              var gif = bits[bits.length-1];
              gif = this.opts.printgifpath + gif.replace(/.png/i,".gif");
              tempstyle.printImage = gif;
              tempstyle.mozPrintImage = gif;
            }
            if (!!this.opts.noshadow) {
              tempstyle.shadow="";
            } else {
              // Try to guess the shadow image
              if (href.indexOf("/red.png")>-1 
               || href.indexOf("/blue.png")>-1 
               || href.indexOf("/green.png")>-1 
               || href.indexOf("/yellow.png")>-1 
               || href.indexOf("/lightblue.png")>-1 
               || href.indexOf("/purple.png")>-1
		|| href.indexOf("/orange.png")>-1 
               || href.indexOf("/pink.png")>-1 
		|| href.indexOf("-dot.png")>-1 ) {
                  tempstyle.shadow="http://maps.google.com/mapfiles/ms/icons/msmarker.shadow.png";
              }
              else if (href.indexOf("-pushpin.png")>-1  
		|| href.indexOf("/pause.png")>-1 
		|| href.indexOf("/go.png")>-1    
		|| href.indexOf("/stop.png")>-1     ) {
                  tempstyle.shadow="http://maps.google.com/mapfiles/ms/icons/pushpin_shadow.png";
              }
              else {
                var shadow = href.replace(".png",".shadow.png");
                tempstyle.shadow=shadow;
              }
            }
          }
        }
	return tempstyle;
	};
GeoXml.prototype.handleStyle = function(style,sid){
      var icons=style.getElementsByTagName("Icon");
      var tempstyle;
      if (icons.length > 0) {
        var href=GXml.value(icons[0].getElementsByTagName("href")[0]);
	tempstyle = this.makeIcon(tempstyle,href);
      	}
      // is it a LineStyle ?
      var linestyles=style.getElementsByTagName("LineStyle");
      if (linestyles.length > 0) {
        var width = parseInt(GXml.value(linestyles[0].getElementsByTagName("width")[0]));
        if (width < 1) {width = 5;}
        var color = GXml.value(linestyles[0].getElementsByTagName("color")[0]);
        var aa = color.substr(0,2);
        var bb = color.substr(2,2);
        var gg = color.substr(4,2);
        var rr = color.substr(6,2);
        color = "#" + rr + gg + bb;
        var opacity = parseInt(aa,16)/256;
        if (!tempstyle) {
          tempstyle = {};
        }
        tempstyle.color=color;
        tempstyle.width=width;
        tempstyle.opacity=opacity;
      }
      // is it a PolyStyle ?
      var polystyles=style.getElementsByTagName("PolyStyle");
      if (polystyles.length > 0) {
        var fill = parseInt(GXml.value(polystyles[0].getElementsByTagName("fill")[0]));
        var outline = parseInt(GXml.value(polystyles[0].getElementsByTagName("outline")[0]));
        var color = GXml.value(polystyles[0].getElementsByTagName("color")[0]);
	var colormode = GXml.value(polystyles[0].getElementsByTagName("colorMode")[0]);

	
        if (polystyles[0].getElementsByTagName("fill").length == 0) {fill = 1;}
        if (polystyles[0].getElementsByTagName("outline").length == 0) {outline = 1;}
tempstyle.fill = fill;
tempstyle.outline = outline;
        var aa = color.substr(0,2);
        var bb = color.substr(2,2);
        var gg = color.substr(4,2);
        var rr = color.substr(6,2);
        color = "#" + rr + gg + bb;
        var opacity = parseInt(aa,16)/256;
        if (!tempstyle) {
          tempstyle = {};
        }
	if(colormode != "random") {
        	tempstyle.fillcolor = color;
		}
	else {
		tempstyle.colortint = color;
		}
        tempstyle.fillopacity=opacity;
        if (!fill) tempstyle.fillopacity = 0; 
        if (!outline) tempstyle.opacity = 0; 
      }
	if(sid)this.styles["#"+sid] = tempstyle;
	return tempstyle;
};
GeoXml.prototype.processKML = function(node, marks, title, sbid, depth) {  
	var that = this;
	if (node.nodeName == "kml")icon = this.docicon;
        if (node.nodeName == "Document" )icon = this.kmlicon;
	if (node.nodeName == "Folder")icon = this.foldericon;
	var pm = [];
	var sf = [];
	var desc= "";
	var snip ="";
	var i;
	var open = that.forcefoldersopen;
	var visible = true;
	var boundsmodified = false;
        networklink = false;
	var url;
	var ground = null;
	var opacity = 1.0;
	var makewms = false;
	for (var ln = 0; ln < node.childNodes.length; ln++) {
		var nextn = node.childNodes.item(ln);
		var nn = nextn.nodeName;
		var nv = nextn.nodeValue;
		switch (nn) {
		 	case "name":  
			case "title": 
				title = GXml.value(nextn);
				if(title.length + depth > this.maxtitlewidth){ this.maxtitlewidth = title.length+depth;	}
			 	break;
			case "Folder" :
			case "Document" :
				sf.push(nextn); 
				break;
		 	case "GroundOverlay":
				url=GXml.value(nextn.getElementsByTagName("href")[0]);
      				var north=parseFloat(GXml.value(nextn.getElementsByTagName("north")[0]));
     		 		var south=parseFloat(GXml.value(nextn.getElementsByTagName("south")[0]));
      				var east=parseFloat(GXml.value(nextn.getElementsByTagName("east")[0]));
      				var west=parseFloat(GXml.value(nextn.getElementsByTagName("west")[0]));
      				var sw = new GLatLng(south,west);
      				var ne = new GLatLng(north,east);  
				this.bounds.extend(sw); 
      				this.bounds.extend(ne); 
				var color=GXml.value(nextn.getElementsByTagName("color")[0]);
				opacity = parseInt(color.substring(1,3),16)/256;
				var rs = /request=getmap/i;     
				if(url.search(rs)==-1 ){
      					ground = new GGroundOverlay(url, new GLatLngBounds(sw,ne));
					ground.sw = sw;
					ground.ne = ne;
					boundsmodified = true;
					ground.getBounds = function(){return new GLatLngBounds(this.sw,this.ne); };
      			 		}
				else {
					url += "&bbox="+west+","+south+","+east+","+north;
					makewms = true;
					}
				var title = GXml.value(nextn.getElementsByTagName("name")[0]);
				break;
		 	case "NetworkLink":
				var url = GXml.value(nextn.getElementsByTagName("href")[0]);
				networklink = true;
				break;
			case "description" :
			case  "Description":
				desc = GXml.value(nextn);
				break;
			case "open":
				if(GXml.value(nextn) == "1") open = true;
				if(GXml.value(nextn) == "0") open = this.forcefoldersopen;
				break;
			case "visibility":
				if(GXml.value(nextn) == "0") visible = false;
				break;
			case "snippet" :
				snip = GXml.value(nextn);
				break;

			default:
				for(var k=0;k<marks.length;k++){
					if(nn == marks[k]){
						pm.push(nextn);
						}					
					}
				}
			}

	
	
	var folderid;
	var idx = this.overlayman.folders.length;
	if(sf.length >1 || pm.length || ground || makewms){
        	this.overlayman.folders.push([]);
		this.overlayman.subfolders.push([]);
    		this.overlayman.folderhtml.push([]);
    		this.overlayman.folderhtmlast.push(0);
		this.overlayman.folderBounds.push(new GLatLngBounds());
  		this.kml.push(new KMLObj(title,desc));
		folderid = this.createFolder(idx, title, sbid, icon, desc, snip, open, visible);
		} 
	else {
		folderid = sbid;
		}
 
	if(ground || makewms){
		this.kml[this.kml.length-1].visibility = visible;
		this.kml[this.kml.length-1].groundOverlays.push({"url":url});
		}
	if(makewms){ 
		var ol = this.makeWMSTileLayer(url, visible, title, opacity);
		if(ol) {
			if (this.basesidebar) {
    				var n = this.overlayman.markers.length;
				url = url.replace(/height=(\d)+/i,"height=100")
				url = url.replace(/width=(\d)+/i,"width=100")
    				var blob = '<img src="'+url+'" style="width:100px" />';
   				parm =  this.myvar+"$$$" +title + "$$$tiledoverlay$$$" + n +"$$$" + blob + "$$$" +visible+"$$$"+(this.baseLayers.length-1); 
   				}
			var html = desc;
			var thismap = this.map; 
			GEvent.addListener(ol,"zoomto", function() { 	
						thismap.setZoom(thismap.getBoundsZoomLevel(this.getBounds()));
						thismap.panTo(this.getBounds().getCenter()); 
						});
	 		this.overlayman.folderBounds[idx].extend(ol.getBounds().getSouthWest());
	 		this.overlayman.folderBounds[idx].extend(ol.getBounds().getNorthEast());
	 		boundsmodified = true;
	 		this.overlayman.AddMarker(ol,title,idx, parm, true, true); 
			}
		 }

	if(networklink){
		var re = /&amp;/;
		url = url.replace(re,"&");
 		this.progress++;	
		if(!top.standalone){
			if(typeof this.proxy!="undefined") { url = this.proxy + escape(url); } 
			}
	 	var comm = this.myvar +".loadXMLUrl('"+url+"','"+title+"',null,null,'"+sbid+"');";
		setTimeout(comm,1000);
		return;
		}

	
	if(ground){
		if (this.basesidebar) {
    			var n = this.overlayman.markers.length;
    			var blob = '<span style="background-color:black;border:2px solid brown;">&nbsp;&nbsp;&nbsp;&nbsp;</span> ';
   			parm =  this.myvar+"$$$" +title + "$$$polygon$$$" + n +"$$$" + blob + "$$$" +visible+"$$$null"; 
   			}
		var html = desc;
		var thismap = this.map;
		GEvent.addListener(ground,"zoomto", function() { 	
						thismap.setZoom(thismap.getBoundsZoomLevel(this.getBounds()));
						thismap.panTo(this.getBounds().getCenter()); });
		this.overlayman.folderBounds[idx].extend(ground.getBounds().getSouthWest());
		this.overlayman.folderBounds[idx].extend(ground.getBounds().getNorthEast());
		boundsmodified = true;
		this.overlayman.AddMarker(ground,title,idx, parm, visible);
		}


	for(i=0;i<pm.length;i++) {
		this.handlePlacemark(pm[i], idx, depth+1);
		}
	var fc = 0;
	for(i=0;i<sf.length;i++) {
	 	var fid = this.processKML(sf[i], marks, title, folderid, depth+1);
		if(typeof fid =="number"){
			this.overlayman.subfolders[idx].push(fid);
			this.overlayman.folderBounds[idx].extend(this.overlayman.folderBounds[fid].getSouthWest());
			this.overlayman.folderBounds[idx].extend(this.overlayman.folderBounds[fid].getNorthEast());
			if(fid!=idx){ this.kml[idx].folders.push(fid); }
			fc++;
			}
		}
	 
	if(fc || pm.length || boundsmodified){
		this.bounds.extend(this.overlayman.folderBounds[idx].getSouthWest());
		this.bounds.extend(this.overlayman.folderBounds[idx].getNorthEast());
		}

	return idx;
	};


GeoXml.prototype.processing = function(xmlDoc,title, latlon, desc, sbid) {
    this.overlayman.miStart = new Date();
    if(!desc)desc = title;
    var that = this;
    if(!sbid)sbid = 0;

    var root = xmlDoc.documentElement;
    if(!root){ alert("No document found"); return 0; }
    var placemarks = [];
    basename = root.nodeName;
    var bases = basename.split(":");
    if(bases.length>1){basename = bases[1];}

    if(basename == "FeatureCollection"){
		var bar = $(this.basesidebar);
		if(title.length > this.maxtitlewidth){
				this.maxtitlewidth = title.length;
				}
		bar.style.display="";
		var idx = this.overlayman.folders.length;
		this.processGML(root,title,latlon,desc);
		this.kml[0].folders.push(idx);
		}

    if(basename =="gpx"){
	if(!title)title = name;
        this.title = title;
	if(title.length >this.maxtitlewidth){
		this.maxtitlewidth = title.length;
		}

	var bar = $(this.basesidebar);
	if(bar)bar.style.display="";
	var idx = this.overlayman.folders.length;
	this.processGPX(root, title, this.basesidebar, sbid);
	this.kml[0].folders.push(idx);
	}
    else {

   if(basename == "kml") {	
   	var styles = root.getElementsByTagName("Style"); 
   	for (var i = 0; i <styles.length; i++) {
    		var sid= styles[i].getAttribute("id");
      		if(sid){ 
     	   		this.handleStyle(styles[i],sid);
	    		}
   	 	}
	var styles = root.getElementsByTagName("StyleMap");
	for (var i = 0; i <styles.length; i++) {
		var sid = styles[i].getAttribute("id");
		if(sid){
			var found = false;
			var node = styles[i];
			for(var j=0;(j<node.childNodes.length && !found);j++){ 
				var pair = node.childNodes[j];
				for(var k =0;(k<pair.childNodes.length && !found);k++){
					var pn = pair.childNodes[k].nodeName;
					if(pn == "styleUrl"){
						var pid = GXml.value(pair.childNodes[k]);
						this.styles["#"+sid] = this.styles[pid];
						found = true;
						}
					if(pn == "Style"){
						this.handleStyle(pair.childNodes[k],sid);
						found = true;
						}
					}
				}
			}
		}

	if(!title)title = name;
        this.title = title;
	if(title.length >this.maxtitlewidth){
		this.maxtitlewidth = title.length;
		}
	var marknames = ["Placemark"];
	var schema = root.getElementsByTagName("Schema");  
	for(var s=0;s<schema.length;s++){
		var pname = schema[s].getAttribute("parent");
		if(pname == "Placemark"){
				var pname = schema[s].getAttribute("name");
			 	marknames.push(pname);
				}
			}

	var bar = $(this.basesidebar);
	if(bar)bar.style.display="";
	var idx = this.overlayman.folders.length;
	var fid = this.processKML(root, marknames, title, this.basesidebar, 0);	
	this.kml[0].folders.push(idx);
	}
     else { 
	placemarks = root.getElementsByTagName("item");
	if(placemarks.length <1){
		placemarks = root.getElementsByTagName("atom");
		}
 
    	if(!title)title = name;
        this.title = title;
	if(title.length >this.maxtitlewidth){
		this.maxtitlewidth = title.length;
		}

        style = new GIcon(G_DEFAULT_ICON,this.rssicon);
        style.iconSize = new GSize(32,32);
        style.shadowSize = new GSize(59,32);
        style.dragCrossAnchor = new GPoint(2,8);
        style.iconAnchor = new GPoint(16,32);
	style.href = this.rssicon;
        var shadow = this.rssicon.replace(".png",".shadow.png");
        style.shadow = shadow +"_shadow.png";
	style.color = "#00FFFF";
	style.width = "3"
	style.opacity = 0.50;
	if(!desc)desc = "RSS feed";
	this.kml[0].folders.push(this.overlayman.folders.length);
    	if(placemarks.length) {
		var bar = $(that.basesidebar);
		if(bar)bar.style.display="";
		this.kml.push(new KMLObj(title,desc));
        	this.overlayman.folders.push([]);
       		this.overlayman.folderhtml.push([]);
		this.overlayman.folderhtmlast.push(0);
		this.overlayman.folderBounds.push(new GLatLngBounds());
        	var idx = this.overlayman.folders.length-1;	
		if(this.basesidebar) { 	var folderid = this.createFolder(idx,title,this.basesidebar,this.globalicon,desc,null,true,true); }
    		for (var i = 0; i < placemarks.length; i++) {
     			this.handlePlacemark(placemarks[i], idx, sbid, style);
    			}
		}
	}

    }

    // Is this the last file to be processed?
    this.progress--;
    if (that.progress == 0) {
      // Shall we zoom to the bounds?
      if (!that.opts.nozoom) {
        that.map.setZoom(that.map.getBoundsZoomLevel(that.bounds));
        that.map.setCenter(that.bounds.getCenter());
      }

      GEvent.trigger(that,"parsed");
      if(!this.opts.sidebarid){
	this.mb.showMess("Finished Parsing",1000);
	}
     
    }
};


 
GeoXml.prototype.createFolder = function(idx, title, sbid, icon, desc, snippet, keepopen, visible){ 	      
		var sb = $(sbid);	
	 	folderid = this.myvar+'_folder'+ idx;
                var checked ="";
		if(visible){ checked = " checked " }
		this.overlayman.folderhtml[folderid]="";
		var disp="display:block";
		var fw= "font-weight:normal";
 		if(typeof keepopen == "undefined" || !keepopen){
			disp ="display:none";
			fw = "font-weight:bold";
	 		}
		if(!desc || desc ==""){
			desc = title;
			}
		desc = escape(desc);
		var htm = ''
//		htm += '<input type="checkbox" id="'+this.myvar+''+idx+'FCB" style="vertical-align:middle" ';
//		htm += checked;
//		htm += 'onclick="'+this.myvar+'.toggleContents('+idx+',this.checked)">';
		htm += '&nbsp;<span title="'+snippet+'" id="'+this.myvar+'TB'+idx+'" oncontextmenu=\"'+this.myvar+'.saveJSON('+idx+');\" onclick="'+this.myvar+'.toggleFolder('+idx+')" style=\"'+fw+'\">';
		htm += '<img style=\"vertical-align:text-top;padding:0;margin:0\" height=\"16\" border=\"0\" src="'+icon+'" /></span>&nbsp;';
		htm += '<a href="#" style="text-decoration: none;font-size: 8pt" onclick="'+this.myvar+'.overlayman.zoomToFolder('+idx+');'+this.myvar+'.mb.showMess(\''+desc+'\',3000);return false;">' + title + '</a><br><span id=\"'+folderid+'\" style="height:14px;'+disp+'"></span>';
		if(sb){ sb.innerHTML = htm + sb.innerHTML; }
		return folderid;
	    };

// PolylineEncoder.js copyright Mark McClure  April/May 2007
//
// This software is placed explicitly in the public
// domain and may be freely distributed or modified.
// No warranty express or implied is provided.

PolylineEncoder = function(numLevels, zoomFactor, verySmall, forceEndpoints) {
  var i;
  if(!numLevels) {
    numLevels = 18;
  }
  if(!zoomFactor) {
    zoomFactor = 2;
  }
  if(!verySmall) {
    verySmall = 0.0000001;
  }
  if(!forceEndpoints) {
    forceEndpoints = true;
  }
  this.numLevels = numLevels;
  this.zoomFactor = zoomFactor;
  this.verySmall = verySmall;
  this.veryTiny = verySmall * verySmall;
  this.forceEndpoints = forceEndpoints;
  this.zoomLevelBreaks = new Array(numLevels);
  for(i = 0; i < numLevels; i++) {
    this.zoomLevelBreaks[i] = verySmall*Math.pow(zoomFactor, numLevels-i-1);
    this.zoomLevelBreaks[i] *= this.zoomLevelBreaks[i];
  }
}

// The main function.  Essentially the Douglas-Peucker
// algorithm, adapted for encoding. Rather than simply
// eliminating points, we record their from the
// segment which occurs at that recursive step.  These
// distances are then easily converted to zoom levels.
PolylineEncoder.prototype.dpEncode = function(points) {
  var absMaxDist = 0;
  var stack = [];
  var dists = new Array(points.length);
  var maxDist, maxLoc, temp, first, last, current;
  var i, encodedPoints, encodedLevels;
  var segmentLength;
  
 // if(points.length > 2) {
    stack.push([0, points.length-1]);
    while(stack.length > 0) {
      current = stack.pop();
      maxDist = 0;
      segmentLength = Math.pow(points[current[1]].lat()-points[current[0]].lat(),2) + Math.pow(points[current[1]].lng()-points[current[0]].lng(),2);
      for(i = current[0]+1; i < current[1]; i++) {
        temp = this.distance(points[i], points[current[0]], points[current[1]], segmentLength);
        if(temp > maxDist) {
          maxDist = temp;
          maxLoc = i;
          if(maxDist > absMaxDist) {
            absMaxDist = maxDist;
          }
        }
      }
      if(maxDist > this.veryTiny ) {
        dists[maxLoc] = maxDist;
        stack.push([current[0], maxLoc]);
        stack.push([maxLoc, current[1]]);
      }
    }
 // }
   
  encodedPoints = this.createEncodings(points, dists);
  encodedLevels = this.encodeLevels(points, dists, absMaxDist);
  return {
    encodedPoints: encodedPoints,
    encodedLevels: encodedLevels,
    encodedPointsLiteral: encodedPoints.replace(/\\/g,"\\\\")
  }
}

PolylineEncoder.prototype.dpEncodeToJSON = function(points,
  color, weight, opacity) {
  var result;

  result = this.dpEncode(points);
  return {
    color: color,
    weight: weight,
    opacity: opacity,
    points: result.encodedPoints,
    levels: result.encodedLevels,
    numLevels: this.numLevels,
    zoomFactor: this.zoomFactor,
    literals: result.encodePointsLiteral
  }
}

PolylineEncoder.prototype.dpEncodeToGPolyline = function(points, color, weight, opacity) {
  return new GPolyline.fromEncoded(
    this.dpEncodeToJSON(points, color, weight, opacity));
}

PolylineEncoder.prototype.dpEncodeToGPolygon = function(pointsArray,
  boundaryColor, boundaryWeight, boundaryOpacity,
  fillColor, fillOpacity, fill, outline) {
  var i, boundaries;

  boundaries = new Array(0);
  for(i=0; i<pointsArray.length; i++) {
    boundaries.push(this.dpEncodeToJSON(pointsArray[i],
      boundaryColor, boundaryWeight, boundaryOpacity));
  }
  return new GPolygon.fromEncoded({
    polylines: boundaries,
    color: fillColor,
    opacity: fillOpacity,
    fill: fill,
    outline: outline
  });
}

// distance(p0, p1, p2) computes the square distance between the point p0
// and the segment [p1,p2].  This could probably be replaced with
// something that is a bit more numerically stable.
PolylineEncoder.prototype.distance = function(p0, p1, p2, segLength) {
  var u, out;
   var dlat2 = p2.lat() - p1.lat();
   var dlong1 = p2.lng() - p1.lng();
  if(dlat2 === 0 && dlong1 === 0) {
    out = Math.pow(p2.lat()-p0.lat(),2) + Math.pow(p2.lng()-p0.lng(),2);
  }
  else {
   var dlat0 = p0.lat() - p1.lat();
   var dlong0 = p0.lng() - p1.lng();
    u = ((dlat0)*(dlat2)+(dlong0)*(dlong1))/segLength;
    if(u <= 0) {
      out = Math.pow(dlat0,2) + Math.pow(dlong0,2);
    }
    if(u >= 1) {
      out =  Math.pow(p0.lat() - p2.lat(),2) + Math.pow(p0.lng() - p2.lng(),2);
    }
    if(0 < u && u < 1) {
      out = Math.pow(dlat0-u*(dlat2),2) + Math.pow(dlong0-u*(dlong1),2);
    }
  }
  return out;
}

// The createEncodings function is very similar to Googles
// http://www.google.com/apis/maps/documentation/polyline.js
// The key difference is that not all points are encoded, 
// since some were eliminated by Douglas-Peucker.
PolylineEncoder.prototype.createEncodings = function(points, dists) {
  var i;
  var j;
  var floor = Math.floor;
  var len = points.length-2;
  var late5;
  var lnge5;
  var dlat = floor(points[0].lat() * 1e5);
  var dlng = floor(points[0].lng() * 1e5);
  var plat = dlat;
  var plng = dlng;
  var encoded_points = this.encodeSignedNumber(dlat) + this.encodeSignedNumber(dlng);
  for(j = len; j >0; j--) {
    var i = len - j+1;
    if(dists[i] != undefined) {
      late5 = floor(points[i].lat() * 1e5);
      lnge5 = floor(points[i].lng() * 1e5);
      dlat = late5 - plat;
      dlng = lnge5 - plng;
      plat = late5;
      plng = lnge5;
      encoded_points += this.encodeSignedNumber(dlat) + this.encodeSignedNumber(dlng);
    }
  }
   dlat = floor(points[len+1].lat() * 1e5) - plat;
   dlng = floor(points[len+1].lng() * 1e5) - plng;
   encoded_points += this.encodeSignedNumber(dlat) + this.encodeSignedNumber(dlng);
  return encoded_points;
}

// This computes the appropriate zoom level of a point in terms of its 
// distance from the relevant segment in the DP algorithm.  Could be done
// in terms of a logarithm, but this approach makes it a bit easier to
// ensure that the level is not too large.
PolylineEncoder.prototype.computeLevel = function(dd) {
  var lev;
  if(dd > this.veryTiny) {
    lev=0;
    while(dd < this.zoomLevelBreaks[lev]) {
      lev++;
    }
    return lev;
  }
}

// Now we can use the previous function to march down the list
// of points and encode the levels.  Like createEncodings, we
// ignore points whose distance (in dists) is undefined.
PolylineEncoder.prototype.encodeLevels = function(points, dists, absMaxDist) {
  var i;
  var encoded_levels = "";
  if(this.forceEndpoints) {
    encoded_levels += this.encodeNumber(this.numLevels-1)
  } else {
    encoded_levels += this.encodeNumber(
      this.numLevels-this.computeLevel(absMaxDist)-1)
  }
  for(i=1; i < points.length-1; i++) {
    if(dists[i] != undefined) {
      encoded_levels += this.encodeNumber(
        this.numLevels-this.computeLevel(dists[i])-1);
    }
  }
  if(this.forceEndpoints) {
    encoded_levels += this.encodeNumber(this.numLevels-1)
  } else {
    encoded_levels += this.encodeNumber(
      this.numLevels-this.computeLevel(absMaxDist)-1)
  }
  return encoded_levels;
}

// This function is very similar to Googles, but I added
// some stuff to deal with the double slash issue.
PolylineEncoder.prototype.encodeNumber = function(num) {
  var encodeString = "";
  var nextValue, finalValue;
  while (num >= 0x20) {
    nextValue = (0x20 | (num & 0x1f)) + 63;
//     if (nextValue == 92) {
//       encodeString += (String.fromCharCode(nextValue));
//     }
    encodeString += (String.fromCharCode(nextValue));
    num >>= 5;
  }
  finalValue = num + 63;
//   if (finalValue == 92) {
//     encodeString += (String.fromCharCode(finalValue));
//   }
  encodeString += (String.fromCharCode(finalValue));
  return encodeString;
}

// This one is Googles verbatim.
PolylineEncoder.prototype.encodeSignedNumber = function(num) {
  var sgn_num = num << 1;
  if (num < 0) {
    sgn_num = ~(sgn_num);
  }
  return(this.encodeNumber(sgn_num));
}


// The remaining code defines a few convenience utilities.
// PolylineEncoder.latLng
PolylineEncoder.latLng = function(y, x) {
	this.y = y;
	this.x = x;
}
PolylineEncoder.latLng.prototype.lat = function() {
	return this.y;
}
PolylineEncoder.latLng.prototype.lng = function() {
	return this.x;
}

// PolylineEncoder.pointsToLatLngs
PolylineEncoder.pointsToLatLngs = function(points) {
	var i, latLngs;
	latLngs = new Array(0);
	for(i=0; i<points.length; i++) {
		latLngs.push(new PolylineEncoder.latLng(points[i][0], points[i][1]));
	}
	return latLngs;
}

// PolylineEncoder.pointsToGLatLngs
PolylineEncoder.pointsToGLatLngs = function(points) {
	var i, gLatLngs;
	gLatLngs = new Array(0);
	for(i=0; i<points.length; i++) {
		gLatLngs.push(new GLatLng(points[i][0], points[i][1]));
	}
	return gLatLngs;
}



GPolyline.prototype.getPoint = function () { return (this.getVertex(Math.round(this.getVertexCount()/2))); };
GPolyline.prototype.computeBounds = function() {
  var bounds = new GLatLngBounds();
  for (var i=0; i < this.getVertexCount() ; i++) {
	var v = this.getVertex(i);
	if(v) bounds.extend(v);
  	}
  this.bounds = bounds;
  return bounds;
}


GTileLayerOverlay.prototype.getBounds = function(){return this.bounds; };


GPolyline.prototype.getBounds = function() {
  if(typeof this.bounds!="undefined") return this.bounds;
   else return (this.computeBounds());
  }

GPolyline.count = 0;  
GPolyline.prototype.myredraw = GPolyline.prototype.redraw;
GPolyline.prototype.myinit = GPolyline.prototype.initialize;
GPolyline.prototype.initialize = function(force) {
   GPolyline.count++;
   this.dash = "";
   this.cursor = "help";
   this.domid = "GLine"+GPolyline.count;
   this.myinit(force); //original method
   this.className = "";
}

GPolyline.prototype.redraw = function(force) {
   this.myredraw(force); //original method
   var dom;
	if(!this.cursor)this.cursor = "help";
   if(navigator.userAgent.indexOf("MSIE") != -1){
        var shps = document.getElementsByTagName("shape"); 
        if(shps.length){
            dom = shps[shps.length-1];//assume ours is the most recently added by the superclass redraw
            if(this.title != null){
        	    dom.title = this.title;
        	    }
            dom.id = this.domid;//assign unique DOM id so we can modify attributes later
            }
   }
   else{
        var shps = document.getElementsByTagName("path"); 
	if(shps.length){
        dom = shps[shps.length-1];//assume ours is the most recently added by the superclass redraw
        if(this.title != null){
        	dom.setAttribute("title",this.title);
        	}
        dom.setAttribute("id",this.domid);//assign unique DOM id so we can modify attributes later
	}
   }
if(dom){
  	 dom.style.cursor = this.cursor; 
   //set up the appearance of our polygon
  	 this.setStrokeColor(this.strokeColor);
  	 this.setStrokeDash(this.dash);
  	 this.setStrokeOpacity(this.strokeOpacity);
	 if(this.strokeWeight) this.setStrokeWeight(this.strokeWeight);
   //set up event handlers
  //	 if(this.onClick) {
   //		var cclick = GEvent.callback(this,this.onClick);
//		}
   	var cover = GEvent.callback(this,this.onOver);
   	var cout = GEvent.callback(this,this.onOut);
   	GEvent.clearInstanceListeners(dom);//safety 
   	if(this.onClick) GEvent.addDomListener(dom,"click",function(event){cclick();});
   	GEvent.addDomListener(dom,"mouseover",function(){cover();});
   	GEvent.addDomListener(dom,"mouseout",function(){cout();});
	}
}

//event handlers
//GPolyline.prototype.onClick = function(){
 //   GEvent.trigger(this,"click");
//}
GPolyline.prototype.onOver = function(){
    GEvent.trigger(this,"mouseover");
}
GPolyline.prototype.onOut = function(){
    GEvent.trigger(this,"mouseout");
}

//once the shape has been drawn, we can modify it with these setX functions;

GPolyline.prototype.setStrokeColor = function(color) {
    this.strokeColor = color;
    var dom = $(this.domid); 
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.stroke.color = this.strokeColor;
    }
    else{
        dom.setAttribute("stroke",this.strokeColor);
    }
}
GPolyline.prototype.getStrokeColor = function() {
    return this.strokeColor;
}
GPolyline.prototype.setStrokeDash = function(dash) {
    this.dash = dash;
    var dom = $(this.domid); 
    if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        if(this.dash == "dash")
            dom.stroke.dashstyle = "dash";       
        else if (this.dash == "dot")
            dom.stroke.dashstyle = "dot";    
        else 
            dom.stroke.dashstyle = "";    
    }
    else{
        if(this.dash == "dash")
            dom.setAttribute("stroke-dasharray","10,10");
        else if (this.dash == "dot")
            dom.setAttribute("stroke-dasharray","3,17");
        else
            dom.setAttribute("stroke-dasharray","none");
    }
}
GPolyline.prototype.getStrokeDash = function() {
    return this.dash;
}
GPolyline.prototype.setStrokeWeight = function(weight) {
    this.strokeWeight = weight;
    var dom = $(this.domid); 
if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.stroke.weight = this.strokeWeight.toString()+"px";   
    }
    else{
        dom.setAttribute("stroke-width",this.strokeWeight.toString()+"px");
    }
}

GPolyline.prototype.setTitle = function(str) {
    this.title = str;
    var dom = $(this.domid); 
if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.title = this.title;   
    }
    else{
        dom.setAttribute("title",str);
    }
}

GPolyline.prototype.getStrokeWeight = function() {
    return this.strokeWeight;
}
GPolyline.prototype.setStrokeOpacity = function(opacity) {
    this.strokeOpacity = opacity;
    var dom = $(this.domid); 
if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.stroke.opacity = this.strokeOpacity;
    }
    else{
        dom.setAttribute("stroke-opacity",this.strokeOpacity);
    }
}
GPolyline.prototype.getStrokeOpacity = function() {
    return this.strokeOpacity;
 	}

/* Extension to GPolygon over-ride to provide mouse events and dashed styles etc for polygons */
GPolygon.count = 0;
GPolygon.prototype.getPoint = function() { return (this.getBounds().getCenter()); };
GPolygon.prototype.myredraw = GPolygon.prototype.redraw;
GPolygon.prototype.myinit = GPolygon.prototype.initialize;

GPolygon.prototype.initialize = function(force) {
   GPolygon.count++;
   this.strokeWeight = 2;
   this.dash = "";
   this.fillColor = this.color;
   this.fillOpacity = 0.4;
   this.strokeOpacity = this.strokeOpacity || 0.6;
   this.cursor = "help";
   this.domid = "gpoly"+GPolygon.count;
   this.myinit(force); //original method
   this.className = "";
}

GPolygon.prototype.redraw = function(force) {
   this.myredraw(force); //original method
   var dom;
   if(navigator.userAgent.indexOf("MSIE") != -1){
       var shps = document.getElementsByTagName("shape"); 
       dom = shps[shps.length-1];//assume ours is the most recently added by the superclass redraw
       dom.style.cursor = "help";//to show mouseover 
       if(this.title)dom.title = this.title;
       dom.id = this.domid;//assign unique DOM id so we can modify attributes later
        }
   else{
        var shps = document.getElementsByTagName("path"); 
	if(shps.length){
       		dom = shps[shps.length-1];//assume ours is the most recently added by the superclass redraw
 		dom.style.cursor = "help";//to show mouseover 
        	if(this.title)dom.setAttribute("title",this.title);
		}
        }
   if(dom){
   	dom.setAttribute("id",this.domid);//assign unique DOM id so we can modify attributes later
   //set up the appearance of our polygon
   	this.setStrokeColor(this.strokeColor);
   	this.setStrokeDash(this.dash);
   	this.setStrokeOpacity(this.strokeOpacity);
   	this.setStrokeWeight(this.strokeWeight);
   	this.setFillColor(this.fillColor);
   	this.setFillOpacity(this.fillOpacity);
   //set up event handlers
   //	if(this.onClick){
    //		var cclick = GEvent.callback(this,this.onClick);
    //	}
   	var cover = GEvent.callback(this,this.onOver);
   	var cout = GEvent.callback(this,this.onOut);
   	GEvent.clearInstanceListeners(dom);//safety 
   	if(this.onClick) GEvent.addDomListener(dom,"click",function(event){cclick();});
   	GEvent.addDomListener(dom,"mouseover",function(){cover();});
   	GEvent.addDomListener(dom,"mouseout",function(){cout();});
	}
}

//event handlers
//GPolygon.prototype.onClick = function(){
  //  GEvent.trigger(this,"click");
//}
GPolygon.prototype.onOver = function(){
    GEvent.trigger(this,"mouseover");
}
GPolygon.prototype.onOut = function(){
    GEvent.trigger(this,"mouseout");
}

//once the shape has been drawn, we can modify it with these setX functions;

GPolygon.prototype.setStrokeColor = function(color) {
    this.strokeColor = color;
    var dom = $(this.domid); if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.stroke.color = this.strokeColor;
    }
    else{
        dom.setAttribute("stroke",this.strokeColor);
    }
}
GPolygon.prototype.getStrokeColor = function() {
    return this.strokeColor;
}
GPolygon.prototype.setStrokeDash = function(dash) {
    this.dash = dash;
    var dom = $(this.domid); if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        if(this.dash == "dash")
            dom.stroke.dashstyle = "dash";       
        else if (this.dash == "dot")
            dom.stroke.dashstyle = "dot";    
        else 
            dom.stroke.dashstyle = "";    
    }
    else{
        if(this.dash == "dash")
            dom.setAttribute("stroke-dasharray","10,10");
        else if (this.dash == "dot")
            dom.setAttribute("stroke-dasharray","3,17");
        else
            dom.setAttribute("stroke-dasharray","none");
    }
}
GPolygon.prototype.getStrokeDash = function() {
    return this.dash;
}
GPolygon.prototype.setStrokeWeight = function(weight) {
    this.strokeWeight = weight;
    var dom = $(this.domid); if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.stroke.weight = this.strokeWeight.toString()+"px";   
    }
    else{
        dom.setAttribute("stroke-width",this.strokeWeight.toString()+"px");
    }
}

GPolygon.prototype.getStrokeWeight = function() {
    return this.strokeWeight;
}
GPolygon.prototype.setStrokeOpacity = function(opacity) {
    this.strokeOpacity = opacity;
    var dom = $(this.domid); if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.stroke.opacity = this.strokeOpacity;
    }
    else{
        dom.setAttribute("stroke-opacity",this.strokeOpacity);
    }
}
GPolygon.prototype.getStrokeOpacity = function() {
    return this.strokeOpacity;
}
GPolygon.prototype.setFillColor = function(color) {
    this.fillColor = color;
    var dom = $(this.domid); if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.fill.color = this.fillColor;
    }
    else{
        dom.setAttribute("fill",this.fillColor);
    }
}
GPolygon.prototype.getFillColor = function() {
    return this.fillColor;
}

GPolygon.prototype.setFillOpacity = function(opacity) {
    this.fillOpacity = opacity;
    var dom = $(this.domid); if(typeof dom=="undefined")return;
    if(navigator.userAgent.indexOf("MSIE") != -1){
        dom.fill.opacity = this.fillOpacity;
    }
    else{
        dom.setAttribute("fill-opacity",this.fillOpacity);
    }
}
GPolygon.prototype.getFillOpacity = function() {
    return this.fillOpacity;
}


Clusterer = function ( map , paren ) {
    this.myvar = paren.myvar;
    this.paren = paren;
    this.map = map;
    this.markers = [];
    this.clusters = [];
    this.timeout = null;
    this.folders = [];
    this.folderBounds = [];
    this.folderhtml = [];
    this.folderhtmlast = [];
    this.subfolders = [];
    this.currentZoomLevel = map.getZoom();

    this.maxVisibleMarkers = Clusterer.defaultMaxVisibleMarkers;
    this.gridSize = Clusterer.defaultGridSize;
    this.minMarkersPerCluster = Clusterer.defaultMinMarkersPerCluster;
    this.maxLinesPerInfoBox = Clusterer.defaultMaxLinesPerInfoBox;
    this.icon = Clusterer.defaultIcon;
   
    GEvent.addListener( map, 'zoomend', Clusterer.MakeCaller( Clusterer.Display, this ) );
    GEvent.addListener( map, 'moveend', Clusterer.MakeCaller( Clusterer.Display, this ) );
    GEvent.addListener( map, 'infowindowclose', Clusterer.MakeCaller( Clusterer.PopDown, this ) );
    };


Clusterer.defaultMaxVisibleMarkers =  650;
Clusterer.defaultGridSize = 15;
Clusterer.defaultMinMarkersPerCluster = 5;
Clusterer.defaultMaxLinesPerInfoBox = 15;
Clusterer.defaultIcon = new GIcon();
Clusterer.defaultIcon.image = 'http://www.acme.com/resources/images/markers/blue_large.PNG';
Clusterer.defaultIcon.shadow = 'http://www.acme.com/resources/images/markers/shadow_large.PNG';
Clusterer.defaultIcon.iconSize = new GSize( 30, 51 );
Clusterer.defaultIcon.shadowSize = new GSize( 56, 51 );
Clusterer.defaultIcon.iconAnchor = new GPoint( 13, 34 );
Clusterer.defaultIcon.infoWindowAnchor = new GPoint( 13, 3 );
Clusterer.defaultIcon.infoShadowAnchor = new GPoint( 27, 37 );


// Call this to change the cluster icon.
Clusterer.prototype.SetIcon = function ( icon )
    {
    this.icon = icon;
    };


// Changes the maximum number of visible markers before clustering kicks in.
Clusterer.prototype.SetMaxVisibleMarkers = function ( n )
    {
    this.maxVisibleMarkers = n;
    };


// Sets the minumum number of markers for a cluster.
Clusterer.prototype.SetMinMarkersPerCluster = function ( n )
    {
    this.minMarkersPerCluster = n;
    };


// Sets the maximum number of lines in an info box.
Clusterer.prototype.SetMaxLinesPerInfoBox = function ( n )
    {
    this.maxLinesPerInfoBox = n;
    };


// Call this to add a marker.
Clusterer.prototype.AddMarker = function (marker, title, idx, sidebar, visible, forcevisible) {
    if (marker.setMap != null){
		marker.setMap( this.map );
		}
    marker.hidden = (!visible || this.hideall);
    marker.title = title;
    this.folders[idx].push(this.markers.length);
    var bounds = this.map.getBounds();
    var vis = false;
    if(typeof marker.getBounds =="undefined"){
		if (bounds.contains(marker.getPoint())) vis = true; 
		}
	else {
	     var b = marker.getBounds();
	     if(!b.isEmpty()){
	            if(bounds.intersects(b))vis = true;  
		}
	}
    if(forcevisible)vis = true;
    var id = this.markers.length;
    this.markers.push(marker);
    if(vis){ 
	marker.onMap = true;
    	this.map.addOverlay(marker);
	if(marker.hidden)marker.hide();
 	}
    this.DisplayLater();
    if(sidebar){
	this.folderhtml[idx].push(sidebar);
	}
    return id;
    };

Clusterer.prototype.zoomToFolder = function (idx) {
	var bounds = this.folderBounds[idx];
	this.map.setZoom(this.map.getBoundsZoomLevel(bounds));
    	this.map.panTo(bounds.getCenter());
	}


// Call this to remove a marker.
Clusterer.prototype.RemoveMarker = function ( marker ) {
    for ( var i = 0; i < this.markers.length; ++i )
	if ( this.markers[i] == marker )
	    {
	    if ( marker.onMap )
		this.map.removeOverlay( marker );
	    for ( var j = 0; j < this.clusters.length; ++j )
		{
		var cluster = this.clusters[j];
		if ( cluster != null )
		    {
		    for ( var k = 0; k < cluster.markers.length; ++k )
			if ( cluster.markers[k] == marker )
			    {
			    cluster.markers[k] = null;
			    --cluster.markerCount;
			    break;
			    }
		    if ( cluster.markerCount == 0 )
			{
			this.ClearCluster( cluster );
			this.clusters[j] = null;
			}
		    else if ( cluster == this.poppedUpCluster )
			Clusterer.RePop( this );
		    }
		}
	    this.markers[i] = null;
	    break;
	    } 
    this.DisplayLater();
    };



Clusterer.prototype.DisplayLater = function ()
    {
    if ( this.timeout != null )
	clearTimeout( this.timeout );
    this.timeout = setTimeout( Clusterer.MakeCaller( Clusterer.Display, this ), 50);
    };


Clusterer.Display = function (clusterer)
    {
    var i, j, k, marker, cluster;
    clearTimeout( clusterer.timeout );
     
    var update_side = false;
    var count = 0;
    for(k = 0; k< clusterer.folderhtml.length ; k++ ){	
	var curlen = clusterer.folderhtml[k].length;
	var con = clusterer.folderhtmlast[k];
	if(con < curlen){
		var destid = clusterer.paren.myvar+"_folder"+k;
		var dest = $(destid);
	
		if(dest){
			if(clusterer.paren.opts.sortbyname){
			
			        content = dest.innerHTML;
				var clon = clusterer.folderhtml[k].sort();
				for(var l=0; l<curlen; l++){
 					var bits = clon[l].split("$$$",7);
          				content += clusterer.paren.sidebarfn(bits[0],bits[1],bits[2],bits[3],bits[4],bits[5],bits[6]); 
					}
				}
			else {
	 		       content = dest.innerHTML;
				var clon = clusterer.folderhtml[k];
				for(var l=con; l<curlen; l++){
 					var bits = clon[l].split("$$$",7);
          				content += clusterer.paren.sidebarfn(bits[0],bits[1],bits[2],bits[3],bits[4],bits[5],bits[6]);  
					}
				}
				
			clusterer.folderhtmlast[k] = curlen;
			dest.innerHTML  = content;
		 	dest.style.height = (2+(clusterer.folderhtml[k].length*clusterer.rowHeight))+"px";
		 	if(clusterer.paren.forcefoldersopen){
	 			dest.style.display = "block";
	 			}
			update_side = true;
			count = curlen;
			}
		else {
			alert("target folder not found "+destid);
			}
		}

	}
	

   	
	if(update_side && count>0){
		var mifinish = new Date();
		var sec = ((mifinish - clusterer.miStart)/1000+" seconds");
		clusterer.paren.mb.showMess("Loaded "+count+"  GeoXML elements in "+sec,5000);
		
		}

    if (typeof resizeKML != "undefined"){
		resizeKML();
		} 

    var bounds, sw, ne, dx, dy;
    var newZoomLevel = clusterer.map.getZoom();
    if ( newZoomLevel != clusterer.currentZoomLevel )
	{
	// When the zoom level changes, we have to remove all the clusters.
	for ( i = 0; i < clusterer.clusters.length; ++i )
	    if ( clusterer.clusters[i] != null )
		{
		clusterer.ClearCluster( clusterer.clusters[i] );
		clusterer.clusters[i] = null;
		}
	clusterer.clusters.length = 0;
	clusterer.currentZoomLevel = newZoomLevel;
	}

    // Get the current bounds of the visible area.
    bounds = clusterer.map.getBounds();

    // Expand the bounds a little, so things look smoother when scrolling
    // by small amounts.
    var sw = bounds.getSouthWest();
    var ne = bounds.getNorthEast();
    var dx = ne.lng() - sw.lng();
    var dy = ne.lat() - sw.lat();
    if ( dx < 300 && dy < 150 ){
	dx *= 0.05;
	dy *= 0.05;
	bounds = new GLatLngBounds(
	new GLatLng( sw.lat() - dy, sw.lng() - dx ),
	new GLatLng( ne.lat() + dy, ne.lng() + dx ) );
	}

    // Partition the markers into visible and non-visible lists.
    var visibleMarkers = [];
    var nonvisibleMarkers = [];
    var viscount = 0;
    for ( i = 0; i < clusterer.markers.length; ++i ) {
	marker = clusterer.markers[i];
	vis = false;
	if (marker != null ){
		var mid = clusterer.paren.myvar+"sb"+i;	
	    	if(typeof marker.getBounds =="undefined"){
			if (bounds.contains(marker.getPoint()) ) {
				vis = true; 
				if($(mid)){ 
					$(mid).className = "inView";
					}
				viscount++;
				}
			else {	if($(mid)){ 
					$(mid).className = "outView";
					}
				}
			}
	     	else {
			var b = marker.getBounds();
			 if($(mid)){ 
	            		if(bounds.intersects(b)){
					$(mid).className = "inView";
					}
				else {  $(mid).className = "outView"; }
					}
			vis = true;
			}
      		if(vis)visibleMarkers.push(i);
	            else nonvisibleMarkers.push(i);
	      
		}
	}

    // Take down the non-visible markers.
    for ( i = 0; i < nonvisibleMarkers.length; ++i )
	{
	marker = clusterer.markers[nonvisibleMarkers[i]];
	if (marker.onMap){
	    clusterer.map.removeOverlay(marker);
	    marker.onMap = false;
	    }
	}

    // Take down the non-visible clusters.
    for ( i = 0; i < clusterer.clusters.length; ++i ) {
	cluster = clusterer.clusters[i];
	if(cluster != null && cluster.marker) {
		var vis = false;
			if(typeof cluster.marker.getBounds =="undefined"){
				if (bounds.contains(cluster.marker.getPoint()) ) vis = true; 
				}
	     	  	else {
		 		//if(bounds.intersects(cluster.marker.getBounds())) vis=true;
				vis = true;
		 		}
		if (!vis && cluster.onMap) {
	    		clusterer.map.removeOverlay(cluster.marker);
	   		    cluster.onMap = false;
	    		}
		}
	}

    // Clustering!  This is some complicated stuff.  We have three goals
    // here.  One, limit the number of markers & clusters displayed, so the
    // maps code doesnt slow to a crawl.  Two, when possible keep existing
    // clusters instead of replacing them with new ones, so that the app pans
    // better.  And three, of course, be CPU and memory efficient.

    if (viscount > clusterer.maxVisibleMarkers)
	{
	// Add to the list of clusters by splitting up the current bounds
	// into a grid.
	if(!update_side){
		clusterer.paren.mb.showMess("Clustering on "+viscount+"  GeoXML elements");
		}

	var latRange = bounds.getNorthEast().lat() - bounds.getSouthWest().lat();
	var latInc = latRange / clusterer.gridSize;
	var lngInc = latInc / Math.cos( ( bounds.getNorthEast().lat() + bounds.getSouthWest().lat() ) / 2.0 * Math.PI / 180.0 );
	for ( var lat = bounds.getSouthWest().lat(); lat <= bounds.getNorthEast().lat(); lat += latInc )
	    for ( var lng = bounds.getSouthWest().lng(); lng <= bounds.getNorthEast().lng(); lng += lngInc )
		{
		cluster = new Object();
		cluster.clusterer = clusterer;
		cluster.bounds = new GLatLngBounds( new GLatLng( lat, lng ), new GLatLng( lat + latInc, lng + lngInc ) );
		cluster.markers = [];
		cluster.markerCount = 0;
		cluster.onMap = false;
		cluster.marker = null;
		clusterer.clusters.push( cluster );
		}

	// Put all the unclustered visible markers into a cluster - the first
	// one it fits in, which favors pre-existing clusters.
	for ( i = 0; i < visibleMarkers.length; ++i ) {
	    marker = clusterer.markers[visibleMarkers[i]];
	    if (marker != null && !marker.inCluster ) {
		for ( j = 0; j < clusterer.clusters.length; ++j ) {
		    cluster = clusterer.clusters[j];
		    if(cluster != null){
		        var vis = false;
		        if(typeof marker.getBounds =="undefined"){ 
				    if (cluster.bounds.contains(marker.getPoint())) { vis = true;   }
				    }
		        if (vis){
				marker.inCluster = true;
			        clusterer.clusters[j].markers.push(marker);
			        ++clusterer.clusters[j].markerCount;
			        }
			    }
		    }   
		}
	    }

	// Get rid of any clusters containing only a few markers.
	for ( i = 0; i < clusterer.clusters.length; ++i ) {
	    if ( clusterer.clusters[i] != null && clusterer.clusters[i].markerCount < clusterer.minMarkersPerCluster )
		{
		clusterer.ClearCluster( clusterer.clusters[i] );
		clusterer.clusters[i] = null;
		}
	     }

	// Shrink the clusters list.
	for ( i = clusterer.clusters.length - 1; i >= 0; --i )
	    if ( clusterer.clusters[i] != null )
		break;
	    else
		--clusterer.clusters.length;
  
	// Ok, we have our clusters.  Go through the markers in each
	// cluster and remove them from the map if they are currently up.
	for ( i = 0; i < clusterer.clusters.length; ++i )
	    {
	    cluster = clusterer.clusters[i];
	    if ( cluster != null )
		{
		for ( j = 0; j < cluster.markers.length; ++j )
		    {
		    marker = cluster.markers[j];
		    if ( marker != null && marker.onMap )
			{
			clusterer.map.removeOverlay( marker );
			marker.onMap = false;
			}
		    }
		}
	    }

	// Now make cluster-markers for any clusters that need one.
	for ( i = 0; i < clusterer.clusters.length; ++i )
	    {
	    cluster = clusterer.clusters[i];
	    if ( cluster != null && cluster.marker == null )
		{
		// Figure out the average coordinates of the markers in this
		// cluster.
		var xTotal = 0.0, yTotal = 0.0;
		for ( j = 0; j < cluster.markers.length; ++j )
		    {
		    marker = cluster.markers[j];
		    if ( marker != null )
			{
			xTotal += ( + marker.getPoint().lng() );
			yTotal += ( + marker.getPoint().lat() );
			}
		    }
		var location = new GLatLng( yTotal / cluster.markerCount, xTotal / cluster.markerCount );
		marker = new GMarker( location, { icon: clusterer.icon } );
		cluster.marker = marker;
		GEvent.addListener( marker, 'click', Clusterer.MakeCaller( Clusterer.PopUp, cluster ) );
		}
	    }

	}

    if(!update_side && viscount){
//		clusterer.paren.mb.showMess("Showing "+viscount+"  GeoXML elements",500);
		}

    // Display the visible markers not already up and not in clusters.
    for ( i = 0; i < visibleMarkers.length; ++i )
	{
	marker = clusterer.markers[visibleMarkers[i]];
	if ( marker != null && ! marker.onMap && ! marker.inCluster) {
	    if (marker.addedToMap != null ) marker.addedToMap();
	    if(marker.hidden){
		if(true){ clusterer.map.addOverlay(marker); marker.hide(); }
		}
	    else { clusterer.map.addOverlay(marker);}
	    marker.onMap = true;
	    }
	}

    // Display the visible clusters not already up.
    for ( i = 0; i < clusterer.clusters.length; ++i ) {
	cluster = clusterer.clusters[i]; 
	if(cluster != null && cluster.marker) {
	    var vis = false;
	    if(typeof marker.getPoint !="undefined"){if (bounds.contains( cluster.marker.getPoint())) vis = true; }
	        else {if(bounds.intersects(cluster.marker.getBounds())) vis=true;}
	    if (!cluster.onMap && vis ) {
	        clusterer.map.addOverlay( cluster.marker );
	        cluster.onMap = true;
	        }
	    }
	}

 
    // In case a cluster is currently popped-up, re-pop to get any new
    // markers into the infobox.
   // if(addHilite)addHilite();
    Clusterer.RePop( clusterer );
    };


Clusterer.PopUp = function ( cluster )
    {
    var clusterer = cluster.clusterer;
    var html = '<table width="300">';
    var n = 0;
    for ( var i = 0; i < cluster.markers.length; ++i )
	{
	var marker = cluster.markers[i];
	if ( marker != null )
	    {
	    ++n;
	    html += '<tr><td>';
	    if (typeof marker.getIcon!="undefined" &&  marker.getIcon().smallImage != null )
		html += '<img src="' + marker.getIcon().smallImage + '">';
	    else
		html += '<img src="' + marker.getIcon().image + '" width="' + ( marker.getIcon().iconSize.width / 2 ) + '" height="' + ( marker.getIcon().iconSize.height / 2 ) + '">';
	    html += '</td><td>' + marker.title + '</td></tr>';
	    if ( n == clusterer.maxLinesPerInfoBox - 1 && cluster.markerCount > clusterer.maxLinesPerInfoBox  )
		{
		html += '<tr><td colspan="2">...and ' + ( cluster.markerCount - n ) + ' more</td></tr>';
		break;
		}
	    }
	}
    html += '</table>';
    clusterer.map.closeInfoWindow();
    cluster.marker.openInfoWindowHtml( html );
    clusterer.poppedUpCluster = cluster;
    };


Clusterer.RePop = function ( clusterer )
    {
    if ( clusterer.poppedUpCluster != null )
	Clusterer.PopUp( clusterer.poppedUpCluster );
    };


Clusterer.PopDown = function ( clusterer )
    {
    clusterer.poppedUpCluster = null;
    };


Clusterer.prototype.ClearCluster = function ( cluster )
    {
    var i, marker;

    for ( i = 0; i < cluster.markers.length; ++i )
	if ( cluster.markers[i] != null )
	    {
	    cluster.markers[i].inCluster = false;
	    cluster.markers[i] = null;
	    }
    cluster.markers.length = 0;
    cluster.markerCount = 0;
    if ( cluster == this.poppedUpCluster )
	this.map.closeInfoWindow();
    if ( cluster.onMap )
	{
	this.map.removeOverlay( cluster.marker );
	cluster.onMap = false;
	}
    };


// This returns a function closure that calls the given routine with the
// specified arg.
Clusterer.MakeCaller = function ( func, arg )
    {
    return function () { func( arg ); };
    };


// Augment GMarker so it handles markers that have been created but
// not yet addOverlayed.

GMarker.prototype.setMap = function ( map )
    {
    this.map = map;
    };

GMarker.prototype.addedToMap = function ()
    {
    this.map = null;
    };

GMarker.prototype.origOpenInfoWindow = GMarker.prototype.openInfoWindow;
GMarker.prototype.openInfoWindow = function ( node, opts )
    {
    if ( this.map != null )
	return this.map.openInfoWindow( this.getPoint(), node, opts );
    else
	return this.origOpenInfoWindow( node, opts );
    };

GMarker.prototype.origOpenInfoWindowHtml = GMarker.prototype.openInfoWindowHtml;
GMarker.prototype.openInfoWindowHtml = function ( html, opts )
    {
    if ( this.map != null )
	return this.map.openInfoWindowHtml( this.getPoint(), html, opts );
    else
	return this.origOpenInfoWindowHtml( html, opts );
    };

GMarker.prototype.origOpenInfoWindowTabs = GMarker.prototype.openInfoWindowTabs;
GMarker.prototype.openInfoWindowTabs = function ( tabNodes, opts )
    {
    if ( this.map != null )
	return this.map.openInfoWindowTabs( this.getPoint(), tabNodes, opts );
    else
	return this.origOpenInfoWindowTabs( tabNodes, opts );
    };

GMarker.prototype.origOpenInfoWindowTabsHtml = GMarker.prototype.openInfoWindowTabsHtml;
GMarker.prototype.openInfoWindowTabsHtml = function ( tabHtmls, opts )
    {
    if ( this.map != null )
	return this.map.openInfoWindowTabsHtml( this.getPoint(), tabHtmls, opts );
    else
	return this.origOpenInfoWindowTabsHtml( tabHtmls, opts );
    };

GMarker.prototype.origShowMapBlowup = GMarker.prototype.showMapBlowup;
GMarker.prototype.showMapBlowup = function ( opts )
    {
    if ( this.map != null )
	return this.map.showMapBlowup( this.getPoint(), opts );
    else
	return this.origShowMapBlowup( opts );
    }; 



MessageBox = function(map,paren,myvar,mb){
	this.map = map;
	this.paren = paren;
	this.myvar = paren.myvar+"."+myvar;
	this.eraseMess = null;
	this.centerMe = null;
	this.mb = null;
	if(mb)this.mb = mb;
	this.id = this.myvar + "_message"
	};

MessageBox.prototype.hideMess = function(){
  	this.mb.style.visiblity ="hidden"; 
	this.mb.style.left = "-1200px";
	this.mb.style.top = "-1200px";
	};

MessageBox.prototype.centerThis = function(){
	var c = this.map.getObjCenter();
	if(!this.mb)this.mb = $(this.id);
	if(this.centerMe){ clearTimeout(this.centerMe);}
	if(this.mb){
		var nw = this.mb.clientWidth;
		if(nw > this.map.getSize().width){
			var nw = parseInt(2*this.map.getSize().width/3);
			this.mb.style.width = nw +"px";
			this.centerMe = setTimeout(this.myvar+".centerThis()",5);
			return; 
			}
		this.mb.style.left = (c.x - (nw/2)) +"px";
		this.mb.style.top = (c.y - 20 - (this.mb.clientHeight/2))+ "px";
		}
	else {
		this.centerMe = setTimeout(this.myvar+".centerThis()",10);
		}
	};

MessageBox.prototype.showMess = function (val,temp){

	val = unescape(val);
	if(this.eraseMess){ clearTimeout(this.eraseMess);
} 
	if(!this.mb)this.mb = $(this.id);

	var c = this.map.getObjCenter();
	if(this.mb){

		this.mb.innerHTML = "<span>"+val+"</span>";

	    	if(temp){

			this.eraseMess = setTimeout(this.myvar+".hideMess();",temp);

			}

		this.mb.style.left = "-1200px";
		this.mb.style.top = "-1200px";
		this.mb.style.width = "";
		this.mb.style.height = "";
		this.centerMe = setTimeout(this.myvar+".centerThis()",5);
		this.mb.style.visibility = "visible"; 

		}

	else {  
		var d = document.createElement("div");
		d.innerHTML = val;
		d.id = this.myvar + "_message";
		d.style.position = "absolute";
		d.style.backgroundColor = this.style.backgroundColor || "silver";
		d.style.opacity = this.style.opacity || 0.80;
		d.style.filter = "alpha(opacity="+parseInt(d.style.opacity*100)+")";
		d.style.color = this.style.color || "black";
		d.style.padding = this.style.padding || "6px";
 		d.style.borderWidth = this.style.borderWidth || "3px";
		d.style.borderColor = this.style.borderColor || "";
		d.style.backgroundImage = this.style.backgroundImage || "";
		d.style.borderStyle = this.style.borderStyle || "outset";
		d.style.visibility = "visible";
		d.style.left = "-1200px";
		d.style.top = "-1200px";
		this.centerMe = setTimeout(this.myvar+".centerThis()",5);
	
		d.style.zIndex = 1000;
		document.body.appendChild(d);
		}

	}; 


GMap2.prototype.getObjCenter = function(){
	var container = obj = this.getContainer();
	var y = x = 0;
	if (obj.offsetParent) {
		x = obj.offsetLeft;
		y = obj.offsetTop;
		while (obj = obj.offsetParent) {
			x += obj.offsetLeft;
			y += obj.offsetTop;
		}
 	return new GPoint(x+parseInt(this.getSize().width)/2,y+parseInt(this.getSize().height)/2);
	}
	
	};



GeoXml.prototype.loadJSONUrl = function (url, title, latlon, desc, idx) {
  var that = this;
  GDownloadUrl(url, function(doc) {
    	that.parseJSON(doc,title, latlon, desc, idx);
  	});
}

GeoXml.prototype.loadXMLUrl = function (url, title, latlon, desc, idx) {
var that = this;	
  GDownloadUrl(url, function(doc) {
    	that.processing(GXml.parse(doc),title, latlon, desc, idx)
  	});
}

GeoXml.prototype.upgradeLayer = function(n) {
	var mt = this.map.getMapTypes();
	var found = false;
	for(var i=0;i<mt.length;i++){
		if(mt[i] == this.baseLayers[n]){
			found = true;
			this.map.removeMapType(this.baseLayers[n])
			}
		}
	if(!found)this.map.addMapType(this.baseLayers[n]);
	};


GeoXml.prototype.makeWMSTileLayer = function(getmapstring, on, title, opac) {
	var that = this;
	getmapstring = getmapstring.replace("&amp;","&");
 	var args = getmapstring.split("?");
	baseurl = args[0]+"?";
	var version = "1.1.0";
	var format = "image/png";
	var styles = "";
	var layers = "";
	var bbox = "";
	var queryable = false;
	var opacity = 1.0;
	if(typeof opac!="undefined")opacity = opac;
	var bbox = "-180,-90,180,90";
	var pairs = args[1].split("&");
	var sld ="";
	var gmcrs = "";
	for(var i=0;i < pairs.length; i++){
		var dstr = pairs[i];
		var duo = pairs[i].split("=");
		var dl = duo[0].toLowerCase();
		switch(dl) {
			case "version" : version = duo[1];break;
			case "bbox": bbox = duo[1]; break;
			case "width":
			case "height":break;
			case "service":break;
			case "styles": styles = duo[1]; break;
			case "layers": layers = duo[1]; break;
			case "format": format = duo[1]; break;
			case "opacity":opacity = parseFloat(duo[1]); break;
			case "crs":
			case "srs":epsg = duo[1]; break;
			case "gmcrs":gmcrs = duo[1];break;
			case "queryable":queryable = duo[1];break;
			default : if(duo[0])baseurl += "&"+pairs[i]; break;
			}
		}

	if(gmcrs)epsg = gmcrs;
	var bbn = bbox.split(",");
	var bb = {"w":parseFloat(bbn[0]),"s":parseFloat(bbn[1]),"e":parseFloat(bbn[2]),"n":parseFloat(bbn[3])};
	
	var lon = (bb.n - bb.s);

	var z = 0; 

	var ex = 180;

 	while(ex >= lon){

		ex = ex/2;

		z++;

		}

	z--;

	if(z<1){ z=1; }

 
	var cr0 = new GCopyright(1, new GLatLngBounds(new GLatLng(bb.s,bb.w),new GLatLng(bb.n,bb.e)),0,"WMS");
    	var cc0 = new GCopyrightCollection("Base Map from:");
     	cc0.addCopyright(cr0);
 	var twms = new GTileLayer(cc0,z,19);
	twms.s = bb.s; twms.n = bb.n; twms.e = bb.e; twms.w = bb.w;
	twms.myBaseURL = baseurl;
	twms.publishdirectory = this.publishdirectory;
	twms.getTileUrl = function(a,b,c) {

		if (typeof(this.myStyles)=="undefined") {

			this.myStyles=""; 

			}

		var lULP = new GPoint(a.x*256,(a.y+1)*256);

		var lLRP = new GPoint((a.x+1)*256,a.y*256);

		var lUL = G_NORMAL_MAP.getProjection().fromPixelToLatLng(lULP,b,c);

		var lLR = G_NORMAL_MAP.getProjection().fromPixelToLatLng(lLRP,b,c);

		var west = lUL.x;

		var east = lLR.x;

		var north = lUL.y;

		var south = lLR.y;

		var ge = east;

		var gw = west;

		var gs = south;

		var gn = north;

		if(gn < gs){ gs = gn; gn = south; }

		if(this.epsg != "EPSG:4326") {

			west = GeoXml.merc2Lon(west);

			north = GeoXml.merc2Lat(north);

			east = GeoXml.merc2Lon(east);

			south = GeoXml.merc2Lat(south);

			}

		var w = Math.abs(east - west);

		var h = Math.abs(north - south);

		var s = h/w;

		h = Math.round((256.0 * s) + 0.5);

		w = 256;

		var sud = south; 

		if(north < south){

			south = north; north = sud; 

			}

 

		if(typeof this.s != "undefined"){

			if(gs>(this.n) || ge < (this.w) || gn < (this.s) || gw > (this.e)  ){

				var retstr = this.publishdirectory +"images/white.gif";

		 		if(this.myFormat=="image/jpeg" || this.myFormat == "JPEG" || this.myFormat == "JPG"){ return retstr; }

					return this.publishdirectory +"images/blank.gif";

					}

			}

    		var lBbox=west+","+south+","+east+","+north;

		var lSRS="EPSG:41001";

		if(typeof this.epsg != "undefined"){

    			lSRS=this.epsg;

			}

		

		var lURL=this.myBaseURL;

		if(typeof this.myVersion == "undefined"){ this.myVersion = "1.1.1"; }

		var ver = parseFloat(this.myVersion);

		lURL+="&SERVICE=WMS";

		if(this.myVersion !="1.0.0"){

			lURL+="&REQUEST=GetMap";

			}

		else {

			lURL+="&REQUEST=Map";

			}

		lURL+="&VERSION="+this.myVersion;

		if(this.myLayers) {

		lURL+="&LAYERS="+this.myLayers;

		lURL+="&STYLES="+this.myStyles; 

		}

		if(this.mySLD){

			lURL+="&SLD="+this.mySLD; 

			}

  		lURL+="&FORMAT="+this.myFormat;

		lURL+="&BGCOLOR=0xFFFFFF";

		lURL+="&TRANSPARENT=TRUE";

		if(this.myVersion == "1.1.1" || ver<1.3 ){

			lURL += "&SRS=" + lSRS;

			}

		else {

			lURL += "&CRS=" + lSRS;

			}

		lURL+="&BBOX="+lBbox;

		lURL+="&WIDTH="+w;

		lURL+="&HEIGHT="+h;

		this.requestCount++;

		return lURL;

		}

	twms.myFormat = format;
	twms.myVersion = version;
	twms.myExtents = bbox;
	twms.queryable = queryable;
	twms.opacity = opacity;
	twms.getOpacity = function() { return this.opacity; };
	if(sld){
		twms.mySLD = sld;
		}
	else {
		twms.myLayers = layers;
		twms.myStyles = styles;
		}
	twms.epsg = epsg;
	var ol = new GTileLayerOverlay(twms);
	ol.bounds = new GLatLngBounds();
	ol.bounds.extend(new GLatLng(bb.n,bb.e));ol.bounds.extend(new GLatLng(bb.s,bb.w));
 

 	var twms2 = new GTileLayer(cc0,z,19);
	twms2.s = bb.s; twms2.n = bb.n; twms2.e = bb.e; twms2.w = bb.w;
	twms2.myBaseURL = baseurl;
	twms2.publishdirectory = this.publishdirectory;
	twms2.getTileUrl = twms.getTileUrl;
	twms2.myFormat = "image/jpeg";
	twms2.myVersion = version;
	twms2.myExtents = bbox;
	twms2.queryable = queryable;
	twms2.opacity = 1.0;
	twms2.title = title;
	twms2.getOpacity = function() { return this.opacity; };
	if(sld){
		twms2.mySLD = sld;
		}
	else {
		twms2.myLayers = layers;
		twms2.myStyles = styles;
		}
	twms2.epsg = epsg;
	this.wmscount++;
	var layer = [twms2];
	var cmap = new GMapType(layer, G_HYBRID_MAP.getProjection(), "WMS ("+title+")", G_HYBRID_MAP);	
	that.baseLayers.push(cmap);
	return ol;
	}

GeoXml.SEMI_MAJOR_AXIS = 6378137.0;

GeoXml.ECCENTRICITY = 0.0818191913108718138;

GeoXml.DEG2RAD = 180.0/(Math.PI);
GeoXml.prototype.merc2Lon = function(lon) {

	return (lon*GeoXml.DEG2RAD)*GeoXml.SEMI_MAJOR_AXIS;

	}

GeoXml.prototype.merc2Lat = function(lat) {

	var rad = lat * DEG2RAD;

	var sinrad = Math.sin(rad);

	return (GeoXml.SEMI_MAJOR_AXIS * Math.log(Math.tan((rad + Math.PI/2) / 2) * Math.pow(((1 - GeoXml.ECCENTRICITY * sinrad) / (1 + GeoXml.ECCENTRICITY * sinrad)), (GeoXml.ECCENTRICITY/2))));

	}


GeoXml.prototype.toggleLabels = function(on) {
	if(!on) {this.removeLabels();
		}
	else { 
	  	this.addLabels();
		}
	}
GeoXml.prototype.addLabels = function() {
	this.labels.onMap = true;
 	this.map.addOverlay(this.labels); 
	}
 
GeoXml.prototype.removeLabels = function() {
	this.labels.onMap = false;
	this.map.removeOverlay(this.labels);
	}
