Select to view content in your preferred language

automatic zoom to layer

1701
5
Jump to solution
06-27-2012 05:00 PM
Egan_LouisFajardo
Occasional Contributor
I'm trying to zoom to layer automatically as soon as the map finishes loading using the guide here. To be clear, I'm only loading one layer and just want to zoom in on one of the layers provided by the service.

js script
dojo.require("dijit.layout.BorderContainer"); dojo.require("dijit.layout.ContentPane"); dojo.require("esri.map"); dojo.require("esri.renderer"); dojo.require("esri.graphic"); dojo.require("esri.dijit.Legend"); dojo.require("esri.utils");  dojo.require("dijit.layout.AccordionContainer"); dojo.require("esri.layers.FeatureLayer");  var map; var basemap; var trafficmap; var basemapServiceSource; var featuremapServiceSource; var trafficServiceSource; var renderer; var loading;  function init() {   // for now, these are referring to a single source  basemapServiceSource = "http://co-gis-01/ArcGIS/rest/services/dev_RTIA/rtia/MapServer";   //Add the topographic layer to the map. View the ArcGIS Online site for services http://arcgisonline/home/search.html?t=content&f=typekeywords:service        basemap = new esri.layers.ArcGISDynamicMapServiceLayer(basemapServiceSource);  basemap.setVisibleLayers([0, 1, 2, 4, 5, 6]);   map = new esri.Map("map", {      // remove logo    logo:false   });   //loading image. id  loading = dojo.byId("loadingImg");  dojo.connect(map,"onUpdateStart",showLoading);  dojo.connect(map,"onUpdateEnd",hideLoading);      /*   IDs of Layers for http://co-gis-01/ArcGIS/rest/services/dev_ExIS/exis/MapServer    Site Name (0)    Section ID (1)    AADT some (2)    AADT all (3)    AdministrativeAreas_CongressionalDistricts (4)    AdministrativeAreas_Provinces (5)    AdministrativeAreas_Regions (6)  */  map.addLayer(basemap);   // executes function on onLoad event  dojo.connect(map, 'onLoad', function(theMap) {      dojo.connect(dijit.byId('map'), 'resize', map,map.resize);   zoomToLayer();  }); }   function zoomToLayer() {   var requestHandle = esri.request({   url: "http://co-gis-01/ArcGIS/rest/services/dev_RTIA/rtia/MapServer/2",   content: { f:"json" },   callbackParamName: "callback"  });    requestHandle.then(requestSucceeded, requestFailed); }  function requestSucceeded(response, io) {   var extent = new esri.geometry.Extent(response.extent);  dojo.byId("extent").innerHTML = dojo.toJson(extent.toJson());  map.setExtent(extent); }  function requestFailed() {   alert("requestFailed"); }  function showLoading() {   esri.show(loading);  map.disableMapNavigation();  map.hideZoomSlider(); }  function hideLoading(error) {   esri.hide(loading);  map.enableMapNavigation();  map.showZoomSlider();        }  dojo.addOnLoad(init);


html code
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd"> <html>   <head>     <title>RTIA Map</title>  <meta http-equiv="X-UA-Compatible" content="IE=EmulateIE8" >    <link rel="stylesheet" type="text/css" href="gis.css" />    <script type="text/javascript" src="http://serverapi.arcgisonline.com/jsapi/arcgis/?v=2.8"></script>  <script type="text/javascript" src="rtia_culled.js"></script>      </head>    <body>     <div id="map" class="mainmap">   <img id="loadingImg" src="images/loading.gif" class="loadimage"/>  </div>  <!--div id="legendDiv" style="width:600px; height:600px; border:1px solid #000"-->  </div>   <span id="extent"></span> </body>  </html>


The problem is that the outputs of dojo.toJson(extent.toJson()) show the same extent, which is the full extent, no matter the layer. This particular layer that I would like to zoom to, http://co-gis-01/ArcGIS/rest/services/dev_RTIA/rtia/MapServer/2, is just a filtered version of one of the "full-extent" layers. The filter is defined in the mapservice.

How do I correctly do this function?
0 Kudos
1 Solution

Accepted Solutions
Egan_LouisFajardo
Occasional Contributor
I've managed to solve this problem using this as a reference. Basically, I looped all of the Polyline graphics (that's the only type the application is interested in) included in the FeatureLayer and do a union operation on each of the graphic's extent. Below is my version.

