Create a basic example for OffscreenCanvas rendering

This commit is contained in:
Andreas Hocevar
2020-03-04 16:06:18 +01:00
parent 8b76f52652
commit 3f5022630b
5 changed files with 170 additions and 0 deletions

View 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;

View 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>

View 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;
}
});

22
package-lock.json generated
View File

@@ -12693,6 +12693,28 @@
"errno": "~0.1.7"
}
},
"worker-loader": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/worker-loader/-/worker-loader-2.0.0.tgz",
"integrity": "sha512-tnvNp4K3KQOpfRnD20m8xltE3eWh89Ye+5oj7wXEEHKac1P4oZ6p9oTj8/8ExqoSBnk9nu5Pr4nKfQ1hn2APJw==",
"dev": true,
"requires": {
"loader-utils": "^1.0.0",
"schema-utils": "^0.4.0"
},
"dependencies": {
"schema-utils": {
"version": "0.4.7",
"resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-0.4.7.tgz",
"integrity": "sha512-v/iwU6wvwGK8HbU9yi3/nhGzP0yGSuhQMzL6ySiec1FSrZZDkhm4noOSWzrNFo/jEc+SJY6jRTwuwbSXJPDUnQ==",
"dev": true,
"requires": {
"ajv": "^6.1.0",
"ajv-keywords": "^3.1.0"
}
}
}
},
"wrap-ansi": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz",

View File

@@ -104,6 +104,7 @@
"webpack-cli": "^3.3.2",
"webpack-dev-middleware": "^3.6.2",
"webpack-dev-server": "^3.3.1",
"worker-loader": "^2.0.0",
"yargs": "^15.0.2"
},
"eslintConfig": {