4
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?

MapLibreAdvent Calendar 2024

Day 16

MapLibre GL JSでのパレットPNGタイルの凡例情報のポップアップ表示について

Last updated at Posted at 2024-12-16

はじめに

この記事では、MapLibre GL JSを使ってデータPNG(パレットPNG)タイルを表示し、クリックした地点の凡例情報をポップアップで表示する仕組みを紹介します。対象とする地図データは、RGB(赤・緑・青)の値で土地利用を表現したラスタータイルです。この仕組みを活用すると、独自のラスタータイルに凡例情報を付けて、インタラクティブな地図アプリを作成することが可能になります。具体には、下記の記事で作成した国土数値情報の土地利用細分メッシュデータ(100mメッシュ)のパレットPNGタイルをMapLibre GL JSで表示し、クリック時に取得したRGB値をもとに凡例情報を表示します。

パレットPNGタイルのRGB値の取得には、産業技術総合研究所の「2. 位置を指定して値を取得」で公開されている「パレットPNGタイル」のサンプル関数を参考にさせていただきました。詳細は以下のリンクから確認できます。このサンプルに少し手を加え、今回のMapLibre GL JSによる凡例情報の表示を行いました。

凡例情報のポップアップ表示までの流れ

  • 凡例情報のポップアップ表示までの流れは以下のとおりになります。
  1. 凡例情報の作成
  2. クリック地点の緯度・経度を取得
  3. タイル座標を計算し、タイル画像を参照
  4. タイル内のクリック地点に対応するピクセルのRGB値を取得
  5. RGB値を凡例情報と照合し、対応する凡例情報を特定
  6. ポップアップで凡例情報を表示

凡例情報の作成

        // 凡例情報セット
        const legend_tochi = [
            { r: 255, g: 255, b: 0, title: '' },
            { r: 255, g: 204, b: 153, title: 'その他の農用地' },
            { r: 0, g: 170, b: 0, title: '森林' },
            { r: 255, g: 153, b: 0, title: '荒地' },
            { r: 255, g: 0, b: 0, title: '建物用地' },
            { r: 140, g: 140, b: 140, title: '道路' },
            { r: 180, g: 180, b: 180, title: '鉄道' },
            { r: 200, g: 70, b: 15, title: 'その他の用地' },
            { r: 0, g: 0, b: 255, title: '河川地及び湖沼' },
            { r: 255, g: 255, b: 153, title: '海浜' },
            { r: 0, g: 204, b: 255, title: '海水域' },
            { r: 0, g: 255, b: 0, title: 'ゴルフ場' },
        ];

クリック地点の座標を取得

  • MapLibre GL JSのclickイベントを使って、クリックした場所の緯度・経度(lat,lng)を取得します。
map.on('click', (e) => {
    中略
    const lat = e.lngLat.lat; // 緯度
    const lng = e.lngLat.lng; // 経度
    中略
});

タイル座標を計算し、タイル画像を参照

  • クリック地点が属するタイルを特定するために、緯度・経度とズームレベルを元にタイル座標(x, y)を計算します。
        /// ****************
        // latLngToTile 緯度経度をタイル座標に変換する関数
        //  lat: 経度
        //  lng: 緯度
        //  z: ズームレベル
        //  戻り値: タイル座標オブジェクト(x, yフィールドを持ちます)
        //    ※通常,地図ライブラリ内に同様の関数が用意されています.
        /// ****************
        function latLngToTile(lat, lng, z) {
            const
                w = Math.pow(2, (z === undefined) ? 0 : z) / 2,		// 世界全体のピクセル幅
                yrad = Math.log(Math.tan(Math.PI * (90 + lat) / 360));

            return { x: (lng / 180 + 1) * w, y: (1 - yrad / Math.PI) * w };
        };

