import "./index.css"; import "bootstrap"; import "bootstrap/dist/css/bootstrap.min.css"; import StylesControl from "@mapbox-controls/styles"; import ZoomControl from "@mapbox-controls/zoom"; import CompassControl from "@mapbox-controls/compass"; import InspectControl from "@mapbox-controls/inspect"; import "@mapbox-controls/styles/src/index.css"; import "@mapbox-controls/zoom/src/index.css"; import "@mapbox-controls/compass/src/index.css"; import "@mapbox-controls/inspect/src/index.css"; import mapboxgl from "mapbox-gl"; import "mapbox-gl/dist/mapbox-gl.css"; import showdown from "showdown"; function loginfo(...str) { let info = str.shift(); console.log( `%c ${info} `, "color:white; background-color: #78d6fa; border-radius:10px;", ...str ); } window.onload = async () => { const params = new Proxy(new URLSearchParams(window.location.search), { get: (searchParams, prop) => searchParams.get(prop), }); const projection = params.projection || "globe"; const mapId = params.id || "worldMap"; let mapDataFromId = ( await ( await fetch( `https://raw.githubusercontent.com/CIMEngine/MapList/main/index.json` ) ).json() )[mapId]; if (mapDataFromId.external) { mapDataFromId = await (await fetch(mapDataFromId.external)).json(); } if (!mapDataFromId) { alert(`Map ${mapId} not found`); } if (mapDataFromId.icon) { document.getElementById("icon").setAttribute("href", mapDataFromId.icon); } let geoURL, countryInfoUrl, debug; if (params.external) { const data = await (await fetch(params.external)).json(); geoURL = data.geoURL; } geoURL = params.geoURL || mapDataFromId.geoURL; countryInfoUrl = params.countryInfoURL || mapDataFromId.countryInfoURL; debug = params.debug || false; mapboxgl.accessToken = "pk.eyJ1IjoiYXJ0ZWdvc2VyIiwiYSI6ImNrcDViN3BhcDAwbW0ydnBnOXZ0ZzFreXUifQ.FIVtaBNr9dr_TIw672Zqdw"; let map = new mapboxgl.Map({ container: "map", style: "mapbox://styles/artegoser/clfm612fg002601nlcika2018?optimize=true", center: [53.19, 41.28], zoom: 3, projection: projection, }); map.addControl( new StylesControl({ styles: [ { label: "Streets", styleName: "Mapbox Streets", styleUrl: "mapbox://styles/artegoser/clfm612fg002601nlcika2018?optimize=true", }, { label: "Satellite", styleName: "Satellite", styleUrl: "mapbox://styles/artegoser/cliskjlhw00ug01pgfs9lesog?optimize=true", }, ], }), "top-left" ); map.addControl(new ZoomControl(), "top-right"); map.addControl(new CompassControl(), "bottom-right"); if (debug) { map.addControl(new InspectControl(), "bottom-right"); } let converter = new showdown.Converter(); map.on("style.load", async () => { map.loadImage( "https://cimengine.github.io/map/icons/city.png", (error, image) => { if (error) throw error; map.addImage("city", image); } ); map.loadImage( "https://cimengine.github.io/map/icons/capital.png", (error, image) => { if (error) throw error; map.addImage("capital-city", image); map.addImage("capital", image); } ); map.loadImage( "https://cimengine.github.io/map/icons/landmark.png", (error, image) => { if (error) throw error; map.addImage("landmark-0", image); } ); let lasticocords; loginfo("Getting country data"); let coarray = await fetch(countryInfoUrl); coarray = await coarray.json(); let countries = {}; for (let i = 0; i < coarray.length; i++) countries[coarray[i].idc] = coarray[i]; map.addSource("map-data", { type: "geojson", data: geoURL, }); map.addLayer({ id: "map-data-fill-outline", type: "line", source: "map-data", paint: { "line-color": ["coalesce", ["get", "stroke"], "#0c7700"], "line-width": 2, "line-opacity": 0.8, }, }); map.addLayer({ id: "map-data-fill", type: "fill", source: "map-data", paint: { "fill-color": ["get", "fill"], "fill-opacity": ["coalesce", ["get", "fill-opacity"], 0.3], }, }); map.addLayer({ id: "map-data-symbol", type: "symbol", source: "map-data", layout: { "icon-image": ["get", "type"], "icon-size": 0.15, }, minzoom: 3, }); map.on("click", "map-data-fill", (e) => { const coordinates = e.lngLat; while (Math.abs(e.lngLat.lng - coordinates[0]) > 180) { coordinates[0] += e.lngLat.lng > coordinates[0] ? 360 : -360; } onEachFeature(e.features[0], coordinates); }); map.on("click", "map-data-symbol", (e) => { const coordinates = e.lngLat; while (Math.abs(e.lngLat.lng - coordinates[0]) > 180) { coordinates[0] += e.lngLat.lng > coordinates[0] ? 360 : -360; } onEachFeature(e.features[0], coordinates); }); function onEachFeature(feature, coordinates) { if (feature.geometry.type === "Point") { lasticocords = coordinates; return new mapboxgl.Popup() .setLngLat(coordinates) .setHTML( ` ${ feature?.properties?.amount ? `
Population - ${feature.properties.amount} people.
` : "" }
${ feature?.properties?.img ? `
${feature.properties.name} img
` : "" }
${feature.properties.name} ${ feature.properties.name_ru ? ` - ${feature.properties.name_ru}` : "" }
${ feature.properties.description ? `
${converter.makeHtml( feature.properties.description )}
` : "" }
` ) .addTo(map); } else if ( feature.geometry.type === "Polygon" || feature.geometry.type === "MultiPolygon" ) { let country = countries[feature.properties.name] || { name: "gl js mapbox is awesome", }; setTimeout(() => { if (country.name !== "gl js mapbox is awesome") if (lasticocords !== coordinates) return new mapboxgl.Popup() .setLngLat(coordinates) .setHTML( `
${country.name}
${JSON.parse( feature.properties.tags || "[]" ).join(", ")}
Founding date: ${country.date}
${ country.description ? `
${converter.makeHtml( country.description )}
` : "" }
Area: ${feature.properties.area} kmĀ²
${ country.about ? `About` : "" }
` ) .addTo(map); }, 1); } } }); };