「地理院地図Vector」(仮称)をArcGIS で表示してみました - 第2弾

Document created by shinji_katayaesrij-esridist Employee on Sep 3, 2020Last modified by shinji_katayaesrij-esridist Employee on Sep 16, 2020
Version 2Show Document
  • View in full screen mode

はじめに

2019年8月1日に「地理院地図Vector」(仮称)をArcGIS で表示してみました として、ベクトルタイルの道路データをArcGIS API for JavaScript で表示してみました。当時は、一部地域のベクトルタイルの提供でしたが、2020年3月19日からは全国データの公開が始まっています。

また、地理院地図Vector(仮称)提供実験 で提供しているスタイルファイル(style.json)は、Style Specification をベースに若干の拡張を施していますが、別のGitHubリポジトリ で、Style Specification に準拠したスタイルと、Mapbox GL JSで地理院地図Vector風の地図を表示するサンプルが公開されています。

 

スタイルファイルの準備

前回の記事では独自にスタイルファイルを作成しましたが、今回は、

地理院地図Vectorの標準地図をベースにしたスタイル(std.json)

地理院地図Vectorの淡色地図をベースにしたスタイル(pale.json)

地理院地図Vectorの白地図をベースにしたスタイル(blank.json)

として公開されている主要な3つのスタイルファイルを編集して用意しました(一部、Expression式 などAPIで非対応のものを除外)。

 

ArcGIS API for JavaScript で表示

スタイルファイルの準備が終われば、後は前回の記事と同様に、VectorTileLayer クラスを使用して次のように実装すると地図の表示は可能です。

// ベクター タイル レイヤーの作成
var vtlLayer = new VectorTileLayer({
url:<スタイル ファイルの参照 URL>"
});
// マップに作成したレイヤーを追加
map.add(vtlLayer);

 

今回はせっかく3つのスタイルファイルがあるので、以前に続編 はじめてのWeb マッピングアプリケーション開発:ウィジェット編 で紹介したBasemapGallery ウィジェットをカスタマイズし、地理院地図、地理院地図Vector風の地図を切り替えするようなサンプルアプリケーションを作成してみました。

 

 

 

実装のポイント

今回はBasemapGallery ウィジェットで、地理院地図、地理院地図Vector風の地図を扱えるようにするための、実装ポイントは3つです。

 

 実装のポイント1:地理院地図

標準地図、淡色地図、写真などの地理院地図はWebTileLayer クラスで扱うことが可能ですが、BasemapGallery で使えるように、Basemap クラスを使います。

実際にAPI リファレンスを見てみると、サードパーティのデータソースとして使う場合には、次のように実装するように例示されています。

// or create from a third party source
var basemap = new Basemap({
  baseLayers: [
    new WebTileLayer(...)
  ],
  referenceLayers: [
    new WebTileLayer(...)
  ],
});

 

今回は上記を参考にしながら、baseLayers に、地理院地図の情報をWebTileLayer  のインスタンスを設定します。

// 以下の地理院タイル(ベースマップ)のページに記載のURLの定義のうち
// URL の /{z}/{x}/{y} は /{level}/{col}/{row} に変更してurlTemplate に設定
//   https://maps.gsi.go.jp/development/ichiran.html#std

// 地理院地図の標準地図の例)
const stdMap = new Basemap({
    baseLayers:[new WebTileLayer({
        urlTemplate: "https://cyberjapandata.gsi.go.jp/xyz/std/{level}/{col}/{row}.png",
        copyright: "地理院タイル (https://maps.gsi.go.jp/development/ichiran.html)"
    })],
    title: "地理院タイル - 標準地図",
    id: "stdmap",
    thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/std/12/3637/1612.png"
});

// ※地理院地図の淡色地図の例、地理院地図の写真の例は、サンプル内に記載してあります

 

 実装のポイント2:地理院地図Vector風の地図

同様に、地理院地図Vector風の地図の場合は、VectorTileLayer クラスで扱うことが可能ですが、baseLayers に、VectorTileLayer のインスタンスを設定します。

