Introduction:

Recently met a special tree, after opening the project timeline, open the layer will be closed before the bug process is that the first open increased urban planning guide block, and then open the above historical layers containing more than one year, land planning guideline, this time to add urban planning guide block layer will be closed.

After locating the code and commenting it line by line, it was found that this code had a problem with closing and displaying layers:

// Handle display hiding for the default year
 var layer = this.defaultLayer;
if (visible) {
    `layer.setVisibleLayers && layer.setVisibleLayers([layerobj.layerindex])`; 
    layer.setVisibility(true);
 } else {
/ / close
   `layer.setVisibleLayers && layer.setVisibleLayers([]); `
    layer.setVisibility(false); 
}
Copy the code

Why does the current layer close all other layers? So let’s start with a couple of concepts.

This article mainly tells the following contents:

  • What are layers
  • How are layers loaded
  • Application of layers: Solve the thematic tree problem

What are layers

Layers are the display mechanism of geographic data sets in Arcgis product suites such as ArcMap, ArcGlobe and ArcScene. A layer refers to a data set and specifies how to draw the data set using symbols and text annotations. When adding a layer to a map, specify its reference data set and set the map symbols and annotation properties.

Each application that contains a map control is assembled through a series of layers. Displays are displayed on the map in a specific order, with bottom-most displays displayed at the top of the map, and first added displays displayed below (similar to the principle of “stack”).

All layers are inherited from the Layer type. Refer to the object model diagram in the downloaded API.

Layer TiledMapServiceLayer | - | -- - | - ArcGISTiledMapServiceLayer | - DynamicLayer | -- - | - DynamicMapServiceLayer | -- -- -- -- -- -- -- -- -- - | - ArcGISDynamicMapServiceLayer | -- -- -- -- -- -- -- -- -- - | - ArcGISImageServiceLayer | -- -- -- -- -- -- -- -- -- - | - GPResultImageLayer | - GraphicsLayer | -- - | - FeatureLayer | - ElementLayerCopy the code

And how does that layer get loaded? It gets loaded from the map service.

Layer loading

What is a map service

Map service is a method to make map accessible through Web by using ArcGIS. We first made the map in ArcMap, and then published the map to ArcGIS Server site. After the success of the local map service issue, we can through the url (XXXX/arcgis/rest/services) to view your map service support operations, containing data map service, and we can also through the url to test the function of the map service.

The address is then requested to use the map service in Web applications, ArcGIS for Desktop, ArcGIS Online, and other client applications

To instantiate a layer object, pass in the url of the layer.

Slice the service

Principle: The slice service has already cut the map by scale, such as the general base map, which is usually loaded by the slice service. When you zoom in on the base map with the mouse, it will load the cut picture according to the current scale. The way to load the cut picture is to request the cut picture through the export interface.

Since the slicing service is already sliced, there is no way to control the display of its layer through setVisibleLayers, only through setVisiblity. When exporting an image, Export exports the entire slice of the current scale.

A published slice is as follows:

Through ArcGISTiledMapServiceLayer to create a new section class instance, and then loaded into the map

var layer = new ArcGISTiledMapServiceLayer(layerobj.url);
map.addLayer(layer);
Copy the code

Controls the display of slice layers

  // Set layer show/Hide
    layerVisibleRefreshByName: function(obj) {
      if (obj == null) return;
      var serviceid = obj.serviceid;
      var layername = obj.layername;
      var layervisible = obj.layervisible;
      // If serviceID does not exist, o&M assigns a label name
      if (serviceid == this.guid || serviceid == this.label) {
        this.setVisibility(layervisible); }}Copy the code

Of course, if slice service launch, checked can use dynamic service load, then slice services can also be loaded by ArcGISDynamicMapServiceLayer.

Dynamic service

How it works: Dynamic services are real-time generated images, not pre-stored images on the server. When the user requests the map service to the server, the server invokes the underlying service according to the received parameters. The underlying service calculates the parameters and generates pixels in real time. These pixels form pictures and are returned to the server, which is then transmitted to the client.

The information for a dynamic service is as follows:

Through ArcGISDynamicMapServiceLayer to create a new dynamic class instance, and then loaded into the map

var layer = new ArcGISDynamicMapServiceLayer(layerobj.url);
map.addLayer(layer);
Copy the code

When we execute setVisibleLayers, we will export the image of the corresponding sub-layer and then load it into the map. SetVisibleLayers (-1) closes all sublayers.

Control the display of dynamic pictures

 @param {Number} layerid sub layerid * @param {Boolean} layervisible */