タイル内のクリック地点に対応するピクセルのRGB値を取得

  • クリック地点がタイル内のどのピクセルに相当するかを計算し、そのピクセルのRGB値を取得します。
  • 取得したRGB値を凡例情報と照合し、対応する凡例情報を特定します。
        /// ****************
        // getLegendItem 凡例情報,タイルURL,座標,ズームレベルを指定して凡例項目を取得する関数
        //  legend: 凡例情報オブジェクト.r,g,b,titleを持つ凡例項目オブジェクトの配列
        //	url: タイル画像のURLテンプレート.
        //		ズームレベル,X, Y座標をそれぞれ{z},{x},{y}として埋め込む
        //  lat: 経度オブジェクト
        //  lng: 緯度オブジェクト
        //  z: ズームレベル
        //  invalid: 追加無効値を相当する数値で指定.デフォルトは指定なし
        //  戻り値: 成功時に凡例項目オブジェクトを受け取るプロミス.該当するものがない場合はnullを受け取ります
        /// ****************
        function getLegendItem(legend, url, lat, lng, z, invalid = undefined) {
            return new Promise(function (resolve, reject) {
                const
                    p = latLngToTile(lat, lng, z),
                    x = Math.floor(p.x),			// タイルX座標
                    y = Math.floor(p.y),			// タイルY座標
                    i = (p.x - x) * 256,			// タイル内i座標
                    j = (p.y - y) * 256,			// タイル内j座標
                    img = new Image();

                img.crossOrigin = 'anonymous';	// クロスオリジン対応
                img.onload = function () {
                    const
                        canvas = document.createElement('canvas'),
                        context = canvas.getContext('2d');
                    let
                        v,
                        d;

                    canvas.width = 1;
                    canvas.height = 1;
                    context.drawImage(img, i, j, 1, 1, 0, 0, 1, 1);
                    d = context.getImageData(0, 0, 1, 1).data;
                    if (d[3] !== 255) {
                        v = null;
                    } else {
                        v = legend.find(o => o.r == d[0] && o.g == d[1] && o.b == d[2])
                    }
                    resolve(v);
                }
                img.onerror = function () {
                    resolve(null);
                }
                img.src = url.replace('{z}', z).replace('{y}', y).replace('{x}', x);
            });
        };

ポップアップで凡例情報を表示

  • 最後に、照合した凡例情報をポップアップとして表示します。
        // クリック時にポップアップ表示
        map.on('click', function (e) {
            // 表示されているレイヤーのIDを格納する配列
            var rasterLayerIds = [];
            const mapLayers = map.getStyle().layers;

            // 全てのレイヤーを走査して、'type'が'raster'のものをフィルタリング
            mapLayers.forEach(layer => {
                // レイヤーのtypeプロパティを取得
                const type = layer.type;
                if (type === 'raster') {
                    rasterLayerIds.push(layer.id);
                }
            });

            console.log('表示されているレイヤー: ' + rasterLayerIds);

            // 現在表示されているラスターレイヤをもとにポップアップ表示
            RasterTileUrl = 'https://shiworks.xsrv.jp/raster-tiles/mlit-ksj/L03-b-21_dissolved/{z}/{x}/{y}.png';
            legend = legend_tochi;

            // クリック地点の座標を取得
            const lng = e.lngLat.lng;
            const lat = e.lngLat.lat;

            getLegendItem(legend, RasterTileUrl, lat, lng, Math.trunc(map.getZoom())).then(function (v) {
                let s = '';
                let res = (v ? v.title : '取得できません');
                if (rasterLayerIds == 'ksj-L03-b-21') {
                    s = '<p>' + '土地利用種別' + '<br>' + '<b>' + res + '</b>' + '<br>' + '<a href=\https://www.google.com/maps?q=' + lat + "," + lng + "&hl=ja' target='_blank'>🌎Google Maps</a>" + '</p>';
                }
                new maplibregl.Popup()
                    .setLngLat(e.lngLat)
                    .setHTML(s)
                    .addTo(map);
            });
        });

デモサイトとソースコード

image.png

引き続き検証を行います。

4
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
4
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?