Arcgis加载带?的转发服务

前言:
    项目是用vue做的政府项目,有个需求是加载国土空间信息平台的Arcgis服务,这个申请下来是带ip的http请求,出于安全性考虑,政务网ip需要授权才能请求该服务,但是又不可能每个浏览器端都这么整,所以需要用国土空间信息平台的转发代理,地址形似于 httpsAddr/proxy?realAddr,前面的httpsAddr是政务网访问通用的。
    问题就来了,这个?会被解析成queryParam的分隔符,相当于把realAddr看成了查询参数的键,值为空字符串。这个解析过程在esri-leaflet中也有,另外,Arcgis3.x与Arcgis4.x的解析不太一样。

API差异

除了地址外,手头上的数据还有服务类型,所以首先,需要先将调用加载服务的API对应情况搞清楚。在调试时,我是凭百度和猜词义的,今天又看了下arcgis文档,找到了对应关系的说明:
https://developers.arcgis.com/javascript/latest/functionality-matrix/#esrilayers
3.x中ArcGISDynamicMapServiceLayer、DynamicLayer对应4.x的MapImageLayer
3.x中ArcGISTiledMapServiceLayer对应4.x的TileLayer
还有个ImageLayer没用到,我是直接作为img.src放上去的,效果还可以。
下文均以Arcgis4.x为例

vue加载arcgis模块

vue加载arcgis模块并不是npm直接下载依赖包,而是安装了esri-loader模块,这个模块会动态请求https://js.arcgis.com/version的依赖,有点像require,但是是require远程的。

esriLoader.loadModules('esri/request', 'esri/core/promiseUtils', 'esri/core/urlUtils'])
      .then(([request, promiseUtils, urlUtils]) => { some code })

这种按需加载的方式效率很好,不会占用项目体积。而且,这样请求是有缓存的,也是一个小坑,后面再提。

修改原型

直接上结果,过程也忘了,还挺曲折的。