// 地理院地図Vector風の標準地図の例)
const stdVMap = new Basemap({
    baseLayers:[new VectorTileLayer({
        url: "https://kataya.github.io/gsivectortile-mapbox-gl-js/std_edit.json",
        copyright: "地理院地図Vector(仮称) (https://maps.gsi.go.jp/vector/)"
    })],
    title: "地理院ベクトルタイル - 標準地図",
    id: "stdvmap",
    thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/std/12/3637/1612.png"
});

// ※地理院地図Vector風の淡色地図、白地図の例は、サンプル内に記載してあります

 

 実装のポイント3:BasemapGallery への設定

定義した地理院地図、地理院地図Vector風のBasemap を、LocalBasemapsSource に設定後、BasemapGallery source として指定します。

// Basemap Sources を LocalBasemapsSource に設定
const localSource = new LocalBasemapsSource({
    basemaps: [esriTopo, stdMap, paleMap, seamlessPhoto, stdVMap, paleVMap, blankVMap]
});
               
// Basemap Gallery Widget に LocalBasemapsSource を設定
const basemapGallery = new BasemapGallery({
    view: view,
    container: document.createElement("div"),
    source: localSource
});

 

 完成版のサンプル

<!DOCTYPE html>
<html>

<head>
  <meta charset="utf-8">
  <meta name="viewport" content="initial-scale=1,maximum-scale=1,user-scalable=no">
  <title>地理院地図Vector風の地図をArcGIS で表示してみました</title>
  <link rel="stylesheet" href="https://js.arcgis.com/4.16/esri/themes/dark/main.css" />
  <script src="https://js.arcgis.com/4.16/"></script>
  <style>
    html,
    body,
    #viewDiv {
      padding: 0;
      margin: 0;
      height: 100%;
      width: 100%;
    }
  </style>

  <script>
    require([
      "esri/config",
      "esri/Map",
      "esri/portal/Portal",
      "esri/views/MapView",
      "esri/Basemap",
      "esri/widgets/Expand",
      "esri/widgets/Search",
      "esri/widgets/BasemapGallery",
      "esri/widgets/BasemapGallery/support/LocalBasemapsSource",
      "esri/layers/WebTileLayer",
      "esri/layers/TileLayer",
      "esri/layers/VectorTileLayer",
      "esri/intl",
      "dojo/domReady!"
    ], function(
        esriConfig,
        Map,
        Portal,
        MapView,
        Basemap,
        Expand,
        Search,
        BasemapGallery,
        LocalBasemapsSource,
        WebTileLayer,
        TileLayer,
        VectorTileLayer,
        intl
    ) {
        // ******************************************************************************
        // ArcGIS のベースマップ の他、
        // 地理院地図、地理院地図Vector風 などの外部サービスの地図をBasemapGallery で利用するサンプル
        //
        // なお、ArcGIS だけの世界で対応可能なレイヤーであれば、カスタムグループを作りベースマップを登録して利用するのが一般的
        // https://blog.esrij.com/2015/07/14/post-c857/
        // ******************************************************************************
       
        // UI のローカライズ
        intl.setLocale("ja");
       
        // リクエストを横取りしてパラメータに culture と region を設定するコード
        const selfUrl = "https://www.arcgis.com/sharing/rest/portals/self";
        esriConfig.request.interceptors.push({
          urls: selfUrl,
          before: function(params) {
            if (params.requestOptions.query) {
                switch(intl.getLocale())
                {
                    case "en":
                        params.requestOptions.query.culture="en";
                        params.requestOptions.query.region="US";
                        break;
                    case "ja":
                        params.requestOptions.query.culture="ja";
                        params.requestOptions.query.region="JP";
                        break;
                    case "de":
                        params.requestOptions.query.culture="de";
                        params.requestOptions.query.region="DE";
                        break;
                    case "fr":
                        params.requestOptions.query.culture="fr";
                        params.requestOptions.query.region="FR";
                        break;
                }
            }
          }
        });
       
        // portal のインスタンスの初期化とロード
        const portal = new Portal();
        //portal.authMode = "immediate"; // public にしていない場合でポータル内の情報を使う場合、通常はログイン必要
       
        portal
            .load()
            .then(function() {
                const basemap = portal.useVectorBasemaps
                    ? portal.defaultVectorBasemap
                    : portal.defaultBasemap;
                const map = new Map({
                    basemap: basemap
                    //basemap: "topo-vector" // 英語のtopo(地形図)ベクタータイル
                    //basemap: "osm"
                    //basemap: {portalItem:{id: "9e74c6647b6e41538fb3b2a4b7e90784"}} // 日本語のtopo(地形図)ベクタータイル
                });
                const view = new MapView({
                    container: "viewDiv",
                    map: map,
                    center: [139.715512, 35.678257], // 関東周辺
                    zoom: 8,
                    constraints: { // 拡大と縮小のスケールを制限
                        minZoom: 5,
                        maxZoom: 18
                    }
                });
               
                // Esri Topo Layers
                //const esriTopo = new Basemap({
                //    baseLayers: [new TileLayer({
                //      url: "https://services.arcgisonline.com/ArcGIS/rest/services/World_Topo_Map/MapServer?f=json"
                //    })],
                //    title: "Esri - Topo",
                //    id: "Topo",
                //    thumbnailUrl:"https://js.arcgis.com/4.13/esri/images/basemap/topo.jpg"
                //});
                const esriTopo = basemap;
                esriTopo.thumbnailUrl = "https://js.arcgis.com/4.16/esri/images/basemap/topo.jpg";
               
                /******************************************************************************
                地理院タイル(ベースマップ)のページから情報を整理
                 https://maps.gsi.go.jp/development/ichiran.html#std
               
                ※ 実装時には、URL の /{z}/{x}/{y} は /{level}/{col}/{row} に変更
               
                標準地図:レベル 5-18
                 urlTemplate: "https://cyberjapandata.gsi.go.jp/xyz/std/{z}/{x}/{y}.png"
                 copyright: "地理院タイル"
                 title: "標準地図"
                 thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/std/12/3637/1612.png"
               
                淡色地図:レベル 5-18
                 urlTemplate: "https://cyberjapandata.gsi.go.jp/xyz/pale/{z}/{x}/{y}.png"
                 copyright: "地理院タイル"
                 title: "淡色地図"
                 thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/pale/12/3637/1612.png"
               
                English:レベル 5-11
                 urlTemplate: "https://cyberjapandata.gsi.go.jp/xyz/english/{z}/{x}/{y}.png"
                 copyright: "地理院タイル"        
                 title: "English"
                 thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/english/10/909/403.png"
               
                写真:レベル 2-18
                 urlTemplate: "https://cyberjapandata.gsi.go.jp/xyz/seamlessphoto/{z}/{x}/{y}.jpg"
                 copyright: "地理院タイル"
                 title: 写真
                 thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/seamlessphoto/16/58274/25716.jpg
                ******************************************************************************/

               
                const stdMap = new Basemap({
                    baseLayers:[new WebTileLayer({
                      urlTemplate: "https://cyberjapandata.gsi.go.jp/xyz/std/{level}/{col}/{row}.png",
                      copyright: "地理院タイル (https://maps.gsi.go.jp/development/ichiran.html)"
                    })],
                    title: "地理院タイル - 標準地図",
                    id: "stdmap",
                    thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/std/12/3637/1612.png"
                });     
               
                const paleMap = new Basemap({
                    baseLayers:[new WebTileLayer({
                        urlTemplate: "https://cyberjapandata.gsi.go.jp/xyz/pale/{level}/{col}/{row}.png",
                        copyright: "地理院タイル (https://maps.gsi.go.jp/development/ichiran.html)"
                    })],
                    title: "地理院タイル - 淡色地図",
                    id: "palemap",
                    thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/pale/12/3637/1612.png"
                });   
               
                const seamlessPhoto = new Basemap({
                    baseLayers:[new WebTileLayer({
                        urlTemplate: "https://cyberjapandata.gsi.go.jp/xyz/seamlessphoto/{level}/{col}/{row}.jpg",
                        copyright: "地理院タイル (https://maps.gsi.go.jp/development/ichiran.html)"
                    })],
                    title: "地理院タイル - 写真",
                    id: "seamlessphoto",
                    thumbnailUrl:"https://cyberjapandata.gsi.go.jp/xyz/seamlessphoto/16/58274/25716.jpg"
                });
               
                const stdVMap = new Basemap({
                    baseLayers:[new VectorTileLayer({
                      url: "https://kataya.github.io/gsivectortile-mapbox-gl-js/std_edit.json",
                      copyright: "地理院地図Vector(仮称) (https://maps.gsi.go.jp/vector/)"
                    })],
                    title: "地理院ベクトルタイル - 標準地図",
                    id: "stdvmap",
                    thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/std/12/3637/1612.png"
                });
               
                const paleVMap = new Basemap({
                    baseLayers:[new VectorTileLayer({
                      url: "https://kataya.github.io/gsivectortile-mapbox-gl-js/pale_edit.json",
                      copyright: "地理院地図Vector(仮称) (https://maps.gsi.go.jp/vector/)"
                    })],
                    title: "地理院ベクトルタイル - 淡色地図",
                    id: "palevmap",
                    thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/pale/12/3637/1612.png"
                });
               
                const blankVMap = new Basemap({
                    baseLayers:[new VectorTileLayer({
                      url: "https://kataya.github.io/gsivectortile-mapbox-gl-js/blank_edit.json",
                      copyright: "地理院地図Vector(仮称) (https://maps.gsi.go.jp/vector/)"
                    })],
                    title: "地理院ベクトルタイル - 白地図",
                    id: "blankvmap",
                    thumbnailUrl: "https://cyberjapandata.gsi.go.jp/xyz/blank/12/3637/1612.png"
                });
               
                // Basemap Sources を LocalBasemapsSource に設定
                const localSource = new LocalBasemapsSource({
                    basemaps: [esriTopo, stdMap, paleMap, seamlessPhoto, stdVMap, paleVMap, blankVMap]
                });
               
                // Basemap Gallery Widget に LocalBasemapsSource を設定
                const basemapGallery = new BasemapGallery({
                    view: view,
                    container: document.createElement("div"),
                    source: localSource
                });
               
                // Basemap Gallery Widget – Expand Button
                const bgExpand = new Expand({
                    view: view,
                    content: basemapGallery
                });
                view.ui.add(bgExpand, "bottom-left");
               
                const search = new Search({ view: view });
                view.ui.add(search, "top-right");
               
                const coordsWidget = document.createElement("div");
                coordsWidget.id = "coordsWidget";
                coordsWidget.className = "esri-widget esri-component";
                coordsWidget.style.padding = "7px 15px 5px";
                view.ui.add(coordsWidget, "bottom-right");
           
                // lat, lon, zoom and scale の更新するfunction を定義
                function showCoordinates(pt) {
                    let coords = "Lat/Lon " + pt.latitude.toFixed(3) + " " + pt.longitude.toFixed(3) +
                                " | Scale 1:" + Math.round(view.scale * 1) / 1 +
                                " | Zoom " + view.zoom;
                    coordsWidget.innerHTML = coords;
                }
           
                // view の動作がが終了したタイミングで中心座標を表示するためのイベントを追加 (例えば zoom, pan などのイベント)
                view.watch(["stationary"], function() {
                    showCoordinates(view.center);
                });
               
                // クリックやムーブ時のマウスの座標を表示するためにイベントを追加
                view.on(["pointer-down","pointer-move"], function(evt) {
                    showCoordinates(view.toMap({ x: evt.x, y: evt.y }));
                });
               
            })
            .catch(function(error) {
                console.warn(error);
            });
        });
  </script>
</head>
<body>
  <div id="viewDiv"></div>
</body>

</html>

 

まとめ

今回の記事ではArcGIS API for JavaScript を利用して、BasemapGallery ウィジェットから、地理院地図、地理院地図Vector風の地図を切り替えして利用できるサンプルアプリケーションを作成してみました。

編集したスタイルファイル(std.json を編集したスタイルpale.json を編集したスタイルblank.json を編集したスタイル)や、完成版サンプルアプリケーションのコードは、いつものようにGitHub上で公開しておきますのでご参照ください。

また、サンプルアプリケーションは、こちらから操作して確認することも可能です。

https://kataya.github.io/gsivectortile-mapbox-gl-js/gsi_maps.html 

 

 

関連情報

 

参考情報

Attachments

    Outcomes