changeLayerVisible: function (dlayer, layerid, layervisible) {
    if (dlayer == null) return;
    if (layerid < 0) return;
    var arrc = dlayer.visibleLayers;
    arrc = this.dealWithLayerInfos(arrc, dlayer.layerInfos);
    if (arrc == null || (arrc.length == 1 && arrc[0] = =- 1)) {
        arrc = [];
    }
    if (layervisible) {
        if (!this.checkLayerId(arrc, layerid)) { arrc.push(layerid); }}else {
        if (this.checkLayerId(arrc, layerid)) {
           arrc = this.removeLayerId(arrc, layerid); }}this.setVisibleLayers(arrc, true);
},
checkLayerId: function (arrc, layerid) {
    if (arrc == null) return false;
        for (var i = 0; i < arrc.length; i++) {
            if (arrc[i] == layerid) {
                return true; }}return false;
},
Copy the code

Return to the scene

When it comes to the historical timeline, the first step is to understand the node information in the topic tree.

The project tree

A topic tree is made up of topics, each leaf node is a topic, so what is a topic?

Topics: Topic is also a layer service. Each topic layer is instantiated by dynamic service or slice service and added to the map during system initialization. Each topic layer has a layer group layers, which is the collection of sublayers displayed by default under the topic service. If the sublayer is loaded via dynamic services, then close the corresponding sublayer. If it is a sliced service, the entire service object is closed.

Implementation of the history timeline

