137 lines
4.4 KiB
JavaScript
137 lines
4.4 KiB
JavaScript
import VectorTileLayer from '../src/ol/layer/VectorTile.js';
|
|
import VectorTileSource from '../src/ol/source/VectorTile.js';
|
|
import MVT from '../src/ol/format/MVT.js';
|
|
import {Projection} from '../src/ol/proj.js';
|
|
import TileQueue from '../src/ol/TileQueue.js';
|
|
import {getTilePriority as tilePriorityFunction} from '../src/ol/TileQueue.js';
|
|
import {renderDeclutterItems} from '../src/ol/render.js';
|
|
import styleFunction from 'ol-mapbox-style/dist/stylefunction.js';
|
|
import {inView} from '../src/ol/layer/Layer.js';
|
|
import stringify from 'json-stringify-safe';
|
|
|
|
/** @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_D6rA4zTHduk6KOKTXzGB'
|
|
}),
|
|
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_D6rA4zTHduk6KOKTXzGB'
|
|
}),
|
|
openmaptiles: new VectorTileSource({
|
|
format: new MVT(),
|
|
maxZoom: 14,
|
|
url: 'https://api.maptiler.com/tiles/v3/{z}/{x}/{y}.pbf?key=get_your_own_D6rA4zTHduk6KOKTXzGB'
|
|
})
|
|
};
|
|
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_D6rA4zTHduk6KOKTXzGB';
|
|
|
|
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);
|
|
}
|
|
});
|
|
renderDeclutterItems(frameState, null);
|
|
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]);
|
|
});
|