Create a basic example for OffscreenCanvas rendering
This commit is contained in:
75
examples/mvtlayer.worker.js
Normal file
75
examples/mvtlayer.worker.js
Normal file
@@ -0,0 +1,75 @@
|
||||
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';
|
||||
|
||||
const key = 'pk.eyJ1IjoiYWhvY2V2YXIiLCJhIjoiY2pzbmg0Nmk5MGF5NzQzbzRnbDNoeHJrbiJ9.7_-_gL8ur7ZtEiNwRfCy7Q';
|
||||
|
||||
/** @type {any} */
|
||||
const worker = self;
|
||||
|
||||
let frameState;
|
||||
|
||||
function getTilePriority(tile, tileSourceKey, tileCenter, tileResolution) {
|
||||
return tilePriorityFunction(frameState, tile, tileSourceKey, tileCenter, tileResolution);
|
||||
}
|
||||
|
||||
const layer = new VectorTileLayer({
|
||||
declutter: true,
|
||||
source: new VectorTileSource({
|
||||
format: new MVT(),
|
||||
url: 'https://{a-d}.tiles.mapbox.com/v4/mapbox.mapbox-streets-v6/' +
|
||||
'{z}/{x}/{y}.vector.pbf?access_token=' + key
|
||||
})
|
||||
});
|
||||
const tileQueue = new TileQueue(getTilePriority, function() {
|
||||
worker.postMessage({type: 'render'});
|
||||
});
|
||||
const maxTotalLoading = 8;
|
||||
const maxNewLoads = 2;
|
||||
|
||||
const renderer = layer.getRenderer();
|
||||
|
||||
renderer.useContainer = function(target, transform, opacity) {
|
||||
target.style = {};
|
||||
this.canvas = target;
|
||||
this.context = target.getContext('2d');
|
||||
this.container = {
|
||||
firstElementChild: target
|
||||
};
|
||||
worker.postMessage({
|
||||
type: 'transform-opacity',
|
||||
transform: transform,
|
||||
opacity: opacity
|
||||
});
|
||||
};
|
||||
|
||||
let canvas;
|
||||
let rendering = false;
|
||||
|
||||
worker.onmessage = function(event) {
|
||||
if (rendering) {
|
||||
// drop this frame
|
||||
return;
|
||||
}
|
||||
if (event.data.canvas) {
|
||||
canvas = event.data.canvas;
|
||||
} else {
|
||||
frameState = event.data.frameState;
|
||||
frameState.tileQueue = tileQueue;
|
||||
frameState.viewState.projection.__proto__ = Projection.prototype;
|
||||
rendering = true;
|
||||
requestAnimationFrame(function() {
|
||||
renderer.renderFrame(frameState, canvas);
|
||||
if (tileQueue.getTilesLoading() < maxTotalLoading) {
|
||||
tileQueue.reprioritize(); // FIXME only call if view has changed
|
||||
tileQueue.loadMoreTiles(maxTotalLoading, maxNewLoads);
|
||||
}
|
||||
rendering = false;
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
export let create;
|
||||
9
examples/offscreen-canvas-tiles.html
Normal file
9
examples/offscreen-canvas-tiles.html
Normal file
@@ -0,0 +1,9 @@
|
||||
---
|
||||
layout: example.html
|
||||
title: Vector tiles rendered in an offscreen canvas
|
||||
shortdesc: Example of a map that delegates rendering to a worker.
|
||||
docs: >
|
||||
The map in this example is rendered in a web worker, using `OffscreenCanvas`. **Note:** This is currently only supported in Chrome and Edge.
|
||||
tags: "worker, offscreencanvas, vector-tiles"
|
||||
---
|
||||
<div id="map" class="map"></div>
|
||||
63
examples/offscreen-canvas-tiles.js
Normal file
63
examples/offscreen-canvas-tiles.js
Normal file
@@ -0,0 +1,63 @@
|
||||
import Map from '../src/ol/Map.js';
|
||||
import View from '../src/ol/View.js';
|
||||
import Layer from '../src/ol/layer/Layer.js';
|
||||
//eslint-disable-next-line
|
||||
import Worker from 'worker-loader!./mvtlayer.worker.js';
|
||||
|
||||
const mvtLayerWorker = new Worker();
|
||||
|
||||
function getCircularReplacer() {
|
||||
const seen = new WeakSet();
|
||||
return function(key, value) {
|
||||
if (typeof value === 'object' && value !== null) {
|
||||
if (seen.has(value)) {
|
||||
return '[circular]';
|
||||
}
|
||||
seen.add(value);
|
||||
}
|
||||
return value;
|
||||
};
|
||||
}
|
||||
|
||||
let container, canvas;
|
||||
|
||||
const map = new Map({
|
||||
layers: [
|
||||
new Layer({
|
||||
render: function(frameState) {
|
||||
if (!container) {
|
||||
container = document.createElement('div');
|
||||
container.style.position = 'absolute';
|
||||
container.style.width = '100%';
|
||||
container.style.height = '100%';
|
||||
canvas = document.createElement('canvas');
|
||||
canvas.style.position = 'absolute';
|
||||
canvas.style.left = '0';
|
||||
canvas.style.transformOrigin = 'top left';
|
||||
container.appendChild(canvas);
|
||||
const offscreen = canvas.transferControlToOffscreen();
|
||||
mvtLayerWorker.postMessage({
|
||||
canvas: offscreen
|
||||
}, [offscreen]);
|
||||
}
|
||||
mvtLayerWorker.postMessage({
|
||||
frameState: JSON.parse(JSON.stringify(frameState, getCircularReplacer()))
|
||||
});
|
||||
return container;
|
||||
}
|
||||
})
|
||||
],
|
||||
target: 'map',
|
||||
view: new View({
|
||||
center: [0, 0],
|
||||
zoom: 2
|
||||
})
|
||||
});
|
||||
mvtLayerWorker.addEventListener('message', message => {
|
||||
if (message.data.type === 'render') {
|
||||
map.render();
|
||||
} else if (canvas && message.data.type === 'transform-opacity') {
|
||||
canvas.style.transform = message.data.transform;
|
||||
canvas.style.opacity = message.data.opacity;
|
||||
}
|
||||
});
|
||||
Reference in New Issue
Block a user