228 lines
7.1 KiB
JavaScript
228 lines
7.1 KiB
JavaScript
/**
|
|
* @module ol/renderer/canvas/ImageLayer
|
|
*/
|
|
import {ENABLE_RASTER_REPROJECTION} from '../../reproj/common.js';
|
|
import {inherits} from '../../index.js';
|
|
import ImageCanvas from '../../ImageCanvas.js';
|
|
import LayerType from '../../LayerType.js';
|
|
import ViewHint from '../../ViewHint.js';
|
|
import {equals} from '../../array.js';
|
|
import {getHeight, getIntersection, getWidth, isEmpty} from '../../extent.js';
|
|
import VectorRenderType from '../../layer/VectorRenderType.js';
|
|
import {assign} from '../../obj.js';
|
|
import IntermediateCanvasRenderer from '../canvas/IntermediateCanvas.js';
|
|
import {create as createTransform, compose as composeTransform} from '../../transform.js';
|
|
|
|
/**
|
|
* @constructor
|
|
* @extends {module:ol/renderer/canvas/IntermediateCanvas}
|
|
* @param {module:ol/layer/Image} imageLayer Single image layer.
|
|
* @api
|
|
*/
|
|
const CanvasImageLayerRenderer = function(imageLayer) {
|
|
|
|
IntermediateCanvasRenderer.call(this, imageLayer);
|
|
|
|
/**
|
|
* @private
|
|
* @type {?module:ol/ImageBase}
|
|
*/
|
|
this.image_ = null;
|
|
|
|
/**
|
|
* @private
|
|
* @type {module:ol/transform~Transform}
|
|
*/
|
|
this.imageTransform_ = createTransform();
|
|
|
|
/**
|
|
* @type {!Array.<string>}
|
|
*/
|
|
this.skippedFeatures_ = [];
|
|
|
|
/**
|
|
* @private
|
|
* @type {module:ol/renderer/canvas/VectorLayer}
|
|
*/
|
|
this.vectorRenderer_ = null;
|
|
|
|
};
|
|
|
|
inherits(CanvasImageLayerRenderer, IntermediateCanvasRenderer);
|
|
|
|
|
|
/**
|
|
* Determine if this renderer handles the provided layer.
|
|
* @param {module:ol/layer/Layer} layer The candidate layer.
|
|
* @return {boolean} The renderer can render the layer.
|
|
*/
|
|
CanvasImageLayerRenderer['handles'] = function(layer) {
|
|
return layer.getType() === LayerType.IMAGE ||
|
|
layer.getType() === LayerType.VECTOR &&
|
|
/** @type {module:ol/layer/Vector} */ (layer).getRenderMode() === VectorRenderType.IMAGE;
|
|
};
|
|
|
|
|
|
/**
|
|
* Create a layer renderer.
|
|
* @param {module:ol/renderer/Map} mapRenderer The map renderer.
|
|
* @param {module:ol/layer/Layer} layer The layer to be rendererd.
|
|
* @return {module:ol/renderer/canvas/ImageLayer} The layer renderer.
|
|
*/
|
|
CanvasImageLayerRenderer['create'] = function(mapRenderer, layer) {
|
|
const renderer = new CanvasImageLayerRenderer(/** @type {module:ol/layer/Image} */ (layer));
|
|
if (layer.getType() === LayerType.VECTOR) {
|
|
const candidates = mapRenderer.getLayerRendererConstructors();
|
|
for (let i = 0, ii = candidates.length; i < ii; ++i) {
|
|
const candidate = /** @type {Object.<string, Function>} */ (candidates[i]);
|
|
if (candidate !== CanvasImageLayerRenderer && candidate['handles'](layer)) {
|
|
renderer.setVectorRenderer(candidate['create'](mapRenderer, layer));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return renderer;
|
|
};
|
|
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
CanvasImageLayerRenderer.prototype.disposeInternal = function() {
|
|
if (this.vectorRenderer_) {
|
|
this.vectorRenderer_.dispose();
|
|
}
|
|
IntermediateCanvasRenderer.prototype.disposeInternal.call(this);
|
|
};
|
|
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
CanvasImageLayerRenderer.prototype.getImage = function() {
|
|
return !this.image_ ? null : this.image_.getImage();
|
|
};
|
|
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
CanvasImageLayerRenderer.prototype.getImageTransform = function() {
|
|
return this.imageTransform_;
|
|
};
|
|
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
CanvasImageLayerRenderer.prototype.prepareFrame = function(frameState, layerState) {
|
|
|
|
const pixelRatio = frameState.pixelRatio;
|
|
const size = frameState.size;
|
|
const viewState = frameState.viewState;
|
|
const viewCenter = viewState.center;
|
|
const viewResolution = viewState.resolution;
|
|
|
|
let image;
|
|
const imageLayer = /** @type {module:ol/layer/Image} */ (this.getLayer());
|
|
const imageSource = imageLayer.getSource();
|
|
|
|
const hints = frameState.viewHints;
|
|
|
|
let renderedExtent = frameState.extent;
|
|
if (layerState.extent !== undefined) {
|
|
renderedExtent = getIntersection(renderedExtent, layerState.extent);
|
|
}
|
|
|
|
if (!hints[ViewHint.ANIMATING] && !hints[ViewHint.INTERACTING] &&
|
|
!isEmpty(renderedExtent)) {
|
|
let projection = viewState.projection;
|
|
if (!ENABLE_RASTER_REPROJECTION) {
|
|
const sourceProjection = imageSource.getProjection();
|
|
if (sourceProjection) {
|
|
projection = sourceProjection;
|
|
}
|
|
}
|
|
const vectorRenderer = this.vectorRenderer_;
|
|
if (vectorRenderer) {
|
|
const context = vectorRenderer.context;
|
|
const imageFrameState = /** @type {module:ol/PluggableMap~FrameState} */ (assign({}, frameState, {
|
|
size: [
|
|
getWidth(renderedExtent) / viewResolution,
|
|
getHeight(renderedExtent) / viewResolution
|
|
],
|
|
viewState: /** @type {module:ol/View~State} */ (assign({}, frameState.viewState, {
|
|
rotation: 0
|
|
}))
|
|
}));
|
|
const skippedFeatures = Object.keys(imageFrameState.skippedFeatureUids).sort();
|
|
if (vectorRenderer.prepareFrame(imageFrameState, layerState) &&
|
|
(vectorRenderer.replayGroupChanged ||
|
|
!equals(skippedFeatures, this.skippedFeatures_))) {
|
|
context.canvas.width = imageFrameState.size[0] * pixelRatio;
|
|
context.canvas.height = imageFrameState.size[1] * pixelRatio;
|
|
vectorRenderer.composeFrame(imageFrameState, layerState, context);
|
|
this.image_ = new ImageCanvas(renderedExtent, viewResolution, pixelRatio, context.canvas);
|
|
this.skippedFeatures_ = skippedFeatures;
|
|
}
|
|
} else {
|
|
image = imageSource.getImage(
|
|
renderedExtent, viewResolution, pixelRatio, projection);
|
|
if (image) {
|
|
const loaded = this.loadImage(image);
|
|
if (loaded) {
|
|
this.image_ = image;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (this.image_) {
|
|
image = this.image_;
|
|
const imageExtent = image.getExtent();
|
|
const imageResolution = image.getResolution();
|
|
const imagePixelRatio = image.getPixelRatio();
|
|
const scale = pixelRatio * imageResolution /
|
|
(viewResolution * imagePixelRatio);
|
|
const transform = composeTransform(this.imageTransform_,
|
|
pixelRatio * size[0] / 2, pixelRatio * size[1] / 2,
|
|
scale, scale,
|
|
0,
|
|
imagePixelRatio * (imageExtent[0] - viewCenter[0]) / imageResolution,
|
|
imagePixelRatio * (viewCenter[1] - imageExtent[3]) / imageResolution);
|
|
composeTransform(this.coordinateToCanvasPixelTransform,
|
|
pixelRatio * size[0] / 2 - transform[4], pixelRatio * size[1] / 2 - transform[5],
|
|
pixelRatio / viewResolution, -pixelRatio / viewResolution,
|
|
0,
|
|
-viewCenter[0], -viewCenter[1]);
|
|
|
|
this.renderedResolution = imageResolution * pixelRatio / imagePixelRatio;
|
|
}
|
|
|
|
return !!this.image_;
|
|
};
|
|
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
CanvasImageLayerRenderer.prototype.forEachFeatureAtCoordinate = function(coordinate, frameState, hitTolerance, callback, thisArg) {
|
|
if (this.vectorRenderer_) {
|
|
return this.vectorRenderer_.forEachFeatureAtCoordinate(coordinate, frameState, hitTolerance, callback, thisArg);
|
|
} else {
|
|
return IntermediateCanvasRenderer.prototype.forEachFeatureAtCoordinate.call(this, coordinate, frameState, hitTolerance, callback, thisArg);
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* @param {module:ol/renderer/canvas/VectorLayer} renderer Vector renderer.
|
|
*/
|
|
CanvasImageLayerRenderer.prototype.setVectorRenderer = function(renderer) {
|
|
if (this.vectorRenderer_) {
|
|
this.vectorRenderer_.dispose();
|
|
}
|
|
this.vectorRenderer_ = renderer;
|
|
};
|
|
export default CanvasImageLayerRenderer;
|