3
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

FOSS4GAdvent Calendar 2020

Day 22

地理院地図Vectorで地図注記の引っ越し

Posted at

これは FOSS4G Advent Calendar 2020 の 22日目の記事です。

はじめに

地図の注記だけを別の場所に持っていってみましょう。要するにこういう話です。

Hello

できあがり

今回は Leaflet 上で 地理院地図Vector@mapbox/vector-tileproj4js を使って実装してみました。

完成品のデモはこちらからどうぞ。

ソースコードはこちらに置いてあります。

使い方

  1. デモページを開くと地理院地図 Vector の注記が表示されています。この画像ではたまたま都道府県名レベルが表示されていますが、ズーム&パンするとそのズームや場所に応じた注記が表示されます。
    04.jpg

  2. 「今見えている注記をどこか別の場所で表示したい」と思ったら、画面下の HOLD ボタンを押します。
    11.jpg

  3. HOLD がチェックされた状態では、ズーム&パンしても注記が画面内に維持されます。

ただし、単純に画面に付箋のように貼り付いているわけではなく、注記の画面中心からの距離が維持されるように都度再計算されている、というのがポイントです。上の画像と下の画像は同じズームレベルですが、上の画像は中心に京都、左端に長崎が表示されているのに対して、下の北海道あたりにズーム&パンした画像では、中心の京都は変わらずですが、左端の長崎が見切れていますね。

26.jpg

  1. 注記をクリックすると、その注記と同カテゴリのものを残して他は見えなくなります。再度クリックすると元に戻ります。「駅だけ表示したい」みたいな場合にどうぞ。

37.jpg

  1. HOLD ボタンのとなりのスライダーを操作すると、画面中心を原点として注記が回転します。お好みでどうぞ。

53.jpg

解説

ソースコードはHTML+JS+CSSすべてひとつのファイルにおさめて200行くらいという簡単なものなので、見てもさほど苦にならないと思いますが、以下ポイントのみ。

1. 基本

いずれも拙稿ですがこのあたりを前提としています。

  • Leaflet のボイラープレート : Leaflet の基本的なボイラープレートです。Leaflet のバージョンが上がるたびにちまちまと更新しているのでたまに思い出してみてください。今年は v1.7.1 のリリースのみでしたね。
  • Leaflet でバイナリベクトルタイル処理の流れを追ってみる : Leaflet でバイナリベクトルタイル(MVT) を扱う方法を紹介した記事です。MVT を GeoJSON 変換して Leaflet で使おう、というものです。このへんはほとんど動きがなかったと思いますが、mapbox-gl-js のバージョンアップ+ライセンス刷新で、今後波風くらいは立つかもしれません。

2. 地理院地図 Vector

地理院地図 Vector に関して注意する点があるとすると、以下のパートでしょうか。

本提供実験によるベクトルタイルにおけるズームレベル({z})は、現在「地理院地図」で提供している地理院タイル(ラスタ)( https://maps.gsi.go.jp/development/ichiran.html )のズームレベルと同一ではありません。
画面上で同じ大きさで表示される際のズームレベルは、ベクトルタイルにおける数値が、地理院タイル(ラスタ)のズームレベルと比べて1小さい数値となります。そのため、ベクトルタイルにおけるズームレベル11のデータは、ズームレベルが12の地理院タイル(ラスタ)と同じデータを用いて作成しています。
また、ベクトルタイルにおいて画面に表示されるタイルの大きさは、地理院タイル(ラスタ)でズームレベルが1大きい(大きさが小さい)タイルの4枚分に相当します。
(出典: gh:gsi-cyberjapan/gsimaps-vector-experiment#ズームレベルについて)

これを Leaflet の語彙で解釈すると、L.TileLayer でいうところの

  • tileSize は(デフォルトの256ではなく) 512 とせよ
  • zoomOffset を -1 とせよ

ということになります。普通に tileSize=256, zoomOffset=0 で実装してしまうと、一画面を表示するのに4倍のタイルを取得する上、さらに本来欲しいものよりもズームレベルがひとつ大きいタイルを取得する、ということになってしまい、激重x超過密、という悲惨な結果になります。注意しましょう。

あと zoomOffsetL.TileLayer に実装されているんですが、派生元である L.GridLayer ではサポートされていないことにはちょっと注意が必要です。今回のように L.GridLayer を拡張するケースでは DIY が必要です。

3. 座標系変換

注記の引っ越しに当たっては、以下のような座標変換が発生しています。

  1. 注記は WGS84 の緯度経度 P0 を持っている
  2. HOLD ボタンが押されたときに「画面中心を原点とした平面直角座標系(CS1)」が設定される
  3. P0 の CS1 上における点 P1 が計算され、注記の属性として保持される
  4. ズーム&パンによって画面の中心が移動すると「そのときの画面中心を原点とした平面直角座標系(CS2)」が設定される
  5. P1 は本来 CS1 上の点だが、これが CS2 上の同じ座標値の点に移動したものとみなして CS2 上の点 P2 が設定される
  6. P2 の WSG84 における点 P3 が計算され、注記の新たな緯度経度として採用される(L.Marker の位置が変更される)

だいたいこういう処理は proj4js を使っておけばいいのですが、Leaflet には Projection というインターフェイスがあるので、それでラッピングするようにしています。
以下は CS1 や CS2 に対応する Projection を作っている部分です。EPSG:4326 から 所与の緯度経度 latlng を原点とした平面直角座標系への投影を行う projection オブジェクトを作り、 Projection.project(latlng) では緯度経度から平面直角座標系の座標への変換を、Projection.unproject(point) では平面直角座標系の座標から緯度経度への変換(逆変換)を行うというものです。

+proj=tmerc +lat_0=... の部分は https://spatialreference.org/ref/epsg/2443/proj4/ などを参考にしました。

const createProjection = function(latlng) {
  const projection = proj4("EPSG:4326", `+proj=tmerc +lat_0=${latlng.lat} +lon_0=${latlng.lng} +k=0.9999 +x_0=0 +y_0=0 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs `);
  return {
    project: function(latlng) {
      const xy = projection.forward([latlng.lng, latlng.lat]);
      return L.point(xy[0], xy[1]);
    },
    unproject: function(point) {
      const ll = projection.inverse([point.x, point.y]);
      return L.latLng(ll[1], ll[0]);
    }
  };
};

4. スタイリング

本当は地理院地図Vector のスタイルをもとに、地理院地図Vector の注記スタイルを模倣できればよいのですが、時間がたりませんでした。ここではとりあえず以下のポリシーとしました。

  • フォントサイズは一律14px
  • レイアウトも固定(文字の回転=なし、text-align=左寄せ)
  • 背景色は各注記に付与された annoCtg (数字3文字) をもとに色相を計算した hsl 着色
  • 注記のあたまに黒背景白文字で annoCtg を表示

annoCtg は注記の分類コードです。
実際にどのコードがなにを指すのかは https://maps.gsi.go.jp/help/pdf/vector/dataspec.pdf を参照。

まとめ

いくつかの技術を寄せ集めてちょっとしたアプリを作ってみました。
枯れた感じのある Leaflet ですが、変則的なアプリを作るときの素体としては相変わらず便利だな、と思っています。あと文字に特化したユースケースだと mapbox-gl-js のフォントローディングのオーバーヘッドと文字品質はやっぱり課題だと思うので適材適所かな、と。

そういう話とは別に、地図注記の引っ越し、面白そうな使い方があったら教えてもらえるとうれしいです。

3
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?