The logic of the historical timeline goes something like this

  1. Select a Layer that is either a service Layer or a sublayer.
  2. The timeline panel opens when it encounters a timeline identifier, initially opening the default year. These layer addresses are obtained by reading the configuration.
{"type": "map map ", "layers": [{"label": "2019"," LayerType ": "dynamic", "LayerName ":" map map ", "LayerIndex ":" map map ", "layerIndex ": 4, "serviceName": "Control Planning Guidelines ", "serviceUid": "", "defaultLayer": true," URL ": "xxxxx/arcgis/rest/services/%E6%8E%A7%E5%88%B6%E6%80%A7%E8%A7%84%E5%88%92%E5%AF%BC%E5%88%99/MapServer/4" }, { "label": "2018", "LayerType ":" Dynamic ", "LayerName ":" Planning guidance block ", "LayerIndex ": 4, "serviceName":" Controlling Planning Guidance ", "serviceUid": "", "defaultLayer": false, "url": "xxxxx/arcgis/rest/services/%E6%8E%A7%E5%88%B6%E6%80%A7%E8%A7%84%E5%88%92%E5%AF%BC%E5%88%992018/MapServer" }, {"label": "2017", "LayerType ": "dynamic"," LayerName ": "Planning guide plot "," LayerIndex ": 4, "serviceName": "Control Planning Guidelines 2017", "defaultLayer": false," URL ": "xxxxx/arcgis/rest/services/%E6%8E%A7%E5%88%B6%E6%80%A7%E8%A7%84%E5%88%92%E5%AF%BC%E5%88%992017/MapServer" }, {"label": "2016", "LayerType ": "dynamic"," LayerName ": "Planning Guide Plot 2016"," LayerIndex ": 4, "serviceName": "Control Planning Guidelines 2016", "defaultLayer": false," URL ": "xxxxx/arcgis/rest/services/%E6%8E%A7%E5%88%B6%E6%80%A7%E8%A7%84%E5%88%92%E5%AF%BC%E5%88%992016/MapServer" } ] },Copy the code
  1. Key switch code
 @param {Object} Layerobj Layer information * @param {Boolean} Visible */
    _changeLayerVisible: function(layerobj, visible) {
      this.map = window._map; // Get the map

      if(! layerobj) {return;
      }
      if(layerobj.label ! =this.clashHisdata.label) {
        // Isolate the year that conflicts with the default year and put the default year in else
        var layerId =
          this.defaultLayer.label + '_historydataservice' + layerobj.label; // For adding multiple layers at the same time multiple layers will appear
        topic.publish('history-layerIds', layerId); // The history layer ID is transferred to the topic tree. The topic book manages the layer switch
        this.historyLayerIds.push(layerId);
        switch (layerobj.layertype) {
          case 'tiled':
            if (visible) {
              var layer = this.map.getLayer(layerId);
              if (layer) {
                this.map.removeLayer(layer);
                layer = new ArcGISTiledMapServiceLayer(layerobj.url);
                this._currentLayer = layer;
                layer.id = layerId;
                this.map.addLayer(layer);
                this.map.reorderLayer(layer, 1);
                layer.setVisibility(true);
              } else {
                layer = new ArcGISTiledMapServiceLayer(layerobj.url);
                this._currentLayer = layer;
                layer.id = layerId;
                this.map.addLayer(layer);
                this.map.reorderLayer(layer, 1);
              }
              this._currentLayer.setOpacity(this._opacity);
            } else {
              var layer = this.map.getLayer(layerId);
              if (layer) {
                //this.map.removeLayer(layer);
                layer.setVisibility(false); }}break;
          case 'dynamic':
            if (visible) {
              var layer = this.map.getLayer(layerId);
              if (layer) {
                if (layer.url == layerobj.url) {
                  layer.setVisibleLayers([layerobj.layerindex]);
                  this._currentLayer = layer;
                  layer.setVisibility(true);
                } else {
                  this.map.removeLayer(layer);
                  layer = new ArcGISDynamicMapServiceLayer(layerobj.url);
                  this._currentLayer = layer;
                  layer.id = layerId;
                  this.map.addLayer(layer);
                  this.map.reorderLayer(layer, 1);
                  layer.setVisibleLayers([layerobj.layerindex]);
                  layer.setVisibility(true); }}else {
                var dlayer = new ArcGISDynamicMapServiceLayer(layerobj.url);
                this._currentLayer = dlayer;
                dlayer.id = layerId;
                this.map.addLayer(dlayer);
                this.map.reorderLayer(dlayer, 1);
                dlayer.setVisibleLayers([layerobj.layerindex]);
              }
              this._currentLayer.setOpacity(1);
              setTimeout(
                lang.hitch(this.function() {
                  this._currentLayer.setOpacity(this._opacity);
                }),
                200
              );
            } else {
              var layer = this.map.getLayer(layerId);
              if (layer) {
                layer.setVisibleLayers([]);
                layer.setVisibility(false); }}break; }}else {
        Var layer = this.defaultLayer; if (visible) { layer.setVisibleLayers && layer.setVisibleLayers([layerobj.layerindex])`; 
            layer.setVisibility(true);
         } else {
        / / close
           layer.setVisibleLayers && layer.setVisibleLayers([]);
            layer.setVisibility(false); }}},Copy the code

Find out why the layer was turned off

Layer: A Layer service contains many sublayers

This is what happens when the history panel is initialized. With the timeline open, serviceId is used to retrieve the current Layer object added to the map. From the project information, get the layer in the current project, the sublayer displayed by default. Then use its serviceUid to retrieve the parent layer loaded into the map (thematic).

Layer off by setVisibility(isVisible) and setVisibleLayers(IDS, doNotRefresh?). Combination control. The layer generated by the slice service only has the setVisibility attribute, while the layer generated by the dynamic service controls the display of the layer by the combination of the two. As shown in the figure, setVisibility controls the display of the entire layer, while setVisibleLayers controls the sublayers within the layer with more fine-grained control.

Looking at the previous code implementation, the breakpoint shows that the sublayer of the default layer layer contains the Increment Guide block layer, so when opening one of the sublayers of the default layer, This line of layer.setVisiblelayers ([Layerobj.layerIndex]) only assigns the index ID of the current sublayer, causing all previous sublayers to be closed.

// Handle display hiding for the default year
 var layer = this.defaultLayer;
if (visible) {
    `layer.setVisibleLayers && layer.setVisibleLayers([layerobj.layerindex])`; 
    layer.setVisibility(true);
 } else {
/ / close
   `layer.setVisibleLayers && layer.setVisibleLayers([]); `
    layer.setVisibility(false); 
}
Copy the code

In this case, only need to add the corresponding layer service type judgment logic, cache the previous can be done.

   // Handle display hiding for the default year
 var layer = this.defaultLayer;
 ` ``var visibleLayers = []; // Visible layer visibleLayers = visibleLayers. Concat (layer.visibleLayers); `` `
 if (visible) {
    / / open
    var index = visibleLayers.indexOf(layerobj.layerindex);
    if (index === - 1&& layerobj.layertype ! = ='tiled') {
        visibleLayers.push(layerobj.layerindex); // Add a new layer index, otherwise passing -1 will close all layers
        layer.setVisibleLayers && layer.setVisibleLayers(visibleLayers); // Note the difference between tiled and dynamic
    }
    layer.setVisibility(true);
 } else {
    / / to meet:
    // 1. Need to solve the problem that the Layer behind the timeline is overwritten by the default Layer.
    // 2. Do not close the entire Layer, otherwise it will affect the display of other layers belonging to the same Layer service instance.
    // 3. Currently, you can only move the following layers to z-index, but use the layers that fit the timeline as the base image.
    'if (layerobj. layerType === 'tiled') {// Determine whether to slice layer.setVisibility(false); } else {var index = Visiblelayers.indexof (layerobj.layerIndex); if (index > -1) { visibleLayers.splice(index, 1); } layer.setVisibleLayers && layer.setVisibleLayers(visibleLayers); } `}}Copy the code

Of course, with the default layer show/hide above, you can pass the layer object directly to the topic tree to unify the processing switch so that you don’t have to write the judgment logic.

(Full text)

conclusion

  • The key is to understand the concept of layers and how they are loaded so that you can quickly locate and solve problems with control layers.
  • The currentarcgis apiIs optimized, only the map scope changes, will go to the map service. Therefore, when switching the timeline, it is important to change its range to better locate errors.

The resources

  • ArcMap uses layers
  • ArcGIS Server publishing dynamic map service