You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
163 lines
4.8 KiB
JavaScript
163 lines
4.8 KiB
JavaScript
import MVT from 'ol/format/MVT';
|
|
import TileQueue from 'ol/TileQueue';
|
|
import VectorTileLayer from 'ol/layer/VectorTile';
|
|
import VectorTileSource from 'ol/source/VectorTile';
|
|
import stringify from 'json-stringify-safe';
|
|
import styleFunction from 'ol-mapbox-style/dist/stylefunction';
|
|
import {Projection} from 'ol/proj';
|
|
import {inView} from 'ol/layer/Layer';
|
|
import {getTilePriority as tilePriorityFunction} from 'ol/TileQueue';
|
|
|
|
/** @type {any} */
|
|
const worker = self;
|
|
|
|
let frameState, pixelRatio, rendererTransform;
|
|
const canvas = new OffscreenCanvas(1, 1);
|
|
// OffscreenCanvas does not have a style, so we mock it
|
|
canvas.style = {};
|
|
const context = canvas.getContext('2d');
|
|
|
|
const sources = {
|
|
landcover: new VectorTileSource({
|
|
maxZoom: 9,
|
|
format: new MVT(),
|
|
url:
|
|
'https://api.maptiler.com/tiles/landcover/{z}/{x}/{y}.pbf?key=Get your own API key at https://www.maptiler.com/cloud/',
|
|
}),
|
|
contours: new VectorTileSource({
|
|
minZoom: 9,
|
|
maxZoom: 14,
|
|
format: new MVT(),
|
|
url:
|
|
'https://api.maptiler.com/tiles/contours/{z}/{x}/{y}.pbf?key=Get your own API key at https://www.maptiler.com/cloud/',
|
|
}),
|
|
openmaptiles: new VectorTileSource({
|
|
format: new MVT(),
|
|
maxZoom: 14,
|
|
url:
|
|
'https://api.maptiler.com/tiles/v3/{z}/{x}/{y}.pbf?key=Get your own API key at https://www.maptiler.com/cloud/',
|
|
}),
|
|
};
|
|
const layers = [];
|
|
|
|
// Font replacement so we do not need to load web fonts in the worker
|
|
function getFont(font) {
|
|
return font[0].replace('Noto Sans', 'serif').replace('Roboto', 'sans-serif');
|
|
}
|
|
|
|
function loadStyles() {
|
|
const styleUrl =
|
|
'https://api.maptiler.com/maps/topo/style.json?key=Get your own API key at https://www.maptiler.com/cloud/';
|
|
|
|
fetch(styleUrl)
|
|
.then((data) => data.json())
|
|
.then((styleJson) => {
|
|
const buckets = [];
|
|
let currentSource;
|
|
styleJson.layers.forEach((layer) => {
|
|
if (!layer.source) {
|
|
return;
|
|
}
|
|
if (currentSource !== layer.source) {
|
|
currentSource = layer.source;
|
|
buckets.push({
|
|
source: layer.source,
|
|
layers: [],
|
|
});
|
|
}
|
|
buckets[buckets.length - 1].layers.push(layer.id);
|
|
});
|
|
|
|
const spriteUrl =
|
|
styleJson.sprite + (pixelRatio > 1 ? '@2x' : '') + '.json';
|
|
const spriteImageUrl =
|
|
styleJson.sprite + (pixelRatio > 1 ? '@2x' : '') + '.png';
|
|
fetch(spriteUrl)
|
|
.then((data) => data.json())
|
|
.then((spriteJson) => {
|
|
buckets.forEach((bucket) => {
|
|
const source = sources[bucket.source];
|
|
if (!source) {
|
|
return;
|
|
}
|
|
const layer = new VectorTileLayer({
|
|
declutter: true,
|
|
source,
|
|
minZoom: source.getTileGrid().getMinZoom(),
|
|
});
|
|
layer.getRenderer().useContainer = function (target, transform) {
|
|
this.containerReused = this.getLayer() !== layers[0];
|
|
this.canvas = canvas;
|
|
this.context = context;
|
|
this.container = {
|
|
firstElementChild: canvas,
|
|
};
|
|
rendererTransform = transform;
|
|
};
|
|
styleFunction(
|
|
layer,
|
|
styleJson,
|
|
bucket.layers,
|
|
undefined,
|
|
spriteJson,
|
|
spriteImageUrl,
|
|
getFont
|
|
);
|
|
layers.push(layer);
|
|
});
|
|
worker.postMessage({action: 'requestRender'});
|
|
});
|
|
});
|
|
}
|
|
|
|
// Minimal map-like functionality for rendering
|
|
|
|
const tileQueue = new TileQueue(
|
|
(tile, tileSourceKey, tileCenter, tileResolution) =>
|
|
tilePriorityFunction(
|
|
frameState,
|
|
tile,
|
|
tileSourceKey,
|
|
tileCenter,
|
|
tileResolution
|
|
),
|
|
() => worker.postMessage({action: 'requestRender'})
|
|
);
|
|
|
|
const maxTotalLoading = 8;
|
|
const maxNewLoads = 2;
|
|
|
|
worker.addEventListener('message', (event) => {
|
|
if (event.data.action !== 'render') {
|
|
return;
|
|
}
|
|
frameState = event.data.frameState;
|
|
if (!pixelRatio) {
|
|
pixelRatio = frameState.pixelRatio;
|
|
loadStyles();
|
|
}
|
|
frameState.tileQueue = tileQueue;
|
|
frameState.viewState.projection.__proto__ = Projection.prototype;
|
|
layers.forEach((layer) => {
|
|
if (inView(layer.getLayerState(), frameState.viewState)) {
|
|
const renderer = layer.getRenderer();
|
|
renderer.renderFrame(frameState, canvas);
|
|
}
|
|
});
|
|
layers.forEach((layer) => layer.renderDeclutter(frameState));
|
|
if (tileQueue.getTilesLoading() < maxTotalLoading) {
|
|
tileQueue.reprioritize();
|
|
tileQueue.loadMoreTiles(maxTotalLoading, maxNewLoads);
|
|
}
|
|
const imageData = canvas.transferToImageBitmap();
|
|
worker.postMessage(
|
|
{
|
|
action: 'rendered',
|
|
imageData: imageData,
|
|
transform: rendererTransform,
|
|
frameState: JSON.parse(stringify(frameState)),
|
|
},
|
|
[imageData]
|
|
);
|
|
});
|