/** * @module ol/renderer/canvas/VectorTileLayer */ import {getUid} from '../../util.js'; import {createCanvasContext2D} from '../../dom.js'; import TileState from '../../TileState.js'; import ViewHint from '../../ViewHint.js'; import {listen, unlisten, unlistenByKey} from '../../events.js'; import EventType from '../../events/EventType.js'; import rbush from 'rbush'; import {buffer, containsCoordinate, equals, getIntersection, getTopLeft, intersects} from '../../extent.js'; import VectorTileRenderType from '../../layer/VectorTileRenderType.js'; import {equivalent as equivalentProjection} from '../../proj.js'; import Units from '../../proj/Units.js'; import ReplayType from '../../render/canvas/BuilderType.js'; import {labelCache} from '../../render/canvas.js'; import CanvasBuilderGroup from '../../render/canvas/BuilderGroup.js'; import CanvasTileLayerRenderer from './TileLayer.js'; import {getSquaredTolerance as getSquaredRenderTolerance, renderFeature} from '../vector.js'; import { apply as applyTransform, create as createTransform, compose as composeTransform, reset as resetTransform, scale as scaleTransform, translate as translateTransform, toString as transformToString, makeScale, makeInverse } from '../../transform.js'; import CanvasExecutorGroup, {replayDeclutter} from '../../render/canvas/ExecutorGroup.js'; /** * @type {!Object>} */ const IMAGE_REPLAYS = { 'image': [ReplayType.POLYGON, ReplayType.CIRCLE, ReplayType.LINE_STRING, ReplayType.IMAGE, ReplayType.TEXT], 'hybrid': [ReplayType.POLYGON, ReplayType.LINE_STRING] }; /** * @type {!Object>} */ const VECTOR_REPLAYS = { 'image': [ReplayType.DEFAULT], 'hybrid': [ReplayType.IMAGE, ReplayType.TEXT, ReplayType.DEFAULT] }; /** * @classdesc * Canvas renderer for vector tile layers. * @api */ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer { /** * @param {import("../../layer/VectorTile.js").default} layer VectorTile layer. */ constructor(layer) { super(layer); const baseCanvas = this.context.canvas; /** * @private * @type {CanvasRenderingContext2D} */ this.overlayContext_ = createCanvasContext2D(); const overlayCanvas = this.overlayContext_.canvas; overlayCanvas.style.position = 'absolute'; overlayCanvas.style.transformOrigin = 'top left'; const container = document.createElement('div'); const style = container.style; style.position = 'absolute'; style.width = '100%'; style.height = '100%'; container.appendChild(baseCanvas); container.appendChild(overlayCanvas); /** * @private * @type {HTMLElement} */ this.container_ = container; /** * The transform for rendered pixels to viewport CSS pixels for the overlay canvas. * @private * @type {import("../../transform.js").Transform} */ this.overlayPixelTransform_ = createTransform(); /** * The transform for viewport CSS pixels to rendered pixels for the overlay canvas. * @private * @type {import("../../transform.js").Transform} */ this.inverseOverlayPixelTransform_ = createTransform(); /** * Declutter tree. * @private */ this.declutterTree_ = layer.getDeclutter() ? rbush(9, undefined) : null; /** * @private * @type {boolean} */ this.dirty_ = false; /** * @private * @type {number} */ this.renderedLayerRevision_; /** * @private * @type {Object} */ this.tilesToPrepare_ = null; /** * @private * @type {import("../../transform.js").Transform} */ this.tmpTransform_ = createTransform(); // Use closest resolution. this.zDirection = 0; listen(labelCache, EventType.CLEAR, this.handleFontsChanged_, this); } /** * @inheritDoc */ disposeInternal() { unlisten(labelCache, EventType.CLEAR, this.handleFontsChanged_, this); super.disposeInternal(); } /** * @inheritDoc */ getTile(z, x, y, pixelRatio, projection) { const tile = /** @type {import("../../VectorImageTile.js").default} */ (super.getTile(z, x, y, pixelRatio, projection)); if (tile.getState() === TileState.IDLE) { const key = listen(tile, EventType.CHANGE, function() { if (tile.sourceTilesLoaded) { this.updateExecutorGroup_(tile, pixelRatio, projection); if (!this.tilesToPrepare_) { this.tilesToPrepare_ = {}; tile.setState(TileState.LOADED); } else { const tileId = getUid(tile); if (!(tileId in this.tilesToPrepare_)) { this.tilesToPrepare_[tileId] = [tile, pixelRatio, projection]; } } unlistenByKey(key); } }.bind(this)); } if (tile.getState() === TileState.LOADED) { this.updateExecutorGroup_(tile, pixelRatio, projection); this.renderTileImage_(tile, pixelRatio, projection); } return tile; } /** * @inheritDoc */ getTileImage(tile) { const tileLayer = /** @type {import("../../layer/Tile.js").default} */ (this.getLayer()); return tile.getImage(tileLayer); } /** * @inheritDoc */ prepareFrame(frameState, layerState) { const layer = /** @type {import("../../layer/VectorTile.js").default} */ (this.getLayer()); const layerRevision = layer.getRevision(); if (this.renderedLayerRevision_ != layerRevision) { this.renderedTiles.length = 0; } this.renderedLayerRevision_ = layerRevision; return super.prepareFrame(frameState, layerState); } /** * @param {import("../../VectorImageTile.js").default} tile Tile. * @param {number} pixelRatio Pixel ratio. * @param {import("../../proj/Projection.js").default} projection Projection. * @private */ updateExecutorGroup_(tile, pixelRatio, projection) { const layer = /** @type {import("../../layer/Vector.js").default} */ (this.getLayer()); const revision = layer.getRevision(); const renderOrder = /** @type {import("../../render.js").OrderFunction} */ (layer.getRenderOrder()) || null; const builderState = tile.getReplayState(layer); if (!builderState.dirty && builderState.renderedRevision == revision && builderState.renderedRenderOrder == renderOrder) { return; } const source = /** @type {import("../../source/VectorTile.js").default} */ (layer.getSource()); const sourceTileGrid = source.getTileGrid(); const tileGrid = source.getTileGridForProjection(projection); const zoom = tile.tileCoord[0]; const resolution = tileGrid.getResolution(zoom); const tileExtent = tile.extent; for (let t = 0, tt = tile.tileKeys.length; t < tt; ++t) { const sourceTile = tile.getTile(tile.tileKeys[t]); if (sourceTile.getState() != TileState.LOADED) { continue; } const sourceTileCoord = sourceTile.tileCoord; const sourceTileExtent = sourceTileGrid.getTileCoordExtent(sourceTileCoord); const sharedExtent = getIntersection(tileExtent, sourceTileExtent); const bufferedExtent = equals(sourceTileExtent, sharedExtent) ? null : buffer(sharedExtent, layer.getRenderBuffer() * resolution, this.tmpExtent); const tileProjection = sourceTile.getProjection(); let reproject = false; if (!equivalentProjection(projection, tileProjection)) { reproject = true; sourceTile.setProjection(projection); } builderState.dirty = false; const builderGroup = new CanvasBuilderGroup(0, sharedExtent, resolution, pixelRatio, !!this.declutterTree_); const squaredTolerance = getSquaredRenderTolerance(resolution, pixelRatio); /** * @param {import("../../Feature.js").FeatureLike} feature Feature. * @this {CanvasVectorTileLayerRenderer} */ const render = function(feature) { let styles; const styleFunction = feature.getStyleFunction() || layer.getStyleFunction(); if (styleFunction) { styles = styleFunction(feature, resolution); } if (styles) { const dirty = this.renderFeature(feature, squaredTolerance, styles, builderGroup); this.dirty_ = this.dirty_ || dirty; builderState.dirty = builderState.dirty || dirty; } }; const features = sourceTile.getFeatures(); if (renderOrder && renderOrder !== builderState.renderedRenderOrder) { features.sort(renderOrder); } for (let i = 0, ii = features.length; i < ii; ++i) { const feature = features[i]; if (reproject) { if (tileProjection.getUnits() == Units.TILE_PIXELS) { // projected tile extent tileProjection.setWorldExtent(sourceTileExtent); // tile extent in tile pixel space tileProjection.setExtent(sourceTile.getExtent()); } feature.getGeometry().transform(tileProjection, projection); } if (!bufferedExtent || intersects(bufferedExtent, feature.getGeometry().getExtent())) { render.call(this, feature); } } const executorGroupInstructions = builderGroup.finish(); const renderingReplayGroup = new CanvasExecutorGroup(sharedExtent, resolution, pixelRatio, source.getOverlaps(), this.declutterTree_, executorGroupInstructions, layer.getRenderBuffer()); sourceTile.setExecutorGroup(getUid(layer), tile.tileCoord.toString(), renderingReplayGroup); } builderState.renderedRevision = revision; builderState.renderedRenderOrder = renderOrder; } /** * @inheritDoc */ forEachFeatureAtCoordinate(coordinate, frameState, hitTolerance, callback, thisArg) { const resolution = frameState.viewState.resolution; const rotation = frameState.viewState.rotation; hitTolerance = hitTolerance == undefined ? 0 : hitTolerance; const layer = this.getLayer(); /** @type {!Object} */ const features = {}; const renderedTiles = /** @type {Array} */ (this.renderedTiles); let bufferedExtent, found; let i, ii; for (i = 0, ii = renderedTiles.length; i < ii; ++i) { const tile = renderedTiles[i]; bufferedExtent = buffer(tile.extent, hitTolerance * resolution, bufferedExtent); if (!containsCoordinate(bufferedExtent, coordinate)) { continue; } for (let t = 0, tt = tile.tileKeys.length; t < tt; ++t) { const sourceTile = tile.getTile(tile.tileKeys[t]); if (sourceTile.getState() != TileState.LOADED) { continue; } const executorGroup = /** @type {CanvasExecutorGroup} */ (sourceTile.getExecutorGroup(getUid(layer), tile.tileCoord.toString())); found = found || executorGroup.forEachFeatureAtCoordinate(coordinate, resolution, rotation, hitTolerance, {}, /** * @param {import("../../Feature.js").FeatureLike} feature Feature. * @return {?} Callback result. */ function(feature) { const key = getUid(feature); if (!(key in features)) { features[key] = true; return callback.call(thisArg, feature, layer); } }, null); } } return found; } /** * @param {import("../../VectorTile.js").default} tile Tile. * @param {import("../../PluggableMap.js").FrameState} frameState Frame state. * @return {import("../../transform.js").Transform} transform Transform. * @private */ getReplayTransform_(tile, frameState) { const layer = this.getLayer(); const source = /** @type {import("../../source/VectorTile.js").default} */ (layer.getSource()); const tileGrid = source.getTileGrid(); const tileCoord = tile.tileCoord; const tileResolution = tileGrid.getResolution(tileCoord[0]); const viewState = frameState.viewState; const pixelRatio = frameState.pixelRatio; const renderResolution = viewState.resolution / pixelRatio; const tileExtent = tileGrid.getTileCoordExtent(tileCoord, this.tmpExtent); const center = viewState.center; const origin = getTopLeft(tileExtent); const size = frameState.size; const offsetX = Math.round(pixelRatio * size[0] / 2); const offsetY = Math.round(pixelRatio * size[1] / 2); return composeTransform(this.tmpTransform_, offsetX, offsetY, tileResolution / renderResolution, tileResolution / renderResolution, viewState.rotation, (origin[0] - center[0]) / tileResolution, (center[1] - origin[1]) / tileResolution); } /** * @param {import("../../events/Event.js").default} event Event. */ handleFontsChanged_(event) { const layer = this.getLayer(); if (layer.getVisible() && this.renderedLayerRevision_ !== undefined) { layer.changed(); } } /** * Handle changes in image style state. * @param {import("../../events/Event.js").default} event Image style change event. * @private */ handleStyleImageChange_(event) { this.renderIfReadyAndVisible(); } /** * @inheritDoc */ renderFrame(frameState, layerState) { super.renderFrame(frameState, layerState); const layer = /** @type {import("../../layer/VectorTile.js").default} */ (this.getLayer()); const renderMode = layer.getRenderMode(); if (renderMode === VectorTileRenderType.IMAGE) { return this.container_; } const context = this.overlayContext_; const declutterReplays = layer.getDeclutter() ? {} : null; const source = /** @type {import("../../source/VectorTile.js").default} */ (layer.getSource()); const replayTypes = VECTOR_REPLAYS[renderMode]; const pixelRatio = frameState.pixelRatio; const rotation = frameState.viewState.rotation; const size = frameState.size; // set forward and inverse pixel transforms makeScale(this.overlayPixelTransform_, 1 / pixelRatio, 1 / pixelRatio); makeInverse(this.inverseOverlayPixelTransform_, this.overlayPixelTransform_); // resize and clear const canvas = context.canvas; const width = Math.round(size[0] * pixelRatio); const height = Math.round(size[1] * pixelRatio); if (canvas.width != width || canvas.height != height) { canvas.width = width; canvas.height = height; const canvasTransform = transformToString(this.overlayPixelTransform_); if (canvas.style.transform !== canvasTransform) { canvas.style.transform = canvasTransform; } } else { context.clearRect(0, 0, width, height); } if (declutterReplays) { this.declutterTree_.clear(); } const viewHints = frameState.viewHints; const hifi = !(viewHints[ViewHint.ANIMATING] || viewHints[ViewHint.INTERACTING]); const tiles = this.renderedTiles; const tileGrid = source.getTileGridForProjection(frameState.viewState.projection); const clips = []; const zs = []; for (let i = tiles.length - 1; i >= 0; --i) { const tile = /** @type {import("../../VectorImageTile.js").default} */ (tiles[i]); if (tile.getState() == TileState.ABORT) { continue; } const tileCoord = tile.tileCoord; const worldOffset = tileGrid.getTileCoordExtent(tileCoord, this.tmpExtent)[0] - tile.extent[0]; const transform = this.getRenderTransform(frameState, width, height, worldOffset); for (let t = 0, tt = tile.tileKeys.length; t < tt; ++t) { const sourceTile = tile.getTile(tile.tileKeys[t]); if (sourceTile.getState() != TileState.LOADED) { continue; } const executorGroup = /** @type {CanvasExecutorGroup} */ (sourceTile.getExecutorGroup(getUid(layer), tileCoord.toString())); if (!executorGroup || !executorGroup.hasExecutors(replayTypes)) { // sourceTile was not yet loaded when this.createReplayGroup_() was // called, or it has no replays of the types we want to render continue; } const currentZ = sourceTile.tileCoord[0]; const currentClip = executorGroup.getClipCoords(transform); context.save(); // Create a clip mask for regions in this low resolution tile that are // already filled by a higher resolution tile for (let j = 0, jj = clips.length; j < jj; ++j) { const clip = clips[j]; if (currentZ < zs[j]) { context.beginPath(); // counter-clockwise (outer ring) for current tile context.moveTo(currentClip[0], currentClip[1]); context.lineTo(currentClip[2], currentClip[3]); context.lineTo(currentClip[4], currentClip[5]); context.lineTo(currentClip[6], currentClip[7]); // clockwise (inner ring) for higher resolution tile context.moveTo(clip[6], clip[7]); context.lineTo(clip[4], clip[5]); context.lineTo(clip[2], clip[3]); context.lineTo(clip[0], clip[1]); context.clip(); } } executorGroup.execute(context, transform, rotation, {}, hifi, replayTypes, declutterReplays); context.restore(); clips.push(currentClip); zs.push(currentZ); } } if (declutterReplays) { replayDeclutter(declutterReplays, context, rotation, hifi); } const opacity = layerState.opacity; if (opacity !== canvas.style.opacity) { canvas.style.opacity = opacity; } if (this.tilesToPrepare_) { for (const key in this.tilesToPrepare_) { if (!hifi || Date.now() - frameState.time >= 16) { break; } const args = this.tilesToPrepare_[key]; delete this.tilesToPrepare_[key]; const tile = args[0]; if (!tile.disposed) { frameState.animate = true; this.renderTileImage_.apply(this, args); tile.setState(TileState.LOADED); } } if (Object.keys(this.tilesToPrepare_).length > 0) { frameState.animate = true; } else { this.tilesToPrepare_ = null; } } return this.container_; } /** * @param {import("../../Feature.js").FeatureLike} feature Feature. * @param {number} squaredTolerance Squared tolerance. * @param {import("../../style/Style.js").default|Array} styles The style or array of styles. * @param {import("../../render/canvas/BuilderGroup.js").default} executorGroup Replay group. * @return {boolean} `true` if an image is loading. */ renderFeature(feature, squaredTolerance, styles, executorGroup) { if (!styles) { return false; } let loading = false; if (Array.isArray(styles)) { for (let i = 0, ii = styles.length; i < ii; ++i) { loading = renderFeature( executorGroup, feature, styles[i], squaredTolerance, this.handleStyleImageChange_, this) || loading; } } else { loading = renderFeature( executorGroup, feature, styles, squaredTolerance, this.handleStyleImageChange_, this); } return loading; } /** * @param {import("../../VectorImageTile.js").default} tile Tile. * @param {number} pixelRatio Pixel ratio. * @param {import("../../proj/Projection.js").default} projection Projection. * @private */ renderTileImage_(tile, pixelRatio, projection) { const layer = /** @type {import("../../layer/VectorTile.js").default} */ (this.getLayer()); const replayState = tile.getReplayState(layer); const revision = layer.getRevision(); const replays = IMAGE_REPLAYS[layer.getRenderMode()]; if (replays && replayState.renderedTileRevision !== revision) { replayState.renderedTileRevision = revision; const tileCoord = tile.wrappedTileCoord; const z = tileCoord[0]; const source = /** @type {import("../../source/VectorTile.js").default} */ (layer.getSource()); const tileGrid = source.getTileGridForProjection(projection); const resolution = tileGrid.getResolution(z); const context = tile.getContext(layer); const size = source.getTilePixelSize(z, pixelRatio, projection); context.canvas.width = size[0]; context.canvas.height = size[1]; const tileExtent = tileGrid.getTileCoordExtent(tileCoord, this.tmpExtent); for (let i = 0, ii = tile.tileKeys.length; i < ii; ++i) { const sourceTile = tile.getTile(tile.tileKeys[i]); if (sourceTile.getState() != TileState.LOADED) { continue; } const pixelScale = pixelRatio / resolution; const transform = resetTransform(this.tmpTransform_); scaleTransform(transform, pixelScale, -pixelScale); translateTransform(transform, -tileExtent[0], -tileExtent[3]); const executorGroup = /** @type {CanvasExecutorGroup} */ (sourceTile.getExecutorGroup(getUid(layer), tile.tileCoord.toString())); executorGroup.execute(context, transform, 0, {}, true, replays); } } } /** * @inheritdoc */ getDataAtPixel(pixel, frameState, hitTolerance) { let data = super.getDataAtPixel(pixel, frameState, hitTolerance); if (data) { return data; } const renderPixel = applyTransform(this.inverseOverlayPixelTransform_, pixel.slice()); const context = this.overlayContext_; try { data = context.getImageData(Math.round(renderPixel[0]), Math.round(renderPixel[1]), 1, 1).data; } catch (err) { if (err.name === 'SecurityError') { // tainted canvas, we assume there is data at the given pixel (although there might not be) return new Uint8Array(); } return data; } if (data[3] === 0) { return null; } return data; } } export default CanvasVectorTileLayerRenderer;