// methods
{
  addTileLayer(id, url) {
    esriLoader.loadModules(['esri/layers/TileLayer', 'esri/request', 'esri/core/promiseUtils', 'esri/core/urlUtils'])
      .then(
        ([TileLayer, request, w, urlUtils]) => {
          urlUtils.addQueryParameters = function (b, d) {
            if (!d || Object.keys(d).length === 0) {
              return b
            } else {
              for (let pkey in d) {
                if (d.hasOwnProperty(pkey) && d[pkey]) {
                  let value = d[pkey]
                  b += `${pkey}=${value}&`
                }
              }
              return b.substr(0, b.length - 1)
            }
          }
          Object.defineProperty(TileLayer.prototype, "url", {
            set: function (a) {
              this._set('url', a)
            },
            enumerable: !0,
            configurable: !0
          })
          TileLayer.prototype._fetchService = function (a) {
            var b = this
            return w.create(function (d, e) {
              if (b.sourceJSON) {
                d({ data: b.sourceJSON })
              } else {
                if (!b.parsedUrl) throw new c('tile-layer:undefined-url',
                  'layer\'s url is not defined')
                // 修改url部分
                request(url + '?f=json', {
                  responseType: 'json',
                  signal: a
                }).then(d, e)
              }
            }).then(function (c) {
              c.ssl && (b.url = b.url.replace(/^http:/i, 'https:'))
              b.sourceJSON = c.data
              b.read(c.data, { origin: 'service', url: b.parsedUrl })
              if (10.1 === b.version && !P.isHostedAgolService(b.url)) return b._fetchServerVersion(b.url, a).then(function (a) {
                b.read({ currentVersion: a })
              }).catch(function () {
              })
            })
          }
          TileLayer.prototype.getTileUrl = function (level, row, col) {
            return this.url + '/tile/' + level + '/' + row + '/' + col
          }
          this.arcGisLayer = null
          let layer = new TileLayer({
            id: id,
            url: url
          })
          this.map.layers.add(layer)
          this.arcGisLayerList[id] = layer
        })
  },
  addDynamicLayer(id, url) {
    esriLoader.loadModules([
      'esri/layers/MapImageLayer',
      'esri/request', 'esri/core/promiseUtils', 'esri/core/tsSupport/awaiterHelper',
      'esri/core/tsSupport/generatorHelper', 'esri/core/tsSupport/assignHelper', 'esri/core/Error'
    ]).then((
      [MapImageLayer, q, x, t, d, k, p]) => {
      MapImageLayer.prototype._fetchService = function (a) {
        return t(this, void 0, void 0, function () {
          var b, e, f;
          return d(this, function (d) {
            switch (d.label) {
              case 0:
                return this.sourceJSON ? (this.read(this.sourceJSON, {
                  origin: "service",
                  url: this.parsedUrl
                }), [2]) :
                  [4, q(url + '?f=json', { signal: a })];
              case 1:
                b = d.sent();
                e = b.data;
                if (f = b.ssl)
                  this.url = this.url.replace(/^http:/i, "https:");
                this.sourceJSON = e;
                // 修改url部分
                this.read(e, {
                  origin: "service",
                  url: url
                });
                return [2]
            }
          })
        })
      }
      MapImageLayer.prototype.fetchImage = function (a, b, d, e) {
        var f = {
          responseType: "image"
        };
        e && e.timestamp && (f.query = {
          _ts: e.timestamp
        });
        e && e.signal && (f.signal = e.signal);
        // 加了个?
        var g = url + "/export?";
        a = k({}, {}, this.createExportImageParameters(a, b, d, e), {
          f: "image",
          _ts: this.alwaysRefetch ? Date.now() : null
        });
        if (null != a.dynamicLayers && !this.capabilities.exportMap.supportsDynamicLayers)
          return x.reject(new p("mapimagelayer:dynamiclayer-not-supported", "service " + this.url + " doesn't support dynamic layers, which is required to be able to change the sublayer's order, rendering, labeling or source.", {
            query: a
          }));
        f.query = f.query ? k({}, a, f.query) : a;
        // 新增
        f.query.dynamicLayers && delete f.query.dynamicLayers
        return q(g, f).then(function (a) {
          return a.data
        }).catch(function (a) {
          if (x.isAbortError(a))
            throw a;
          throw new p("mapimagelayer:image-fetch-error", "Unable to load image: " + g, {
            error: a
          });
        })
      }
      this.arcGisLayer = null
      let layer = new MapImageLayer({
        id: id,
        url: url
      })
      this.map.layers.add(layer)
      this.arcGisLayerList[id] = layer
    })
  }
}

修改部分主要在于_fetchService,将其request的url参数(原为this.url)修改为原始url,当然也可以修改this.url的set处理,示例里简单粗暴地完全赋值,实际也没用到。
_fetchService之后,TileLayer会调用getTileUrl获取瓦片数据,这里除了重写getTileUrl,还要重写urlUtils.addQueryParameters,具体过程有点迂回,在request的内部处理里面。到addQueryParameters时,原来会解析成realAddr=,我当时的处理是直接改成

// b是url,d是参数对象
urlUtils.addQueryParameters = function (b, d) {
    return b
}

也确实没问题,图个省事,d为空对象,当时也没有请求带参数的情况。
但是由于请求缓存的原因,这里的改动相当于全局修改,urlUtils是esri/core下的工具类。
MapImageLayer在_fetchService之后会调用fetchImage,这个方法是带参的,在request时也调用了addQueryParameters,所以就出了问题。最终是改成了上面那样。
至此,Arcgis加载带?的转发服务的实现思路就完了。
虽然产出成果不多,但也花了好几天时间,主要是压缩源码看的太费精力了,一是单字母的含义需要对应require的模块,二是三目表达式太多了,还带括号换行的,很容易漏掉断点,最难的是request部分的断点,跟进到addQueryParameters花了很多时间。不过好在肝出来了,下次看源码会更有经验一些。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,366评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,521评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,689评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,925评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,942评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,727评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,447评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,349评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,820评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,990评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,127评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,812评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,471评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,017评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,142评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,388评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,066评论 2 355

推荐阅读更多精彩内容