function featureUpdateEnd(error, info) {   var localExtent;  if (featuremap.graphics.length > 0) {     for (var i = 0; i < featuremap.graphics.length; i++) {       if (featuremap.graphics.geometry instanceof esri.geometry.Polyline) {         try {           if (i == 0) {             localExtent = new esri.geometry.Extent(featuremap.graphics.geometry.getExtent().toJson());      }      else {             localExtent = localExtent.union(featuremap.graphics.geometry.getExtent());      }     }     catch (err) {           //alert(String(i) + " is caught: " + err.message);     }    }   }      localExtent = localExtent.expand(1.25);   map.setExtent(localExtent.getExtent());  } }


featuremap is the FeatureLayer. featureUpdateEnd is a function that is called by FeatureLayer's onUpdateEnd event. I did not use onLoad because the event fires even if the FeatureLayer hasn't finished loading yet, giving me a zero-length array to loop on. I had to enclose part of it in a try-catch statement because of an undefined instance is encountered (but this is probably only due to my data).

View solution in original post

0 Kudos
5 Replies
Egan_LouisFajardo
Occasional Contributor
I tried setting only the layer of interest as visible, and then setting the map extent to full extent. Still, only the whole full extent is displayed instead of the extent that can only be shown by the layer.
0 Kudos
Egan_LouisFajardo
Occasional Contributor
I tried exporting the filtered layer and importing it, resulting in a layer whose content is only those originally defined by the definition query. I published the service and was finally able to automatically zoom to the desired extent. The definition query seems to be ignored when getting the full extent of the layer, in other words, the full extent still considers all features, even those that fails the definition query. is there a way to circumvent this in the api?
0 Kudos
Egan_LouisFajardo
Occasional Contributor
I've managed to solve this problem using this as a reference. Basically, I looped all of the Polyline graphics (that's the only type the application is interested in) included in the FeatureLayer and do a union operation on each of the graphic's extent. Below is my version.

function featureUpdateEnd(error, info) {   var localExtent;  if (featuremap.graphics.length > 0) {     for (var i = 0; i < featuremap.graphics.length; i++) {       if (featuremap.graphics.geometry instanceof esri.geometry.Polyline) {         try {           if (i == 0) {             localExtent = new esri.geometry.Extent(featuremap.graphics.geometry.getExtent().toJson());      }      else {             localExtent = localExtent.union(featuremap.graphics.geometry.getExtent());      }     }     catch (err) {           //alert(String(i) + " is caught: " + err.message);     }    }   }      localExtent = localExtent.expand(1.25);   map.setExtent(localExtent.getExtent());  } }


featuremap is the FeatureLayer. featureUpdateEnd is a function that is called by FeatureLayer's onUpdateEnd event. I did not use onLoad because the event fires even if the FeatureLayer hasn't finished loading yet, giving me a zero-length array to loop on. I had to enclose part of it in a try-catch statement because of an undefined instance is encountered (but this is probably only due to my data).
0 Kudos
JohnnyPenet
Emerging Contributor
A dynamic map service has a property fullextent, so you do not need to do the calculation on the features. However, it will be available at the onload event of the map service.
So you simple need to handle the onload event of the map service to have the full extent of the map service.
0 Kudos
Egan_LouisFajardo
Occasional Contributor
As explained in my first post, I cannot use the respective fullExtent of the dynamically segmented layer as it references the fullExtent of the the layer it originated from.
0 Kudos