New decluttering implementation
This commit is contained in:
@@ -144,6 +144,7 @@ worker.addEventListener('message', (event) => {
|
|||||||
renderer.renderFrame(frameState, canvas);
|
renderer.renderFrame(frameState, canvas);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
layers.forEach((layer) => layer.renderDeclutter(frameState));
|
||||||
if (tileQueue.getTilesLoading() < maxTotalLoading) {
|
if (tileQueue.getTilesLoading() < maxTotalLoading) {
|
||||||
tileQueue.reprioritize();
|
tileQueue.reprioritize();
|
||||||
tileQueue.loadMoreTiles(maxTotalLoading, maxNewLoads);
|
tileQueue.loadMoreTiles(maxTotalLoading, maxNewLoads);
|
||||||
|
|||||||
@@ -50,6 +50,7 @@ import {removeNode} from './dom.js';
|
|||||||
* @property {import("./View.js").State} viewState The state of the current view.
|
* @property {import("./View.js").State} viewState The state of the current view.
|
||||||
* @property {boolean} animate
|
* @property {boolean} animate
|
||||||
* @property {import("./transform.js").Transform} coordinateToPixelTransform
|
* @property {import("./transform.js").Transform} coordinateToPixelTransform
|
||||||
|
* @property {import("rbush").default} declutterTree
|
||||||
* @property {null|import("./extent.js").Extent} extent
|
* @property {null|import("./extent.js").Extent} extent
|
||||||
* @property {number} index
|
* @property {number} index
|
||||||
* @property {Array<import("./layer/Layer.js").State>} layerStatesArray
|
* @property {Array<import("./layer/Layer.js").State>} layerStatesArray
|
||||||
@@ -1372,6 +1373,7 @@ class PluggableMap extends BaseObject {
|
|||||||
frameState = {
|
frameState = {
|
||||||
animate: false,
|
animate: false,
|
||||||
coordinateToPixelTransform: this.coordinateToPixelTransform_,
|
coordinateToPixelTransform: this.coordinateToPixelTransform_,
|
||||||
|
declutterTree: null,
|
||||||
extent: getForViewAndSize(
|
extent: getForViewAndSize(
|
||||||
viewState.center,
|
viewState.center,
|
||||||
viewState.resolution,
|
viewState.resolution,
|
||||||
|
|||||||
@@ -45,6 +45,12 @@ class VectorRenderTile extends Tile {
|
|||||||
*/
|
*/
|
||||||
this.executorGroups = {};
|
this.executorGroups = {};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Executor groups for decluttering, by layer uid. Entries are read/written by the renderer.
|
||||||
|
* @type {Object<string, Array<import("./render/canvas/ExecutorGroup.js").default>>}
|
||||||
|
*/
|
||||||
|
this.declutterExecutorGroups = {};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Number of loading source tiles. Read/written by the source.
|
* Number of loading source tiles. Read/written by the source.
|
||||||
* @type {number}
|
* @type {number}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
* @module ol/layer/BaseVector
|
* @module ol/layer/BaseVector
|
||||||
*/
|
*/
|
||||||
import Layer from './Layer.js';
|
import Layer from './Layer.js';
|
||||||
|
import RBush from 'rbush';
|
||||||
import {assign} from '../obj.js';
|
import {assign} from '../obj.js';
|
||||||
import {
|
import {
|
||||||
createDefaultStyle,
|
createDefaultStyle,
|
||||||
@@ -214,6 +215,17 @@ class BaseVectorLayer extends Layer {
|
|||||||
return this.updateWhileInteracting_;
|
return this.updateWhileInteracting_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render declutter items for this layer
|
||||||
|
* @param {import("../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
|
*/
|
||||||
|
renderDeclutter(frameState) {
|
||||||
|
if (!frameState.declutterTree) {
|
||||||
|
frameState.declutterTree = new RBush(9);
|
||||||
|
}
|
||||||
|
/** @type {*} */ (this.getRenderer()).renderDeclutter(frameState);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import("../render.js").OrderFunction|null|undefined} renderOrder
|
* @param {import("../render.js").OrderFunction|null|undefined} renderOrder
|
||||||
* Render order.
|
* Render order.
|
||||||
|
|||||||
@@ -309,6 +309,8 @@ class Heatmap extends VectorLayer {
|
|||||||
],
|
],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
renderDeclutter() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -100,13 +100,15 @@ class VectorContext {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import("../style/Image.js").default} imageStyle Image style.
|
* @param {import("../style/Image.js").default} imageStyle Image style.
|
||||||
|
* @param {Object=} opt_sharedData Shared data for combined decluttering with a text style.
|
||||||
*/
|
*/
|
||||||
setImageStyle(imageStyle) {}
|
setImageStyle(imageStyle, opt_sharedData) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import("../style/Text.js").default} textStyle Text style.
|
* @param {import("../style/Text.js").default} textStyle Text style.
|
||||||
|
* @param {Object=} opt_sharedData Shared data for combined decluttering with an image style.
|
||||||
*/
|
*/
|
||||||
setTextStyle(textStyle) {}
|
setTextStyle(textStyle, opt_sharedData) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default VectorContext;
|
export default VectorContext;
|
||||||
|
|||||||
@@ -67,6 +67,16 @@ import {toString} from '../transform.js';
|
|||||||
* @property {Array<number>} [padding]
|
* @property {Array<number>} [padding]
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} SerializableInstructions
|
||||||
|
* @property {Array<*>} instructions The rendering instructions.
|
||||||
|
* @property {Array<*>} hitDetectionInstructions The rendering hit detection instructions.
|
||||||
|
* @property {Array<number>} coordinates The array of all coordinates.
|
||||||
|
* @property {!Object<string, TextState>} [textStates] The text states (decluttering).
|
||||||
|
* @property {!Object<string, FillState>} [fillStates] The fill states (decluttering).
|
||||||
|
* @property {!Object<string, StrokeState>} [strokeStates] The stroke states (decluttering).
|
||||||
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @const
|
* @const
|
||||||
* @type {string}
|
* @type {string}
|
||||||
@@ -276,9 +286,8 @@ export const measureTextHeight = (function () {
|
|||||||
* @type {HTMLDivElement}
|
* @type {HTMLDivElement}
|
||||||
*/
|
*/
|
||||||
let div;
|
let div;
|
||||||
const heights = textHeights;
|
|
||||||
return function (fontSpec) {
|
return function (fontSpec) {
|
||||||
let height = heights[fontSpec];
|
let height = textHeights[fontSpec];
|
||||||
if (height == undefined) {
|
if (height == undefined) {
|
||||||
if (WORKER_OFFSCREEN_CANVAS) {
|
if (WORKER_OFFSCREEN_CANVAS) {
|
||||||
const font = getFontParameters(fontSpec);
|
const font = getFontParameters(fontSpec);
|
||||||
@@ -286,7 +295,7 @@ export const measureTextHeight = (function () {
|
|||||||
const lineHeight = isNaN(Number(font.lineHeight))
|
const lineHeight = isNaN(Number(font.lineHeight))
|
||||||
? 1.2
|
? 1.2
|
||||||
: Number(font.lineHeight);
|
: Number(font.lineHeight);
|
||||||
textHeights[fontSpec] =
|
height =
|
||||||
lineHeight *
|
lineHeight *
|
||||||
(metrics.actualBoundingBoxAscent + metrics.actualBoundingBoxDescent);
|
(metrics.actualBoundingBoxAscent + metrics.actualBoundingBoxDescent);
|
||||||
} else {
|
} else {
|
||||||
@@ -301,9 +310,9 @@ export const measureTextHeight = (function () {
|
|||||||
div.style.font = fontSpec;
|
div.style.font = fontSpec;
|
||||||
document.body.appendChild(div);
|
document.body.appendChild(div);
|
||||||
height = div.offsetHeight;
|
height = div.offsetHeight;
|
||||||
heights[fontSpec] = height;
|
|
||||||
document.body.removeChild(div);
|
document.body.removeChild(div);
|
||||||
}
|
}
|
||||||
|
textHeights[fontSpec] = height;
|
||||||
}
|
}
|
||||||
return height;
|
return height;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -29,16 +29,6 @@ import {
|
|||||||
inflateMultiCoordinatesArray,
|
inflateMultiCoordinatesArray,
|
||||||
} from '../../geom/flat/inflate.js';
|
} from '../../geom/flat/inflate.js';
|
||||||
|
|
||||||
/**
|
|
||||||
* @typedef {Object} SerializableInstructions
|
|
||||||
* @property {Array<*>} instructions The rendering instructions.
|
|
||||||
* @property {Array<*>} hitDetectionInstructions The rendering hit detection instructions.
|
|
||||||
* @property {Array<number>} coordinates The array of all coordinates.
|
|
||||||
* @property {!Object<string, import("../canvas.js").TextState>} [textStates] The text states (decluttering).
|
|
||||||
* @property {!Object<string, import("../canvas.js").FillState>} [fillStates] The fill states (decluttering).
|
|
||||||
* @property {!Object<string, import("../canvas.js").StrokeState>} [strokeStates] The stroke states (decluttering).
|
|
||||||
*/
|
|
||||||
|
|
||||||
class CanvasBuilder extends VectorContext {
|
class CanvasBuilder extends VectorContext {
|
||||||
/**
|
/**
|
||||||
* @param {number} tolerance Tolerance.
|
* @param {number} tolerance Tolerance.
|
||||||
@@ -383,7 +373,7 @@ class CanvasBuilder extends VectorContext {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return {SerializableInstructions} the serializable instructions.
|
* @return {import("../canvas.js").SerializableInstructions} the serializable instructions.
|
||||||
*/
|
*/
|
||||||
finish() {
|
finish() {
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -28,13 +28,25 @@ import {lineStringLength} from '../../geom/flat/length.js';
|
|||||||
import {transform2D} from '../../geom/flat/transform.js';
|
import {transform2D} from '../../geom/flat/transform.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @typedef {Object} SerializableInstructions
|
* @typedef {Object} BBox
|
||||||
* @property {Array<*>} instructions The rendering instructions.
|
* @property {number} minX
|
||||||
* @property {Array<*>} hitDetectionInstructions The rendering hit detection instructions.
|
* @property {number} minY
|
||||||
* @property {Array<number>} coordinates The array of all coordinates.
|
* @property {number} maxX
|
||||||
* @property {!Object<string, import("../canvas.js").TextState>} textStates The text states (decluttering).
|
* @property {number} maxY
|
||||||
* @property {!Object<string, import("../canvas.js").FillState>} fillStates The fill states (decluttering).
|
* @property {*} value
|
||||||
* @property {!Object<string, import("../canvas.js").StrokeState>} strokeStates The stroke states (decluttering).
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} ImageOrLabelDimensions
|
||||||
|
* @property {number} drawImageX
|
||||||
|
* @property {number} drawImageY
|
||||||
|
* @property {number} drawImageW
|
||||||
|
* @property {number} drawImageH
|
||||||
|
* @property {number} originX
|
||||||
|
* @property {number} originY
|
||||||
|
* @property {Array<number>} scale
|
||||||
|
* @property {BBox} declutterBox
|
||||||
|
* @property {import("../../transform.js").Transform} canvasTransform
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -42,11 +54,6 @@ import {transform2D} from '../../geom/flat/transform.js';
|
|||||||
*/
|
*/
|
||||||
const tmpExtent = createEmpty();
|
const tmpExtent = createEmpty();
|
||||||
|
|
||||||
/**
|
|
||||||
* @type {!import("../../transform.js").Transform}
|
|
||||||
*/
|
|
||||||
const tmpTransform = createTransform();
|
|
||||||
|
|
||||||
/** @type {import("../../coordinate.js").Coordinate} */
|
/** @type {import("../../coordinate.js").Coordinate} */
|
||||||
const p1 = [];
|
const p1 = [];
|
||||||
/** @type {import("../../coordinate.js").Coordinate} */
|
/** @type {import("../../coordinate.js").Coordinate} */
|
||||||
@@ -56,12 +63,21 @@ const p3 = [];
|
|||||||
/** @type {import("../../coordinate.js").Coordinate} */
|
/** @type {import("../../coordinate.js").Coordinate} */
|
||||||
const p4 = [];
|
const p4 = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {Array<*>} replayImageOrLabelArgs Arguments to replayImageOrLabel
|
||||||
|
* @return {BBox} Declutter bbox.
|
||||||
|
*/
|
||||||
|
function getDeclutterBox(replayImageOrLabelArgs) {
|
||||||
|
return /** @type {ImageOrLabelDimensions} */ (replayImageOrLabelArgs[3])
|
||||||
|
.declutterBox;
|
||||||
|
}
|
||||||
|
|
||||||
class Executor {
|
class Executor {
|
||||||
/**
|
/**
|
||||||
* @param {number} resolution Resolution.
|
* @param {number} resolution Resolution.
|
||||||
* @param {number} pixelRatio Pixel ratio.
|
* @param {number} pixelRatio Pixel ratio.
|
||||||
* @param {boolean} overlaps The replay can have overlapping geometries.
|
* @param {boolean} overlaps The replay can have overlapping geometries.
|
||||||
* @param {SerializableInstructions} instructions The serializable instructions
|
* @param {import("../canvas.js").SerializableInstructions} instructions The serializable instructions
|
||||||
* @param {import("../../size.js").Size} renderBuffer Render buffer (width/height) in pixels.
|
* @param {import("../../size.js").Size} renderBuffer Render buffer (width/height) in pixels.
|
||||||
*/
|
*/
|
||||||
constructor(resolution, pixelRatio, overlaps, instructions, renderBuffer) {
|
constructor(resolution, pixelRatio, overlaps, instructions, renderBuffer) {
|
||||||
@@ -293,60 +309,49 @@ class Executor {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @private
|
* @private
|
||||||
* @param {CanvasRenderingContext2D} context Context.
|
* @param {number} sheetWidth Width of the sprite sheet.
|
||||||
* @param {number} contextScale Scale of the context.
|
* @param {number} sheetHeight Height of the sprite sheet.
|
||||||
* @param {number} x X.
|
* @param {number} centerX X.
|
||||||
* @param {number} y Y.
|
* @param {number} centerY Y.
|
||||||
* @param {import("../canvas.js").Label|HTMLImageElement|HTMLCanvasElement|HTMLVideoElement} imageOrLabel Image.
|
* @param {number} width Width.
|
||||||
|
* @param {number} height Height.
|
||||||
* @param {number} anchorX Anchor X.
|
* @param {number} anchorX Anchor X.
|
||||||
* @param {number} anchorY Anchor Y.
|
* @param {number} anchorY Anchor Y.
|
||||||
* @param {number} height Height.
|
|
||||||
* @param {number} opacity Opacity.
|
|
||||||
* @param {number} originX Origin X.
|
* @param {number} originX Origin X.
|
||||||
* @param {number} originY Origin Y.
|
* @param {number} originY Origin Y.
|
||||||
* @param {number} rotation Rotation.
|
* @param {number} rotation Rotation.
|
||||||
* @param {import("../../size.js").Size} scale Scale.
|
* @param {import("../../size.js").Size} scale Scale.
|
||||||
* @param {boolean} snapToPixel Snap to pixel.
|
* @param {boolean} snapToPixel Snap to pixel.
|
||||||
* @param {number} width Width.
|
|
||||||
* @param {Array<number>} padding Padding.
|
* @param {Array<number>} padding Padding.
|
||||||
* @param {Array<*>} fillInstruction Fill instruction.
|
* @param {boolean} fillStroke Background fill or stroke.
|
||||||
* @param {Array<*>} strokeInstruction Stroke instruction.
|
* @param {import("../../Feature.js").FeatureLike} feature Feature.
|
||||||
* @return {boolean} The image or label was rendered.
|
* @return {ImageOrLabelDimensions} Dimensions for positioning and decluttering the image or label.
|
||||||
*/
|
*/
|
||||||
replayImageOrLabel_(
|
calculateImageOrLabelDimensions_(
|
||||||
context,
|
sheetWidth,
|
||||||
contextScale,
|
sheetHeight,
|
||||||
x,
|
centerX,
|
||||||
y,
|
centerY,
|
||||||
imageOrLabel,
|
width,
|
||||||
|
height,
|
||||||
anchorX,
|
anchorX,
|
||||||
anchorY,
|
anchorY,
|
||||||
height,
|
|
||||||
opacity,
|
|
||||||
originX,
|
originX,
|
||||||
originY,
|
originY,
|
||||||
rotation,
|
rotation,
|
||||||
scale,
|
scale,
|
||||||
snapToPixel,
|
snapToPixel,
|
||||||
width,
|
|
||||||
padding,
|
padding,
|
||||||
fillInstruction,
|
fillStroke,
|
||||||
strokeInstruction
|
feature
|
||||||
) {
|
) {
|
||||||
const fillStroke = fillInstruction || strokeInstruction;
|
|
||||||
anchorX *= scale[0];
|
anchorX *= scale[0];
|
||||||
anchorY *= scale[1];
|
anchorY *= scale[1];
|
||||||
x -= anchorX;
|
let x = centerX - anchorX;
|
||||||
y -= anchorY;
|
let y = centerY - anchorY;
|
||||||
|
|
||||||
const w =
|
const w = width + originX > sheetWidth ? sheetWidth - originX : width;
|
||||||
width + originX > imageOrLabel.width
|
const h = height + originY > sheetHeight ? sheetHeight - originY : height;
|
||||||
? imageOrLabel.width - originX
|
|
||||||
: width;
|
|
||||||
const h =
|
|
||||||
height + originY > imageOrLabel.height
|
|
||||||
? imageOrLabel.height - originY
|
|
||||||
: height;
|
|
||||||
const boxW = padding[3] + w * scale[0] + padding[1];
|
const boxW = padding[3] + w * scale[0] + padding[1];
|
||||||
const boxH = padding[0] + h * scale[1] + padding[2];
|
const boxH = padding[0] + h * scale[1] + padding[2];
|
||||||
const boxX = x - padding[3];
|
const boxX = x - padding[3];
|
||||||
@@ -363,12 +368,10 @@ class Executor {
|
|||||||
p4[1] = p3[1];
|
p4[1] = p3[1];
|
||||||
}
|
}
|
||||||
|
|
||||||
let transform = null;
|
let transform;
|
||||||
if (rotation !== 0) {
|
if (rotation !== 0) {
|
||||||
const centerX = x + anchorX;
|
|
||||||
const centerY = y + anchorY;
|
|
||||||
transform = composeTransform(
|
transform = composeTransform(
|
||||||
tmpTransform,
|
createTransform(),
|
||||||
centerX,
|
centerX,
|
||||||
centerY,
|
centerY,
|
||||||
1,
|
1,
|
||||||
@@ -378,10 +381,10 @@ class Executor {
|
|||||||
-centerY
|
-centerY
|
||||||
);
|
);
|
||||||
|
|
||||||
applyTransform(tmpTransform, p1);
|
applyTransform(transform, p1);
|
||||||
applyTransform(tmpTransform, p2);
|
applyTransform(transform, p2);
|
||||||
applyTransform(tmpTransform, p3);
|
applyTransform(transform, p3);
|
||||||
applyTransform(tmpTransform, p4);
|
applyTransform(transform, p4);
|
||||||
createOrUpdate(
|
createOrUpdate(
|
||||||
Math.min(p1[0], p2[0], p3[0], p4[0]),
|
Math.min(p1[0], p2[0], p3[0], p4[0]),
|
||||||
Math.min(p1[1], p2[1], p3[1], p4[1]),
|
Math.min(p1[1], p2[1], p3[1], p4[1]),
|
||||||
@@ -398,24 +401,61 @@ class Executor {
|
|||||||
tmpExtent
|
tmpExtent
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
const renderBufferX = 0; // increase this.renderBuffer_ for decluttering
|
|
||||||
const renderBufferY = 0; // increase this.renderBuffer_ for decluttering
|
|
||||||
const canvas = context.canvas;
|
|
||||||
const strokePadding = strokeInstruction
|
|
||||||
? (strokeInstruction[2] * scale[0]) / 2
|
|
||||||
: 0;
|
|
||||||
const intersects =
|
|
||||||
tmpExtent[0] - strokePadding <=
|
|
||||||
(canvas.width + renderBufferX) / contextScale &&
|
|
||||||
tmpExtent[2] + strokePadding >= -renderBufferX / contextScale &&
|
|
||||||
tmpExtent[1] - strokePadding <=
|
|
||||||
(canvas.height + renderBufferY) / contextScale &&
|
|
||||||
tmpExtent[3] + strokePadding >= -renderBufferY / contextScale;
|
|
||||||
|
|
||||||
if (snapToPixel) {
|
if (snapToPixel) {
|
||||||
x = Math.round(x);
|
x = Math.round(x);
|
||||||
y = Math.round(y);
|
y = Math.round(y);
|
||||||
}
|
}
|
||||||
|
return {
|
||||||
|
drawImageX: x,
|
||||||
|
drawImageY: y,
|
||||||
|
drawImageW: w,
|
||||||
|
drawImageH: h,
|
||||||
|
originX: originX,
|
||||||
|
originY: originY,
|
||||||
|
declutterBox: {
|
||||||
|
minX: tmpExtent[0],
|
||||||
|
minY: tmpExtent[1],
|
||||||
|
maxX: tmpExtent[2],
|
||||||
|
maxY: tmpExtent[3],
|
||||||
|
value: feature,
|
||||||
|
},
|
||||||
|
canvasTransform: transform,
|
||||||
|
scale: scale,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
* @param {CanvasRenderingContext2D} context Context.
|
||||||
|
* @param {number} contextScale Scale of the context.
|
||||||
|
* @param {import("../canvas.js").Label|HTMLImageElement|HTMLCanvasElement|HTMLVideoElement} imageOrLabel Image.
|
||||||
|
* @param {ImageOrLabelDimensions} dimensions Dimensions.
|
||||||
|
* @param {number} opacity Opacity.
|
||||||
|
* @param {Array<*>} fillInstruction Fill instruction.
|
||||||
|
* @param {Array<*>} strokeInstruction Stroke instruction.
|
||||||
|
* @return {boolean} The image or label was rendered.
|
||||||
|
*/
|
||||||
|
replayImageOrLabel_(
|
||||||
|
context,
|
||||||
|
contextScale,
|
||||||
|
imageOrLabel,
|
||||||
|
dimensions,
|
||||||
|
opacity,
|
||||||
|
fillInstruction,
|
||||||
|
strokeInstruction
|
||||||
|
) {
|
||||||
|
const fillStroke = !!(fillInstruction || strokeInstruction);
|
||||||
|
|
||||||
|
const box = dimensions.declutterBox;
|
||||||
|
const canvas = context.canvas;
|
||||||
|
const strokePadding = strokeInstruction
|
||||||
|
? (strokeInstruction[2] * dimensions.scale[0]) / 2
|
||||||
|
: 0;
|
||||||
|
const intersects =
|
||||||
|
box.minX - strokePadding <= canvas.width / contextScale &&
|
||||||
|
box.maxX + strokePadding >= 0 &&
|
||||||
|
box.minY - strokePadding <= canvas.height / contextScale &&
|
||||||
|
box.maxY + strokePadding >= 0;
|
||||||
|
|
||||||
if (intersects) {
|
if (intersects) {
|
||||||
if (fillStroke) {
|
if (fillStroke) {
|
||||||
@@ -431,16 +471,16 @@ class Executor {
|
|||||||
}
|
}
|
||||||
drawImageOrLabel(
|
drawImageOrLabel(
|
||||||
context,
|
context,
|
||||||
transform,
|
dimensions.canvasTransform,
|
||||||
opacity,
|
opacity,
|
||||||
imageOrLabel,
|
imageOrLabel,
|
||||||
originX,
|
dimensions.originX,
|
||||||
originY,
|
dimensions.originY,
|
||||||
w,
|
dimensions.drawImageW,
|
||||||
h,
|
dimensions.drawImageH,
|
||||||
x,
|
dimensions.drawImageX,
|
||||||
y,
|
dimensions.drawImageY,
|
||||||
scale
|
dimensions.scale
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
@@ -470,7 +510,9 @@ class Executor {
|
|||||||
* @param {Array<*>} instruction Instruction.
|
* @param {Array<*>} instruction Instruction.
|
||||||
*/
|
*/
|
||||||
setStrokeStyle_(context, instruction) {
|
setStrokeStyle_(context, instruction) {
|
||||||
context.strokeStyle = /** @type {import("../../colorlike.js").ColorLike} */ (instruction[1]);
|
context[
|
||||||
|
'strokeStyle'
|
||||||
|
] = /** @type {import("../../colorlike.js").ColorLike} */ (instruction[1]);
|
||||||
context.lineWidth = /** @type {number} */ (instruction[2]);
|
context.lineWidth = /** @type {number} */ (instruction[2]);
|
||||||
context.lineCap = /** @type {CanvasLineCap} */ (instruction[3]);
|
context.lineCap = /** @type {CanvasLineCap} */ (instruction[3]);
|
||||||
context.lineJoin = /** @type {CanvasLineJoin} */ (instruction[4]);
|
context.lineJoin = /** @type {CanvasLineJoin} */ (instruction[4]);
|
||||||
@@ -525,6 +567,7 @@ class Executor {
|
|||||||
* @param {function(import("../../Feature.js").FeatureLike): T|undefined} featureCallback Feature callback.
|
* @param {function(import("../../Feature.js").FeatureLike): T|undefined} featureCallback Feature callback.
|
||||||
* @param {import("../../extent.js").Extent=} opt_hitExtent Only check features that intersect this
|
* @param {import("../../extent.js").Extent=} opt_hitExtent Only check features that intersect this
|
||||||
* extent.
|
* extent.
|
||||||
|
* @param {import("rbush").default=} opt_declutterTree Declutter tree.
|
||||||
* @return {T|undefined} Callback result.
|
* @return {T|undefined} Callback result.
|
||||||
* @template T
|
* @template T
|
||||||
*/
|
*/
|
||||||
@@ -535,7 +578,8 @@ class Executor {
|
|||||||
instructions,
|
instructions,
|
||||||
snapToPixel,
|
snapToPixel,
|
||||||
featureCallback,
|
featureCallback,
|
||||||
opt_hitExtent
|
opt_hitExtent,
|
||||||
|
opt_declutterTree
|
||||||
) {
|
) {
|
||||||
/** @type {Array<number>} */
|
/** @type {Array<number>} */
|
||||||
let pixelCoordinates;
|
let pixelCoordinates;
|
||||||
@@ -559,8 +603,17 @@ class Executor {
|
|||||||
const ii = instructions.length; // end of instructions
|
const ii = instructions.length; // end of instructions
|
||||||
let d = 0; // data index
|
let d = 0; // data index
|
||||||
let dd; // end of per-instruction data
|
let dd; // end of per-instruction data
|
||||||
let anchorX, anchorY, prevX, prevY, roundX, roundY, image, text, textKey;
|
let anchorX,
|
||||||
let strokeKey, fillKey;
|
anchorY,
|
||||||
|
prevX,
|
||||||
|
prevY,
|
||||||
|
roundX,
|
||||||
|
roundY,
|
||||||
|
image,
|
||||||
|
text,
|
||||||
|
textKey,
|
||||||
|
strokeKey,
|
||||||
|
fillKey;
|
||||||
let pendingFill = 0;
|
let pendingFill = 0;
|
||||||
let pendingStroke = 0;
|
let pendingStroke = 0;
|
||||||
let lastFillInstruction = null;
|
let lastFillInstruction = null;
|
||||||
@@ -671,13 +724,14 @@ class Executor {
|
|||||||
let rotation = /** @type {number} */ (instruction[11]);
|
let rotation = /** @type {number} */ (instruction[11]);
|
||||||
const scale = /** @type {import("../../size.js").Size} */ (instruction[12]);
|
const scale = /** @type {import("../../size.js").Size} */ (instruction[12]);
|
||||||
let width = /** @type {number} */ (instruction[13]);
|
let width = /** @type {number} */ (instruction[13]);
|
||||||
|
const sharedData = instruction[14];
|
||||||
|
|
||||||
if (!image && instruction.length >= 18) {
|
if (!image && instruction.length >= 19) {
|
||||||
// create label images
|
// create label images
|
||||||
text = /** @type {string} */ (instruction[17]);
|
text = /** @type {string} */ (instruction[18]);
|
||||||
textKey = /** @type {string} */ (instruction[18]);
|
textKey = /** @type {string} */ (instruction[19]);
|
||||||
strokeKey = /** @type {string} */ (instruction[19]);
|
strokeKey = /** @type {string} */ (instruction[20]);
|
||||||
fillKey = /** @type {string} */ (instruction[20]);
|
fillKey = /** @type {string} */ (instruction[21]);
|
||||||
const labelWithAnchor = this.drawLabelWithPointPlacement_(
|
const labelWithAnchor = this.drawLabelWithPointPlacement_(
|
||||||
text,
|
text,
|
||||||
textKey,
|
textKey,
|
||||||
@@ -686,10 +740,10 @@ class Executor {
|
|||||||
);
|
);
|
||||||
image = labelWithAnchor.label;
|
image = labelWithAnchor.label;
|
||||||
instruction[3] = image;
|
instruction[3] = image;
|
||||||
const textOffsetX = /** @type {number} */ (instruction[21]);
|
const textOffsetX = /** @type {number} */ (instruction[22]);
|
||||||
anchorX = (labelWithAnchor.anchorX - textOffsetX) * this.pixelRatio;
|
anchorX = (labelWithAnchor.anchorX - textOffsetX) * this.pixelRatio;
|
||||||
instruction[4] = anchorX;
|
instruction[4] = anchorX;
|
||||||
const textOffsetY = /** @type {number} */ (instruction[22]);
|
const textOffsetY = /** @type {number} */ (instruction[23]);
|
||||||
anchorY = (labelWithAnchor.anchorY - textOffsetY) * this.pixelRatio;
|
anchorY = (labelWithAnchor.anchorY - textOffsetY) * this.pixelRatio;
|
||||||
instruction[5] = anchorY;
|
instruction[5] = anchorY;
|
||||||
height = image.height;
|
height = image.height;
|
||||||
@@ -699,15 +753,15 @@ class Executor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let geometryWidths;
|
let geometryWidths;
|
||||||
if (instruction.length > 23) {
|
if (instruction.length > 24) {
|
||||||
geometryWidths = /** @type {number} */ (instruction[23]);
|
geometryWidths = /** @type {number} */ (instruction[24]);
|
||||||
}
|
}
|
||||||
|
|
||||||
let padding, backgroundFill, backgroundStroke;
|
let padding, backgroundFill, backgroundStroke;
|
||||||
if (instruction.length > 15) {
|
if (instruction.length > 16) {
|
||||||
padding = /** @type {Array<number>} */ (instruction[14]);
|
padding = /** @type {Array<number>} */ (instruction[15]);
|
||||||
backgroundFill = /** @type {boolean} */ (instruction[15]);
|
backgroundFill = /** @type {boolean} */ (instruction[16]);
|
||||||
backgroundStroke = /** @type {boolean} */ (instruction[16]);
|
backgroundStroke = /** @type {boolean} */ (instruction[17]);
|
||||||
} else {
|
} else {
|
||||||
padding = defaultPadding;
|
padding = defaultPadding;
|
||||||
backgroundFill = false;
|
backgroundFill = false;
|
||||||
@@ -729,30 +783,67 @@ class Executor {
|
|||||||
) {
|
) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
this.replayImageOrLabel_(
|
const dimensions = this.calculateImageOrLabelDimensions_(
|
||||||
context,
|
image.width,
|
||||||
contextScale,
|
image.height,
|
||||||
pixelCoordinates[d],
|
pixelCoordinates[d],
|
||||||
pixelCoordinates[d + 1],
|
pixelCoordinates[d + 1],
|
||||||
image,
|
width,
|
||||||
|
height,
|
||||||
anchorX,
|
anchorX,
|
||||||
anchorY,
|
anchorY,
|
||||||
height,
|
|
||||||
opacity,
|
|
||||||
originX,
|
originX,
|
||||||
originY,
|
originY,
|
||||||
rotation,
|
rotation,
|
||||||
scale,
|
scale,
|
||||||
snapToPixel,
|
snapToPixel,
|
||||||
width,
|
|
||||||
padding,
|
padding,
|
||||||
|
backgroundFill || backgroundStroke,
|
||||||
|
feature
|
||||||
|
);
|
||||||
|
const args = [
|
||||||
|
context,
|
||||||
|
contextScale,
|
||||||
|
image,
|
||||||
|
dimensions,
|
||||||
|
opacity,
|
||||||
backgroundFill
|
backgroundFill
|
||||||
? /** @type {Array<*>} */ (lastFillInstruction)
|
? /** @type {Array<*>} */ (lastFillInstruction)
|
||||||
: null,
|
: null,
|
||||||
backgroundStroke
|
backgroundStroke
|
||||||
? /** @type {Array<*>} */ (lastStrokeInstruction)
|
? /** @type {Array<*>} */ (lastStrokeInstruction)
|
||||||
: null
|
: null,
|
||||||
);
|
];
|
||||||
|
let imageArgs;
|
||||||
|
let imageDeclutterBox;
|
||||||
|
if (opt_declutterTree && sharedData) {
|
||||||
|
if (!sharedData[d]) {
|
||||||
|
sharedData[d] = args;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
imageArgs = sharedData[d];
|
||||||
|
delete sharedData[d];
|
||||||
|
imageDeclutterBox = getDeclutterBox(imageArgs);
|
||||||
|
if (opt_declutterTree.collides(imageDeclutterBox)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
opt_declutterTree &&
|
||||||
|
opt_declutterTree.collides(dimensions.declutterBox)
|
||||||
|
) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (imageArgs) {
|
||||||
|
if (opt_declutterTree) {
|
||||||
|
opt_declutterTree.insert(imageDeclutterBox);
|
||||||
|
}
|
||||||
|
this.replayImageOrLabel_.apply(this, imageArgs);
|
||||||
|
}
|
||||||
|
if (opt_declutterTree) {
|
||||||
|
opt_declutterTree.insert(dimensions.declutterBox);
|
||||||
|
}
|
||||||
|
this.replayImageOrLabel_.apply(this, args);
|
||||||
}
|
}
|
||||||
++i;
|
++i;
|
||||||
break;
|
break;
|
||||||
@@ -810,8 +901,8 @@ class Executor {
|
|||||||
cachedWidths,
|
cachedWidths,
|
||||||
viewRotationFromTransform ? 0 : this.viewRotation_
|
viewRotationFromTransform ? 0 : this.viewRotation_
|
||||||
);
|
);
|
||||||
if (parts) {
|
drawChars: if (parts) {
|
||||||
let rendered = false;
|
const replayImageOrLabelArgs = [];
|
||||||
let c, cc, chars, label, part;
|
let c, cc, chars, label, part;
|
||||||
if (strokeKey) {
|
if (strokeKey) {
|
||||||
for (c = 0, cc = parts.length; c < cc; ++c) {
|
for (c = 0, cc = parts.length; c < cc; ++c) {
|
||||||
@@ -824,27 +915,39 @@ class Executor {
|
|||||||
((0.5 - baseline) * 2 * strokeWidth * textScale[1]) /
|
((0.5 - baseline) * 2 * strokeWidth * textScale[1]) /
|
||||||
textScale[0] -
|
textScale[0] -
|
||||||
offsetY;
|
offsetY;
|
||||||
rendered =
|
const dimensions = this.calculateImageOrLabelDimensions_(
|
||||||
this.replayImageOrLabel_(
|
label.width,
|
||||||
context,
|
label.height,
|
||||||
contextScale,
|
part[0],
|
||||||
/** @type {number} */ (part[0]),
|
part[1],
|
||||||
/** @type {number} */ (part[1]),
|
label.width,
|
||||||
label,
|
label.height,
|
||||||
anchorX,
|
anchorX,
|
||||||
anchorY,
|
anchorY,
|
||||||
label.height,
|
0,
|
||||||
1,
|
0,
|
||||||
0,
|
part[3],
|
||||||
0,
|
pixelRatioScale,
|
||||||
/** @type {number} */ (part[3]),
|
false,
|
||||||
pixelRatioScale,
|
defaultPadding,
|
||||||
false,
|
false,
|
||||||
label.width,
|
feature
|
||||||
defaultPadding,
|
);
|
||||||
null,
|
if (
|
||||||
null
|
opt_declutterTree &&
|
||||||
) || rendered;
|
opt_declutterTree.collides(dimensions.declutterBox)
|
||||||
|
) {
|
||||||
|
break drawChars;
|
||||||
|
}
|
||||||
|
replayImageOrLabelArgs.push([
|
||||||
|
context,
|
||||||
|
contextScale,
|
||||||
|
label,
|
||||||
|
dimensions,
|
||||||
|
1,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (fillKey) {
|
if (fillKey) {
|
||||||
@@ -854,29 +957,49 @@ class Executor {
|
|||||||
label = this.createLabel(chars, textKey, fillKey, '');
|
label = this.createLabel(chars, textKey, fillKey, '');
|
||||||
anchorX = /** @type {number} */ (part[2]);
|
anchorX = /** @type {number} */ (part[2]);
|
||||||
anchorY = baseline * label.height - offsetY;
|
anchorY = baseline * label.height - offsetY;
|
||||||
rendered =
|
const dimensions = this.calculateImageOrLabelDimensions_(
|
||||||
this.replayImageOrLabel_(
|
label.width,
|
||||||
context,
|
label.height,
|
||||||
contextScale,
|
part[0],
|
||||||
/** @type {number} */ (part[0]),
|
part[1],
|
||||||
/** @type {number} */ (part[1]),
|
label.width,
|
||||||
label,
|
label.height,
|
||||||
anchorX,
|
anchorX,
|
||||||
anchorY,
|
anchorY,
|
||||||
label.height,
|
0,
|
||||||
1,
|
0,
|
||||||
0,
|
part[3],
|
||||||
0,
|
pixelRatioScale,
|
||||||
/** @type {number} */ (part[3]),
|
false,
|
||||||
pixelRatioScale,
|
defaultPadding,
|
||||||
false,
|
false,
|
||||||
label.width,
|
feature
|
||||||
defaultPadding,
|
);
|
||||||
null,
|
if (
|
||||||
null
|
opt_declutterTree &&
|
||||||
) || rendered;
|
opt_declutterTree.collides(dimensions.declutterBox)
|
||||||
|
) {
|
||||||
|
break drawChars;
|
||||||
|
}
|
||||||
|
replayImageOrLabelArgs.push([
|
||||||
|
context,
|
||||||
|
contextScale,
|
||||||
|
label,
|
||||||
|
dimensions,
|
||||||
|
1,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (opt_declutterTree) {
|
||||||
|
opt_declutterTree.load(
|
||||||
|
replayImageOrLabelArgs.map(getDeclutterBox)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
for (let i = 0, ii = replayImageOrLabelArgs.length; i < ii; ++i) {
|
||||||
|
this.replayImageOrLabel_.apply(this, replayImageOrLabelArgs[i]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
++i;
|
++i;
|
||||||
@@ -977,8 +1100,16 @@ class Executor {
|
|||||||
* @param {import("../../transform.js").Transform} transform Transform.
|
* @param {import("../../transform.js").Transform} transform Transform.
|
||||||
* @param {number} viewRotation View rotation.
|
* @param {number} viewRotation View rotation.
|
||||||
* @param {boolean} snapToPixel Snap point symbols and text to integer pixels.
|
* @param {boolean} snapToPixel Snap point symbols and text to integer pixels.
|
||||||
|
* @param {import("rbush").default=} opt_declutterTree Declutter tree.
|
||||||
*/
|
*/
|
||||||
execute(context, contextScale, transform, viewRotation, snapToPixel) {
|
execute(
|
||||||
|
context,
|
||||||
|
contextScale,
|
||||||
|
transform,
|
||||||
|
viewRotation,
|
||||||
|
snapToPixel,
|
||||||
|
opt_declutterTree
|
||||||
|
) {
|
||||||
this.viewRotation_ = viewRotation;
|
this.viewRotation_ = viewRotation;
|
||||||
this.execute_(
|
this.execute_(
|
||||||
context,
|
context,
|
||||||
@@ -987,7 +1118,8 @@ class Executor {
|
|||||||
this.instructions,
|
this.instructions,
|
||||||
snapToPixel,
|
snapToPixel,
|
||||||
undefined,
|
undefined,
|
||||||
undefined
|
undefined,
|
||||||
|
opt_declutterTree
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ class ExecutorGroup {
|
|||||||
* @param {number} resolution Resolution.
|
* @param {number} resolution Resolution.
|
||||||
* @param {number} pixelRatio Pixel ratio.
|
* @param {number} pixelRatio Pixel ratio.
|
||||||
* @param {boolean} overlaps The executor group can have overlapping geometries.
|
* @param {boolean} overlaps The executor group can have overlapping geometries.
|
||||||
* @param {!Object<string, !Object<import("./BuilderType.js").default, import("./Builder.js").SerializableInstructions>>} allInstructions
|
* @param {!Object<string, !Object<import("./BuilderType.js").default, import("../canvas.js").SerializableInstructions>>} allInstructions
|
||||||
* The serializable instructions.
|
* The serializable instructions.
|
||||||
* @param {number=} opt_renderBuffer Optional rendering buffer.
|
* @param {number=} opt_renderBuffer Optional rendering buffer.
|
||||||
*/
|
*/
|
||||||
@@ -116,7 +116,7 @@ class ExecutorGroup {
|
|||||||
/**
|
/**
|
||||||
* Create executors and populate them using the provided instructions.
|
* Create executors and populate them using the provided instructions.
|
||||||
* @private
|
* @private
|
||||||
* @param {!Object<string, !Object<import("./BuilderType.js").default, import("./Builder.js").SerializableInstructions>>} allInstructions The serializable instructions
|
* @param {!Object<string, !Object<import("./BuilderType.js").default, import("../canvas.js").SerializableInstructions>>} allInstructions The serializable instructions
|
||||||
*/
|
*/
|
||||||
createExecutors_(allInstructions) {
|
createExecutors_(allInstructions) {
|
||||||
for (const zIndex in allInstructions) {
|
for (const zIndex in allInstructions) {
|
||||||
@@ -162,6 +162,7 @@ class ExecutorGroup {
|
|||||||
* @param {number} rotation Rotation.
|
* @param {number} rotation Rotation.
|
||||||
* @param {number} hitTolerance Hit tolerance in pixels.
|
* @param {number} hitTolerance Hit tolerance in pixels.
|
||||||
* @param {function(import("../../Feature.js").FeatureLike): T} callback Feature callback.
|
* @param {function(import("../../Feature.js").FeatureLike): T} callback Feature callback.
|
||||||
|
* @param {Array<import("../../Feature.js").FeatureLike>} declutteredFeatures Decluttered features.
|
||||||
* @return {T|undefined} Callback result.
|
* @return {T|undefined} Callback result.
|
||||||
* @template T
|
* @template T
|
||||||
*/
|
*/
|
||||||
@@ -170,7 +171,8 @@ class ExecutorGroup {
|
|||||||
resolution,
|
resolution,
|
||||||
rotation,
|
rotation,
|
||||||
hitTolerance,
|
hitTolerance,
|
||||||
callback
|
callback,
|
||||||
|
declutteredFeatures
|
||||||
) {
|
) {
|
||||||
hitTolerance = Math.round(hitTolerance);
|
hitTolerance = Math.round(hitTolerance);
|
||||||
const contextSize = hitTolerance * 2 + 1;
|
const contextSize = hitTolerance * 2 + 1;
|
||||||
@@ -232,7 +234,17 @@ class ExecutorGroup {
|
|||||||
for (let j = 0; j < contextSize; j++) {
|
for (let j = 0; j < contextSize; j++) {
|
||||||
if (mask[i][j]) {
|
if (mask[i][j]) {
|
||||||
if (imageData[(j * contextSize + i) * 4 + 3] > 0) {
|
if (imageData[(j * contextSize + i) * 4 + 3] > 0) {
|
||||||
const result = callback(feature);
|
let result;
|
||||||
|
if (
|
||||||
|
!(
|
||||||
|
declutteredFeatures &&
|
||||||
|
(builderType == BuilderType.IMAGE ||
|
||||||
|
builderType == BuilderType.TEXT)
|
||||||
|
) ||
|
||||||
|
declutteredFeatures.indexOf(feature) !== -1
|
||||||
|
) {
|
||||||
|
result = callback(feature);
|
||||||
|
}
|
||||||
if (result) {
|
if (result) {
|
||||||
return result;
|
return result;
|
||||||
} else {
|
} else {
|
||||||
@@ -306,6 +318,7 @@ class ExecutorGroup {
|
|||||||
* @param {boolean} snapToPixel Snap point symbols and test to integer pixel.
|
* @param {boolean} snapToPixel Snap point symbols and test to integer pixel.
|
||||||
* @param {Array<import("./BuilderType.js").default>=} opt_builderTypes Ordered replay types to replay.
|
* @param {Array<import("./BuilderType.js").default>=} opt_builderTypes Ordered replay types to replay.
|
||||||
* Default is {@link module:ol/render/replay~ORDER}
|
* Default is {@link module:ol/render/replay~ORDER}
|
||||||
|
* @param {import("rbush").default=} opt_declutterTree Declutter tree.
|
||||||
*/
|
*/
|
||||||
execute(
|
execute(
|
||||||
context,
|
context,
|
||||||
@@ -313,7 +326,8 @@ class ExecutorGroup {
|
|||||||
transform,
|
transform,
|
||||||
viewRotation,
|
viewRotation,
|
||||||
snapToPixel,
|
snapToPixel,
|
||||||
opt_builderTypes
|
opt_builderTypes,
|
||||||
|
opt_declutterTree
|
||||||
) {
|
) {
|
||||||
/** @type {Array<number>} */
|
/** @type {Array<number>} */
|
||||||
const zs = Object.keys(this.executorsByZIndex_).map(Number);
|
const zs = Object.keys(this.executorsByZIndex_).map(Number);
|
||||||
@@ -328,6 +342,9 @@ class ExecutorGroup {
|
|||||||
|
|
||||||
const builderTypes = opt_builderTypes ? opt_builderTypes : ORDER;
|
const builderTypes = opt_builderTypes ? opt_builderTypes : ORDER;
|
||||||
let i, ii, j, jj, replays, replay;
|
let i, ii, j, jj, replays, replay;
|
||||||
|
if (opt_declutterTree) {
|
||||||
|
zs.reverse();
|
||||||
|
}
|
||||||
for (i = 0, ii = zs.length; i < ii; ++i) {
|
for (i = 0, ii = zs.length; i < ii; ++i) {
|
||||||
const zIndexKey = zs[i].toString();
|
const zIndexKey = zs[i].toString();
|
||||||
replays = this.executorsByZIndex_[zIndexKey];
|
replays = this.executorsByZIndex_[zIndexKey];
|
||||||
@@ -340,7 +357,8 @@ class ExecutorGroup {
|
|||||||
contextScale,
|
contextScale,
|
||||||
transform,
|
transform,
|
||||||
viewRotation,
|
viewRotation,
|
||||||
snapToPixel
|
snapToPixel,
|
||||||
|
opt_declutterTree
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -91,6 +91,13 @@ class CanvasImageBuilder extends CanvasBuilder {
|
|||||||
* @type {number|undefined}
|
* @type {number|undefined}
|
||||||
*/
|
*/
|
||||||
this.width_ = undefined;
|
this.width_ = undefined;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Data shared with a text builder for combined decluttering.
|
||||||
|
* @private
|
||||||
|
* @type {Object}
|
||||||
|
*/
|
||||||
|
this.sharedData_ = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -125,6 +132,7 @@ class CanvasImageBuilder extends CanvasBuilder {
|
|||||||
(this.scale_[1] * this.pixelRatio) / this.imagePixelRatio_,
|
(this.scale_[1] * this.pixelRatio) / this.imagePixelRatio_,
|
||||||
],
|
],
|
||||||
Math.ceil(this.width_ * this.imagePixelRatio_),
|
Math.ceil(this.width_ * this.imagePixelRatio_),
|
||||||
|
this.sharedData_,
|
||||||
]);
|
]);
|
||||||
this.hitDetectionInstructions.push([
|
this.hitDetectionInstructions.push([
|
||||||
CanvasInstruction.DRAW_IMAGE,
|
CanvasInstruction.DRAW_IMAGE,
|
||||||
@@ -142,6 +150,7 @@ class CanvasImageBuilder extends CanvasBuilder {
|
|||||||
this.rotation_,
|
this.rotation_,
|
||||||
this.scale_,
|
this.scale_,
|
||||||
this.width_,
|
this.width_,
|
||||||
|
this.sharedData_,
|
||||||
]);
|
]);
|
||||||
this.endGeometry(feature);
|
this.endGeometry(feature);
|
||||||
}
|
}
|
||||||
@@ -178,6 +187,7 @@ class CanvasImageBuilder extends CanvasBuilder {
|
|||||||
(this.scale_[1] * this.pixelRatio) / this.imagePixelRatio_,
|
(this.scale_[1] * this.pixelRatio) / this.imagePixelRatio_,
|
||||||
],
|
],
|
||||||
Math.ceil(this.width_ * this.imagePixelRatio_),
|
Math.ceil(this.width_ * this.imagePixelRatio_),
|
||||||
|
this.sharedData_,
|
||||||
]);
|
]);
|
||||||
this.hitDetectionInstructions.push([
|
this.hitDetectionInstructions.push([
|
||||||
CanvasInstruction.DRAW_IMAGE,
|
CanvasInstruction.DRAW_IMAGE,
|
||||||
@@ -195,12 +205,13 @@ class CanvasImageBuilder extends CanvasBuilder {
|
|||||||
this.rotation_,
|
this.rotation_,
|
||||||
this.scale_,
|
this.scale_,
|
||||||
this.width_,
|
this.width_,
|
||||||
|
this.sharedData_,
|
||||||
]);
|
]);
|
||||||
this.endGeometry(feature);
|
this.endGeometry(feature);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return {import("./Builder.js").SerializableInstructions} the serializable instructions.
|
* @return {import("../canvas.js").SerializableInstructions} the serializable instructions.
|
||||||
*/
|
*/
|
||||||
finish() {
|
finish() {
|
||||||
this.reverseHitDetectionInstructions();
|
this.reverseHitDetectionInstructions();
|
||||||
@@ -223,8 +234,9 @@ class CanvasImageBuilder extends CanvasBuilder {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import("../../style/Image.js").default} imageStyle Image style.
|
* @param {import("../../style/Image.js").default} imageStyle Image style.
|
||||||
|
* @param {Object=} opt_sharedData Shared data.
|
||||||
*/
|
*/
|
||||||
setImageStyle(imageStyle) {
|
setImageStyle(imageStyle, opt_sharedData) {
|
||||||
const anchor = imageStyle.getAnchor();
|
const anchor = imageStyle.getAnchor();
|
||||||
const size = imageStyle.getSize();
|
const size = imageStyle.getSize();
|
||||||
const hitDetectionImage = imageStyle.getHitDetectionImage();
|
const hitDetectionImage = imageStyle.getHitDetectionImage();
|
||||||
@@ -243,6 +255,7 @@ class CanvasImageBuilder extends CanvasBuilder {
|
|||||||
this.rotation_ = imageStyle.getRotation();
|
this.rotation_ = imageStyle.getRotation();
|
||||||
this.scale_ = imageStyle.getScaleArray();
|
this.scale_ = imageStyle.getScaleArray();
|
||||||
this.width_ = size[0];
|
this.width_ = size[0];
|
||||||
|
this.sharedData_ = opt_sharedData;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -127,7 +127,7 @@ class CanvasLineStringBuilder extends CanvasBuilder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return {import("./Builder.js").SerializableInstructions} the serializable instructions.
|
* @return {import("../canvas.js").SerializableInstructions} the serializable instructions.
|
||||||
*/
|
*/
|
||||||
finish() {
|
finish() {
|
||||||
const state = this.state;
|
const state = this.state;
|
||||||
|
|||||||
@@ -220,7 +220,7 @@ class CanvasPolygonBuilder extends CanvasBuilder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return {import("./Builder.js").SerializableInstructions} the serializable instructions.
|
* @return {import("../canvas.js").SerializableInstructions} the serializable instructions.
|
||||||
*/
|
*/
|
||||||
finish() {
|
finish() {
|
||||||
this.reverseHitDetectionInstructions();
|
this.reverseHitDetectionInstructions();
|
||||||
|
|||||||
@@ -138,10 +138,17 @@ class CanvasTextBuilder extends CanvasBuilder {
|
|||||||
* @type {string}
|
* @type {string}
|
||||||
*/
|
*/
|
||||||
this.strokeKey_ = '';
|
this.strokeKey_ = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Data shared with an image builder for combined decluttering.
|
||||||
|
* @private
|
||||||
|
* @type {Object}
|
||||||
|
*/
|
||||||
|
this.sharedData_ = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return {import("./Builder.js").SerializableInstructions} the serializable instructions.
|
* @return {import("../canvas.js").SerializableInstructions} the serializable instructions.
|
||||||
*/
|
*/
|
||||||
finish() {
|
finish() {
|
||||||
const instructions = super.finish();
|
const instructions = super.finish();
|
||||||
@@ -328,6 +335,7 @@ class CanvasTextBuilder extends CanvasBuilder {
|
|||||||
this.textRotation_,
|
this.textRotation_,
|
||||||
[1, 1],
|
[1, 1],
|
||||||
NaN,
|
NaN,
|
||||||
|
this.sharedData_,
|
||||||
padding == defaultPadding
|
padding == defaultPadding
|
||||||
? defaultPadding
|
? defaultPadding
|
||||||
: padding.map(function (p) {
|
: padding.map(function (p) {
|
||||||
@@ -359,6 +367,7 @@ class CanvasTextBuilder extends CanvasBuilder {
|
|||||||
this.textRotation_,
|
this.textRotation_,
|
||||||
[scale, scale],
|
[scale, scale],
|
||||||
NaN,
|
NaN,
|
||||||
|
this.sharedData_,
|
||||||
padding,
|
padding,
|
||||||
!!textState.backgroundFill,
|
!!textState.backgroundFill,
|
||||||
!!textState.backgroundStroke,
|
!!textState.backgroundStroke,
|
||||||
@@ -475,8 +484,9 @@ class CanvasTextBuilder extends CanvasBuilder {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import("../../style/Text.js").default} textStyle Text style.
|
* @param {import("../../style/Text.js").default} textStyle Text style.
|
||||||
|
* @param {Object=} opt_sharedData Shared data.
|
||||||
*/
|
*/
|
||||||
setTextStyle(textStyle) {
|
setTextStyle(textStyle, opt_sharedData) {
|
||||||
let textState, fillState, strokeState;
|
let textState, fillState, strokeState;
|
||||||
if (!textStyle) {
|
if (!textStyle) {
|
||||||
this.text_ = '';
|
this.text_ = '';
|
||||||
@@ -576,6 +586,7 @@ class CanvasTextBuilder extends CanvasBuilder {
|
|||||||
: '|' + getUid(fillState.fillStyle)
|
: '|' + getUid(fillState.fillStyle)
|
||||||
: '';
|
: '';
|
||||||
}
|
}
|
||||||
|
this.sharedData_ = opt_sharedData;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -102,6 +102,10 @@ class CompositeMapRenderer extends MapRenderer {
|
|||||||
const viewState = frameState.viewState;
|
const viewState = frameState.viewState;
|
||||||
|
|
||||||
this.children_.length = 0;
|
this.children_.length = 0;
|
||||||
|
/**
|
||||||
|
* @type {Array<import("../layer/BaseVector.js").default>}
|
||||||
|
*/
|
||||||
|
const declutterLayers = [];
|
||||||
let previousElement = null;
|
let previousElement = null;
|
||||||
for (let i = 0, ii = layerStatesArray.length; i < ii; ++i) {
|
for (let i = 0, ii = layerStatesArray.length; i < ii; ++i) {
|
||||||
const layerState = layerStatesArray[i];
|
const layerState = layerStatesArray[i];
|
||||||
@@ -123,8 +127,13 @@ class CompositeMapRenderer extends MapRenderer {
|
|||||||
this.children_.push(element);
|
this.children_.push(element);
|
||||||
previousElement = element;
|
previousElement = element;
|
||||||
}
|
}
|
||||||
|
if ('getDeclutter' in layer) {
|
||||||
|
declutterLayers.push(layer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (let i = declutterLayers.length - 1; i >= 0; --i) {
|
||||||
|
declutterLayers[i].renderDeclutter(frameState);
|
||||||
}
|
}
|
||||||
super.renderFrame(frameState);
|
|
||||||
|
|
||||||
replaceChildren(this.element_, this.children_);
|
replaceChildren(this.element_, this.children_);
|
||||||
|
|
||||||
|
|||||||
@@ -25,6 +25,11 @@ class LayerRenderer extends Observable {
|
|||||||
* @type {LayerType}
|
* @type {LayerType}
|
||||||
*/
|
*/
|
||||||
this.layer_ = layer;
|
this.layer_ = layer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @type {import("../render/canvas/ExecutorGroup").default}
|
||||||
|
*/
|
||||||
|
this.declutterExecutorGroup = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -211,9 +211,12 @@ class MapRenderer extends Disposable {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Render.
|
* Render.
|
||||||
|
* @abstract
|
||||||
* @param {?import("../PluggableMap.js").FrameState} frameState Frame state.
|
* @param {?import("../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
*/
|
*/
|
||||||
renderFrame(frameState) {}
|
renderFrame(frameState) {
|
||||||
|
abstract();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import("../PluggableMap.js").FrameState} frameState Frame state.
|
* @param {import("../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import CanvasVectorLayerRenderer from './VectorLayer.js';
|
|||||||
import EventType from '../../events/EventType.js';
|
import EventType from '../../events/EventType.js';
|
||||||
import ImageCanvas from '../../ImageCanvas.js';
|
import ImageCanvas from '../../ImageCanvas.js';
|
||||||
import ImageState from '../../ImageState.js';
|
import ImageState from '../../ImageState.js';
|
||||||
|
import RBush from 'rbush';
|
||||||
import ViewHint from '../../ViewHint.js';
|
import ViewHint from '../../ViewHint.js';
|
||||||
import {apply, compose, create} from '../../transform.js';
|
import {apply, compose, create} from '../../transform.js';
|
||||||
import {assign} from '../../obj.js';
|
import {assign} from '../../obj.js';
|
||||||
@@ -114,6 +115,7 @@ class CanvasVectorImageLayerRenderer extends CanvasImageLayerRenderer {
|
|||||||
{},
|
{},
|
||||||
frameState,
|
frameState,
|
||||||
{
|
{
|
||||||
|
declutterTree: new RBush(9),
|
||||||
extent: renderedExtent,
|
extent: renderedExtent,
|
||||||
size: [width, height],
|
size: [width, height],
|
||||||
viewState: /** @type {import("../../View.js").State} */ (assign(
|
viewState: /** @type {import("../../View.js").State} */ (assign(
|
||||||
@@ -137,6 +139,7 @@ class CanvasVectorImageLayerRenderer extends CanvasImageLayerRenderer {
|
|||||||
) {
|
) {
|
||||||
vectorRenderer.clipping = false;
|
vectorRenderer.clipping = false;
|
||||||
vectorRenderer.renderFrame(imageFrameState, null);
|
vectorRenderer.renderFrame(imageFrameState, null);
|
||||||
|
vectorRenderer.renderDeclutter(imageFrameState);
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -183,6 +186,10 @@ class CanvasVectorImageLayerRenderer extends CanvasImageLayerRenderer {
|
|||||||
*/
|
*/
|
||||||
postRender() {}
|
postRender() {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*/
|
||||||
|
renderDeclutter() {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import("../../coordinate.js").Coordinate} coordinate Coordinate.
|
* @param {import("../../coordinate.js").Coordinate} coordinate Coordinate.
|
||||||
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
|
|||||||
@@ -128,6 +128,11 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
*/
|
*/
|
||||||
this.replayGroupChanged = true;
|
this.replayGroupChanged = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @type {import("../../render/canvas/ExecutorGroup").default}
|
||||||
|
*/
|
||||||
|
this.declutterExecutorGroup = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Clipping to be performed by `renderFrame()`
|
* Clipping to be performed by `renderFrame()`
|
||||||
* @type {boolean}
|
* @type {boolean}
|
||||||
@@ -148,6 +153,73 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
super.useContainer(target, transform, opacity);
|
super.useContainer(target, transform, opacity);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {ExecutorGroup} executorGroup Executor group.
|
||||||
|
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
|
* @param {import("rbush").default=} opt_declutterTree Declutter tree.
|
||||||
|
*/
|
||||||
|
renderWorlds(executorGroup, frameState, opt_declutterTree) {
|
||||||
|
const extent = frameState.extent;
|
||||||
|
const viewState = frameState.viewState;
|
||||||
|
const center = viewState.center;
|
||||||
|
const resolution = viewState.resolution;
|
||||||
|
const projection = viewState.projection;
|
||||||
|
const rotation = viewState.rotation;
|
||||||
|
const projectionExtent = projection.getExtent();
|
||||||
|
const vectorSource = this.getLayer().getSource();
|
||||||
|
const pixelRatio = frameState.pixelRatio;
|
||||||
|
const viewHints = frameState.viewHints;
|
||||||
|
const snapToPixel = !(
|
||||||
|
viewHints[ViewHint.ANIMATING] || viewHints[ViewHint.INTERACTING]
|
||||||
|
);
|
||||||
|
const context = this.context;
|
||||||
|
const width = Math.round(frameState.size[0] * pixelRatio);
|
||||||
|
const height = Math.round(frameState.size[1] * pixelRatio);
|
||||||
|
|
||||||
|
const multiWorld = vectorSource.getWrapX() && projection.canWrapX();
|
||||||
|
const worldWidth = multiWorld ? getWidth(projectionExtent) : null;
|
||||||
|
const endWorld = multiWorld
|
||||||
|
? Math.ceil((extent[2] - projectionExtent[2]) / worldWidth) + 1
|
||||||
|
: 1;
|
||||||
|
let world = multiWorld
|
||||||
|
? Math.floor((extent[0] - projectionExtent[0]) / worldWidth)
|
||||||
|
: 0;
|
||||||
|
do {
|
||||||
|
const transform = this.getRenderTransform(
|
||||||
|
center,
|
||||||
|
resolution,
|
||||||
|
rotation,
|
||||||
|
pixelRatio,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
world * worldWidth
|
||||||
|
);
|
||||||
|
executorGroup.execute(
|
||||||
|
context,
|
||||||
|
1,
|
||||||
|
transform,
|
||||||
|
rotation,
|
||||||
|
snapToPixel,
|
||||||
|
undefined,
|
||||||
|
opt_declutterTree
|
||||||
|
);
|
||||||
|
} while (++world < endWorld);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render declutter items for this layer
|
||||||
|
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
|
*/
|
||||||
|
renderDeclutter(frameState) {
|
||||||
|
if (this.declutterExecutorGroup) {
|
||||||
|
this.renderWorlds(
|
||||||
|
this.declutterExecutorGroup,
|
||||||
|
frameState,
|
||||||
|
frameState.declutterTree
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Render the layer.
|
* Render the layer.
|
||||||
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
@@ -169,7 +241,11 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
const canvas = context.canvas;
|
const canvas = context.canvas;
|
||||||
|
|
||||||
const replayGroup = this.replayGroup_;
|
const replayGroup = this.replayGroup_;
|
||||||
if (!replayGroup || replayGroup.isEmpty()) {
|
const declutterExecutorGroup = this.declutterExecutorGroup;
|
||||||
|
if (
|
||||||
|
(!replayGroup || replayGroup.isEmpty()) &&
|
||||||
|
(!declutterExecutorGroup || declutterExecutorGroup.isEmpty())
|
||||||
|
) {
|
||||||
if (!this.containerReused && canvas.width > 0) {
|
if (!this.containerReused && canvas.width > 0) {
|
||||||
canvas.width = 0;
|
canvas.width = 0;
|
||||||
}
|
}
|
||||||
@@ -191,14 +267,8 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
|
|
||||||
this.preRender(context, frameState);
|
this.preRender(context, frameState);
|
||||||
|
|
||||||
const extent = frameState.extent;
|
|
||||||
const viewState = frameState.viewState;
|
const viewState = frameState.viewState;
|
||||||
const center = viewState.center;
|
|
||||||
const resolution = viewState.resolution;
|
|
||||||
const projection = viewState.projection;
|
const projection = viewState.projection;
|
||||||
const rotation = viewState.rotation;
|
|
||||||
const projectionExtent = projection.getExtent();
|
|
||||||
const vectorSource = this.getLayer().getSource();
|
|
||||||
|
|
||||||
// clipped rendering if layer extent is set
|
// clipped rendering if layer extent is set
|
||||||
let clipped = false;
|
let clipped = false;
|
||||||
@@ -212,38 +282,7 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const viewHints = frameState.viewHints;
|
this.renderWorlds(replayGroup, frameState);
|
||||||
const snapToPixel = !(
|
|
||||||
viewHints[ViewHint.ANIMATING] || viewHints[ViewHint.INTERACTING]
|
|
||||||
);
|
|
||||||
|
|
||||||
const multiWorld = vectorSource.getWrapX() && projection.canWrapX();
|
|
||||||
const worldWidth = multiWorld ? getWidth(projectionExtent) : null;
|
|
||||||
const endWorld = multiWorld
|
|
||||||
? Math.ceil((extent[2] - projectionExtent[2]) / worldWidth) + 1
|
|
||||||
: 1;
|
|
||||||
let world = multiWorld
|
|
||||||
? Math.floor((extent[0] - projectionExtent[0]) / worldWidth)
|
|
||||||
: 0;
|
|
||||||
do {
|
|
||||||
const transform = this.getRenderTransform(
|
|
||||||
center,
|
|
||||||
resolution,
|
|
||||||
rotation,
|
|
||||||
pixelRatio,
|
|
||||||
width,
|
|
||||||
height,
|
|
||||||
world * worldWidth
|
|
||||||
);
|
|
||||||
replayGroup.execute(
|
|
||||||
context,
|
|
||||||
1,
|
|
||||||
transform,
|
|
||||||
rotation,
|
|
||||||
snapToPixel,
|
|
||||||
undefined
|
|
||||||
);
|
|
||||||
} while (++world < endWorld);
|
|
||||||
|
|
||||||
if (clipped) {
|
if (clipped) {
|
||||||
context.restore();
|
context.restore();
|
||||||
@@ -378,26 +417,41 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
const resolution = frameState.viewState.resolution;
|
const resolution = frameState.viewState.resolution;
|
||||||
const rotation = frameState.viewState.rotation;
|
const rotation = frameState.viewState.rotation;
|
||||||
const layer = this.getLayer();
|
const layer = this.getLayer();
|
||||||
|
|
||||||
/** @type {!Object<string, boolean>} */
|
/** @type {!Object<string, boolean>} */
|
||||||
const features = {};
|
const features = {};
|
||||||
|
|
||||||
const result = this.replayGroup_.forEachFeatureAtCoordinate(
|
/**
|
||||||
coordinate,
|
* @param {import("../../Feature.js").FeatureLike} feature Feature.
|
||||||
resolution,
|
* @return {?} Callback result.
|
||||||
rotation,
|
*/
|
||||||
hitTolerance,
|
const featureCallback = function (feature) {
|
||||||
/**
|
const key = getUid(feature);
|
||||||
* @param {import("../../Feature.js").FeatureLike} feature Feature.
|
if (!(key in features)) {
|
||||||
* @return {?} Callback result.
|
features[key] = true;
|
||||||
*/
|
return callback(feature, layer);
|
||||||
function (feature) {
|
|
||||||
const key = getUid(feature);
|
|
||||||
if (!(key in features)) {
|
|
||||||
features[key] = true;
|
|
||||||
return callback(feature, layer);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
);
|
};
|
||||||
|
|
||||||
|
let result;
|
||||||
|
const executorGroups = [this.replayGroup_];
|
||||||
|
if (this.declutterExecutorGroup) {
|
||||||
|
executorGroups.push(this.declutterExecutorGroup);
|
||||||
|
}
|
||||||
|
executorGroups.forEach((executorGroup) => {
|
||||||
|
result =
|
||||||
|
result ||
|
||||||
|
executorGroup.forEachFeatureAtCoordinate(
|
||||||
|
coordinate,
|
||||||
|
resolution,
|
||||||
|
rotation,
|
||||||
|
hitTolerance,
|
||||||
|
featureCallback,
|
||||||
|
executorGroup === this.declutterExecutorGroup
|
||||||
|
? frameState.declutterTree.all().map((item) => item.value)
|
||||||
|
: null
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -531,6 +585,16 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
pixelRatio
|
pixelRatio
|
||||||
);
|
);
|
||||||
|
|
||||||
|
let declutterBuilderGroup;
|
||||||
|
if (this.getLayer().getDeclutter()) {
|
||||||
|
declutterBuilderGroup = new CanvasBuilderGroup(
|
||||||
|
getRenderTolerance(resolution, pixelRatio),
|
||||||
|
extent,
|
||||||
|
resolution,
|
||||||
|
pixelRatio
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const userProjection = getUserProjection();
|
const userProjection = getUserProjection();
|
||||||
let userTransform;
|
let userTransform;
|
||||||
if (userProjection) {
|
if (userProjection) {
|
||||||
@@ -568,7 +632,8 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
styles,
|
styles,
|
||||||
replayGroup,
|
replayGroup,
|
||||||
userTransform
|
userTransform,
|
||||||
|
declutterBuilderGroup
|
||||||
);
|
);
|
||||||
this.dirty_ = this.dirty_ || dirty;
|
this.dirty_ = this.dirty_ || dirty;
|
||||||
}
|
}
|
||||||
@@ -595,6 +660,17 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
vectorLayer.getRenderBuffer()
|
vectorLayer.getRenderBuffer()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (declutterBuilderGroup) {
|
||||||
|
this.declutterExecutorGroup = new ExecutorGroup(
|
||||||
|
extent,
|
||||||
|
resolution,
|
||||||
|
pixelRatio,
|
||||||
|
vectorSource.getOverlaps(),
|
||||||
|
declutterBuilderGroup.finish(),
|
||||||
|
vectorLayer.getRenderBuffer()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
this.renderedResolution_ = resolution;
|
this.renderedResolution_ = resolution;
|
||||||
this.renderedRevision_ = vectorLayerRevision;
|
this.renderedRevision_ = vectorLayerRevision;
|
||||||
this.renderedRenderOrder_ = vectorLayerRenderOrder;
|
this.renderedRenderOrder_ = vectorLayerRenderOrder;
|
||||||
@@ -614,6 +690,7 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
* @param {import("../../style/Style.js").default|Array<import("../../style/Style.js").default>} styles The style or array of styles.
|
* @param {import("../../style/Style.js").default|Array<import("../../style/Style.js").default>} styles The style or array of styles.
|
||||||
* @param {import("../../render/canvas/BuilderGroup.js").default} builderGroup Builder group.
|
* @param {import("../../render/canvas/BuilderGroup.js").default} builderGroup Builder group.
|
||||||
* @param {import("../../proj.js").TransformFunction=} opt_transform Transform from user to view projection.
|
* @param {import("../../proj.js").TransformFunction=} opt_transform Transform from user to view projection.
|
||||||
|
* @param {import("../../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
* @return {boolean} `true` if an image is loading.
|
* @return {boolean} `true` if an image is loading.
|
||||||
*/
|
*/
|
||||||
renderFeature(
|
renderFeature(
|
||||||
@@ -621,7 +698,8 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
styles,
|
styles,
|
||||||
builderGroup,
|
builderGroup,
|
||||||
opt_transform
|
opt_transform,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
) {
|
) {
|
||||||
if (!styles) {
|
if (!styles) {
|
||||||
return false;
|
return false;
|
||||||
@@ -636,7 +714,8 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
styles[i],
|
styles[i],
|
||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
this.boundHandleStyleImageChange_,
|
this.boundHandleStyleImageChange_,
|
||||||
opt_transform
|
opt_transform,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
) || loading;
|
) || loading;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -646,7 +725,8 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
styles,
|
styles,
|
||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
this.boundHandleStyleImageChange_,
|
this.boundHandleStyleImageChange_,
|
||||||
opt_transform
|
opt_transform,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return loading;
|
return loading;
|
||||||
|
|||||||
@@ -260,6 +260,7 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const source = layer.getSource();
|
const source = layer.getSource();
|
||||||
|
const declutter = layer.getDeclutter();
|
||||||
const sourceTileGrid = source.getTileGrid();
|
const sourceTileGrid = source.getTileGrid();
|
||||||
const tileGrid = source.getTileGridForProjection(projection);
|
const tileGrid = source.getTileGridForProjection(projection);
|
||||||
const tileExtent = tileGrid.getTileCoordExtent(tile.wrappedTileCoord);
|
const tileExtent = tileGrid.getTileCoordExtent(tile.wrappedTileCoord);
|
||||||
@@ -268,6 +269,9 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
const layerUid = getUid(layer);
|
const layerUid = getUid(layer);
|
||||||
delete tile.hitDetectionImageData[layerUid];
|
delete tile.hitDetectionImageData[layerUid];
|
||||||
tile.executorGroups[layerUid] = [];
|
tile.executorGroups[layerUid] = [];
|
||||||
|
if (declutter) {
|
||||||
|
tile.declutterExecutorGroups[layerUid] = [];
|
||||||
|
}
|
||||||
for (let t = 0, tt = sourceTiles.length; t < tt; ++t) {
|
for (let t = 0, tt = sourceTiles.length; t < tt; ++t) {
|
||||||
const sourceTile = sourceTiles[t];
|
const sourceTile = sourceTiles[t];
|
||||||
if (sourceTile.getState() != TileState.LOADED) {
|
if (sourceTile.getState() != TileState.LOADED) {
|
||||||
@@ -292,6 +296,9 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
resolution,
|
resolution,
|
||||||
pixelRatio
|
pixelRatio
|
||||||
);
|
);
|
||||||
|
const declutterBuilderGroup = declutter
|
||||||
|
? new CanvasBuilderGroup(0, sharedExtent, resolution, pixelRatio)
|
||||||
|
: undefined;
|
||||||
const squaredTolerance = getSquaredRenderTolerance(
|
const squaredTolerance = getSquaredRenderTolerance(
|
||||||
resolution,
|
resolution,
|
||||||
pixelRatio
|
pixelRatio
|
||||||
@@ -313,7 +320,8 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
feature,
|
feature,
|
||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
styles,
|
styles,
|
||||||
builderGroup
|
builderGroup,
|
||||||
|
declutterBuilderGroup
|
||||||
);
|
);
|
||||||
this.dirty_ = this.dirty_ || dirty;
|
this.dirty_ = this.dirty_ || dirty;
|
||||||
builderState.dirty = builderState.dirty || dirty;
|
builderState.dirty = builderState.dirty || dirty;
|
||||||
@@ -337,7 +345,7 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
// no need to clip when the render tile is covered by a single source tile
|
// no need to clip when the render tile is covered by a single source tile
|
||||||
const replayExtent =
|
const replayExtent =
|
||||||
layer.getRenderMode() !== VectorTileRenderType.VECTOR &&
|
layer.getRenderMode() !== VectorTileRenderType.VECTOR &&
|
||||||
layer.getDeclutter() &&
|
declutter &&
|
||||||
sourceTiles.length === 1
|
sourceTiles.length === 1
|
||||||
? null
|
? null
|
||||||
: sharedExtent;
|
: sharedExtent;
|
||||||
@@ -350,6 +358,17 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
layer.getRenderBuffer()
|
layer.getRenderBuffer()
|
||||||
);
|
);
|
||||||
tile.executorGroups[layerUid].push(renderingReplayGroup);
|
tile.executorGroups[layerUid].push(renderingReplayGroup);
|
||||||
|
if (declutterBuilderGroup) {
|
||||||
|
const declutterExecutorGroup = new CanvasExecutorGroup(
|
||||||
|
replayExtent,
|
||||||
|
resolution,
|
||||||
|
pixelRatio,
|
||||||
|
source.getOverlaps(),
|
||||||
|
declutterBuilderGroup.finish(),
|
||||||
|
layer.getRenderBuffer()
|
||||||
|
);
|
||||||
|
tile.declutterExecutorGroups[layerUid].push(declutterExecutorGroup);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
builderState.renderedRevision = revision;
|
builderState.renderedRevision = revision;
|
||||||
builderState.renderedZ = tile.sourceZ;
|
builderState.renderedZ = tile.sourceZ;
|
||||||
@@ -395,34 +414,44 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const executorGroups = tile.executorGroups[getUid(layer)];
|
const layerUid = getUid(layer);
|
||||||
for (let t = 0, tt = executorGroups.length; t < tt; ++t) {
|
const executorGroups = [tile.executorGroups[layerUid]];
|
||||||
const executorGroup = executorGroups[t];
|
const declutterExecutorGroups = tile.declutterExecutorGroups[layerUid];
|
||||||
found =
|
if (declutterExecutorGroups) {
|
||||||
found ||
|
executorGroups.push(declutterExecutorGroups);
|
||||||
executorGroup.forEachFeatureAtCoordinate(
|
|
||||||
coordinate,
|
|
||||||
resolution,
|
|
||||||
rotation,
|
|
||||||
hitTolerance,
|
|
||||||
/**
|
|
||||||
* @param {import("../../Feature.js").FeatureLike} feature Feature.
|
|
||||||
* @return {?} Callback result.
|
|
||||||
*/
|
|
||||||
function (feature) {
|
|
||||||
if (tileContainsCoordinate) {
|
|
||||||
let key = feature.getId();
|
|
||||||
if (key === undefined) {
|
|
||||||
key = getUid(feature);
|
|
||||||
}
|
|
||||||
if (!(key in features)) {
|
|
||||||
features[key] = true;
|
|
||||||
return callback(feature, layer);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
executorGroups.forEach((executorGroups) => {
|
||||||
|
for (let t = 0, tt = executorGroups.length; t < tt; ++t) {
|
||||||
|
const executorGroup = executorGroups[t];
|
||||||
|
found =
|
||||||
|
found ||
|
||||||
|
executorGroup.forEachFeatureAtCoordinate(
|
||||||
|
coordinate,
|
||||||
|
resolution,
|
||||||
|
rotation,
|
||||||
|
hitTolerance,
|
||||||
|
/**
|
||||||
|
* @param {import("../../Feature.js").FeatureLike} feature Feature.
|
||||||
|
* @return {?} Callback result.
|
||||||
|
*/
|
||||||
|
function (feature) {
|
||||||
|
if (tileContainsCoordinate) {
|
||||||
|
let key = feature.getId();
|
||||||
|
if (key === undefined) {
|
||||||
|
key = getUid(feature);
|
||||||
|
}
|
||||||
|
if (!(key in features)) {
|
||||||
|
features[key] = true;
|
||||||
|
return callback(feature, layer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
executorGroups === declutterExecutorGroups
|
||||||
|
? frameState.declutterTree.all().map((item) => item.value)
|
||||||
|
: null
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
return found;
|
return found;
|
||||||
}
|
}
|
||||||
@@ -539,6 +568,70 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
this.renderIfReadyAndVisible();
|
this.renderIfReadyAndVisible();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render declutter items for this layer
|
||||||
|
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
|
*/
|
||||||
|
renderDeclutter(frameState) {
|
||||||
|
const viewHints = frameState.viewHints;
|
||||||
|
const hifi = !(
|
||||||
|
viewHints[ViewHint.ANIMATING] || viewHints[ViewHint.INTERACTING]
|
||||||
|
);
|
||||||
|
const tiles = /** @type {Array<import("../../VectorRenderTile.js").default>} */ (this
|
||||||
|
.renderedTiles);
|
||||||
|
for (let i = 0, ii = tiles.length; i < ii; ++i) {
|
||||||
|
const tile = tiles[i];
|
||||||
|
const declutterExecutorGroups =
|
||||||
|
tile.declutterExecutorGroups[getUid(this.getLayer())];
|
||||||
|
if (declutterExecutorGroups) {
|
||||||
|
for (let j = declutterExecutorGroups.length - 1; j >= 0; --j) {
|
||||||
|
declutterExecutorGroups[j].execute(
|
||||||
|
this.context,
|
||||||
|
1,
|
||||||
|
this.getTileRenderTransform(tile, frameState),
|
||||||
|
frameState.viewState.rotation,
|
||||||
|
hifi,
|
||||||
|
undefined,
|
||||||
|
frameState.declutterTree
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getTileRenderTransform(tile, frameState) {
|
||||||
|
const pixelRatio = frameState.pixelRatio;
|
||||||
|
const viewState = frameState.viewState;
|
||||||
|
const center = viewState.center;
|
||||||
|
const resolution = viewState.resolution;
|
||||||
|
const rotation = viewState.rotation;
|
||||||
|
const size = frameState.size;
|
||||||
|
const width = Math.round(size[0] * pixelRatio);
|
||||||
|
const height = Math.round(size[1] * pixelRatio);
|
||||||
|
|
||||||
|
const source = this.getLayer().getSource();
|
||||||
|
const tileGrid = source.getTileGridForProjection(
|
||||||
|
frameState.viewState.projection
|
||||||
|
);
|
||||||
|
const tileCoord = tile.tileCoord;
|
||||||
|
const tileExtent = tileGrid.getTileCoordExtent(tile.wrappedTileCoord);
|
||||||
|
const worldOffset =
|
||||||
|
tileGrid.getTileCoordExtent(tileCoord, this.tmpExtent)[0] - tileExtent[0];
|
||||||
|
const transform = multiply(
|
||||||
|
scale(this.inversePixelTransform.slice(), 1 / pixelRatio, 1 / pixelRatio),
|
||||||
|
this.getRenderTransform(
|
||||||
|
center,
|
||||||
|
resolution,
|
||||||
|
rotation,
|
||||||
|
pixelRatio,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
worldOffset
|
||||||
|
)
|
||||||
|
);
|
||||||
|
return transform;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Render the layer.
|
* Render the layer.
|
||||||
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
* @param {import("../../PluggableMap.js").FrameState} frameState Frame state.
|
||||||
@@ -573,47 +666,17 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
|
|
||||||
const context = this.context;
|
const context = this.context;
|
||||||
const replayTypes = VECTOR_REPLAYS[renderMode];
|
const replayTypes = VECTOR_REPLAYS[renderMode];
|
||||||
const pixelRatio = frameState.pixelRatio;
|
|
||||||
const viewState = frameState.viewState;
|
const viewState = frameState.viewState;
|
||||||
const center = viewState.center;
|
|
||||||
const resolution = viewState.resolution;
|
|
||||||
const rotation = viewState.rotation;
|
const rotation = viewState.rotation;
|
||||||
const size = frameState.size;
|
|
||||||
|
|
||||||
const width = Math.round(size[0] * pixelRatio);
|
|
||||||
const height = Math.round(size[1] * pixelRatio);
|
|
||||||
|
|
||||||
const tiles = this.renderedTiles;
|
const tiles = this.renderedTiles;
|
||||||
const tileGrid = source.getTileGridForProjection(
|
|
||||||
frameState.viewState.projection
|
|
||||||
);
|
|
||||||
const clips = [];
|
const clips = [];
|
||||||
const clipZs = [];
|
const clipZs = [];
|
||||||
for (let i = tiles.length - 1; i >= 0; --i) {
|
for (let i = tiles.length - 1; i >= 0; --i) {
|
||||||
const tile = /** @type {import("../../VectorRenderTile.js").default} */ (tiles[
|
const tile = /** @type {import("../../VectorRenderTile.js").default} */ (tiles[
|
||||||
i
|
i
|
||||||
]);
|
]);
|
||||||
const tileCoord = tile.tileCoord;
|
const transform = this.getTileRenderTransform(tile, frameState);
|
||||||
const tileExtent = tileGrid.getTileCoordExtent(tile.wrappedTileCoord);
|
|
||||||
const worldOffset =
|
|
||||||
tileGrid.getTileCoordExtent(tileCoord, this.tmpExtent)[0] -
|
|
||||||
tileExtent[0];
|
|
||||||
const transform = multiply(
|
|
||||||
scale(
|
|
||||||
this.inversePixelTransform.slice(),
|
|
||||||
1 / pixelRatio,
|
|
||||||
1 / pixelRatio
|
|
||||||
),
|
|
||||||
this.getRenderTransform(
|
|
||||||
center,
|
|
||||||
resolution,
|
|
||||||
rotation,
|
|
||||||
pixelRatio,
|
|
||||||
width,
|
|
||||||
height,
|
|
||||||
worldOffset
|
|
||||||
)
|
|
||||||
);
|
|
||||||
const executorGroups = tile.executorGroups[getUid(layer)];
|
const executorGroups = tile.executorGroups[getUid(layer)];
|
||||||
let clipped = false;
|
let clipped = false;
|
||||||
for (let t = 0, tt = executorGroups.length; t < tt; ++t) {
|
for (let t = 0, tt = executorGroups.length; t < tt; ++t) {
|
||||||
@@ -692,10 +755,17 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
* @param {import("../../Feature.js").FeatureLike} feature Feature.
|
* @param {import("../../Feature.js").FeatureLike} feature Feature.
|
||||||
* @param {number} squaredTolerance Squared tolerance.
|
* @param {number} squaredTolerance Squared tolerance.
|
||||||
* @param {import("../../style/Style.js").default|Array<import("../../style/Style.js").default>} styles The style or array of styles.
|
* @param {import("../../style/Style.js").default|Array<import("../../style/Style.js").default>} styles The style or array of styles.
|
||||||
* @param {import("../../render/canvas/BuilderGroup.js").default} executorGroup Replay group.
|
* @param {import("../../render/canvas/BuilderGroup.js").default} builderGroup Replay group.
|
||||||
|
* @param {import("../../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder group for decluttering.
|
||||||
* @return {boolean} `true` if an image is loading.
|
* @return {boolean} `true` if an image is loading.
|
||||||
*/
|
*/
|
||||||
renderFeature(feature, squaredTolerance, styles, executorGroup) {
|
renderFeature(
|
||||||
|
feature,
|
||||||
|
squaredTolerance,
|
||||||
|
styles,
|
||||||
|
builderGroup,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
) {
|
||||||
if (!styles) {
|
if (!styles) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -704,20 +774,24 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
for (let i = 0, ii = styles.length; i < ii; ++i) {
|
for (let i = 0, ii = styles.length; i < ii; ++i) {
|
||||||
loading =
|
loading =
|
||||||
renderFeature(
|
renderFeature(
|
||||||
executorGroup,
|
builderGroup,
|
||||||
feature,
|
feature,
|
||||||
styles[i],
|
styles[i],
|
||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
this.boundHandleStyleImageChange_
|
this.boundHandleStyleImageChange_,
|
||||||
|
undefined,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
) || loading;
|
) || loading;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
loading = renderFeature(
|
loading = renderFeature(
|
||||||
executorGroup,
|
builderGroup,
|
||||||
feature,
|
feature,
|
||||||
styles,
|
styles,
|
||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
this.boundHandleStyleImageChange_
|
this.boundHandleStyleImageChange_,
|
||||||
|
undefined,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return loading;
|
return loading;
|
||||||
|
|||||||
@@ -62,8 +62,15 @@ export function getTolerance(resolution, pixelRatio) {
|
|||||||
* @param {import("../geom/Circle.js").default} geometry Geometry.
|
* @param {import("../geom/Circle.js").default} geometry Geometry.
|
||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {import("../Feature.js").default} feature Feature.
|
* @param {import("../Feature.js").default} feature Feature.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderCircleGeometry(builderGroup, geometry, style, feature) {
|
function renderCircleGeometry(
|
||||||
|
builderGroup,
|
||||||
|
geometry,
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
) {
|
||||||
const fillStyle = style.getFill();
|
const fillStyle = style.getFill();
|
||||||
const strokeStyle = style.getStroke();
|
const strokeStyle = style.getStroke();
|
||||||
if (fillStyle || strokeStyle) {
|
if (fillStyle || strokeStyle) {
|
||||||
@@ -76,7 +83,7 @@ function renderCircleGeometry(builderGroup, geometry, style, feature) {
|
|||||||
}
|
}
|
||||||
const textStyle = style.getText();
|
const textStyle = style.getText();
|
||||||
if (textStyle) {
|
if (textStyle) {
|
||||||
const textReplay = builderGroup.getBuilder(
|
const textReplay = (opt_declutterBuilderGroup || builderGroup).getBuilder(
|
||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.TEXT
|
BuilderType.TEXT
|
||||||
);
|
);
|
||||||
@@ -92,8 +99,8 @@ function renderCircleGeometry(builderGroup, geometry, style, feature) {
|
|||||||
* @param {number} squaredTolerance Squared tolerance.
|
* @param {number} squaredTolerance Squared tolerance.
|
||||||
* @param {function(import("../events/Event.js").default): void} listener Listener function.
|
* @param {function(import("../events/Event.js").default): void} listener Listener function.
|
||||||
* @param {import("../proj.js").TransformFunction} [opt_transform] Transform from user to view projection.
|
* @param {import("../proj.js").TransformFunction} [opt_transform] Transform from user to view projection.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
* @return {boolean} `true` if style is loading.
|
* @return {boolean} `true` if style is loading.
|
||||||
* @template T
|
|
||||||
*/
|
*/
|
||||||
export function renderFeature(
|
export function renderFeature(
|
||||||
replayGroup,
|
replayGroup,
|
||||||
@@ -101,7 +108,8 @@ export function renderFeature(
|
|||||||
style,
|
style,
|
||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
listener,
|
listener,
|
||||||
opt_transform
|
opt_transform,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
) {
|
) {
|
||||||
let loading = false;
|
let loading = false;
|
||||||
const imageStyle = style.getImage();
|
const imageStyle = style.getImage();
|
||||||
@@ -123,7 +131,8 @@ export function renderFeature(
|
|||||||
feature,
|
feature,
|
||||||
style,
|
style,
|
||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
opt_transform
|
opt_transform,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
);
|
);
|
||||||
|
|
||||||
return loading;
|
return loading;
|
||||||
@@ -135,13 +144,15 @@ export function renderFeature(
|
|||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {number} squaredTolerance Squared tolerance.
|
* @param {number} squaredTolerance Squared tolerance.
|
||||||
* @param {import("../proj.js").TransformFunction} [opt_transform] Optional transform function.
|
* @param {import("../proj.js").TransformFunction} [opt_transform] Optional transform function.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderFeatureInternal(
|
function renderFeatureInternal(
|
||||||
replayGroup,
|
replayGroup,
|
||||||
feature,
|
feature,
|
||||||
style,
|
style,
|
||||||
squaredTolerance,
|
squaredTolerance,
|
||||||
opt_transform
|
opt_transform,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
) {
|
) {
|
||||||
const geometry = style.getGeometryFunction()(feature);
|
const geometry = style.getGeometryFunction()(feature);
|
||||||
if (!geometry) {
|
if (!geometry) {
|
||||||
@@ -156,7 +167,13 @@ function renderFeatureInternal(
|
|||||||
renderGeometry(replayGroup, simplifiedGeometry, style, feature);
|
renderGeometry(replayGroup, simplifiedGeometry, style, feature);
|
||||||
} else {
|
} else {
|
||||||
const geometryRenderer = GEOMETRY_RENDERERS[simplifiedGeometry.getType()];
|
const geometryRenderer = GEOMETRY_RENDERERS[simplifiedGeometry.getType()];
|
||||||
geometryRenderer(replayGroup, simplifiedGeometry, style, feature);
|
geometryRenderer(
|
||||||
|
replayGroup,
|
||||||
|
simplifiedGeometry,
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -187,18 +204,26 @@ function renderGeometry(replayGroup, geometry, style, feature) {
|
|||||||
* @param {import("../geom/GeometryCollection.js").default} geometry Geometry.
|
* @param {import("../geom/GeometryCollection.js").default} geometry Geometry.
|
||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {import("../Feature.js").default} feature Feature.
|
* @param {import("../Feature.js").default} feature Feature.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderGeometryCollectionGeometry(
|
function renderGeometryCollectionGeometry(
|
||||||
replayGroup,
|
replayGroup,
|
||||||
geometry,
|
geometry,
|
||||||
style,
|
style,
|
||||||
feature
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
) {
|
) {
|
||||||
const geometries = geometry.getGeometriesArray();
|
const geometries = geometry.getGeometriesArray();
|
||||||
let i, ii;
|
let i, ii;
|
||||||
for (i = 0, ii = geometries.length; i < ii; ++i) {
|
for (i = 0, ii = geometries.length; i < ii; ++i) {
|
||||||
const geometryRenderer = GEOMETRY_RENDERERS[geometries[i].getType()];
|
const geometryRenderer = GEOMETRY_RENDERERS[geometries[i].getType()];
|
||||||
geometryRenderer(replayGroup, geometries[i], style, feature);
|
geometryRenderer(
|
||||||
|
replayGroup,
|
||||||
|
geometries[i],
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -207,8 +232,15 @@ function renderGeometryCollectionGeometry(
|
|||||||
* @param {import("../geom/LineString.js").default|import("../render/Feature.js").default} geometry Geometry.
|
* @param {import("../geom/LineString.js").default|import("../render/Feature.js").default} geometry Geometry.
|
||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderLineStringGeometry(builderGroup, geometry, style, feature) {
|
function renderLineStringGeometry(
|
||||||
|
builderGroup,
|
||||||
|
geometry,
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
) {
|
||||||
const strokeStyle = style.getStroke();
|
const strokeStyle = style.getStroke();
|
||||||
if (strokeStyle) {
|
if (strokeStyle) {
|
||||||
const lineStringReplay = builderGroup.getBuilder(
|
const lineStringReplay = builderGroup.getBuilder(
|
||||||
@@ -220,7 +252,7 @@ function renderLineStringGeometry(builderGroup, geometry, style, feature) {
|
|||||||
}
|
}
|
||||||
const textStyle = style.getText();
|
const textStyle = style.getText();
|
||||||
if (textStyle) {
|
if (textStyle) {
|
||||||
const textReplay = builderGroup.getBuilder(
|
const textReplay = (opt_declutterBuilderGroup || builderGroup).getBuilder(
|
||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.TEXT
|
BuilderType.TEXT
|
||||||
);
|
);
|
||||||
@@ -234,8 +266,15 @@ function renderLineStringGeometry(builderGroup, geometry, style, feature) {
|
|||||||
* @param {import("../geom/MultiLineString.js").default|import("../render/Feature.js").default} geometry Geometry.
|
* @param {import("../geom/MultiLineString.js").default|import("../render/Feature.js").default} geometry Geometry.
|
||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderMultiLineStringGeometry(builderGroup, geometry, style, feature) {
|
function renderMultiLineStringGeometry(
|
||||||
|
builderGroup,
|
||||||
|
geometry,
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
) {
|
||||||
const strokeStyle = style.getStroke();
|
const strokeStyle = style.getStroke();
|
||||||
if (strokeStyle) {
|
if (strokeStyle) {
|
||||||
const lineStringReplay = builderGroup.getBuilder(
|
const lineStringReplay = builderGroup.getBuilder(
|
||||||
@@ -247,7 +286,7 @@ function renderMultiLineStringGeometry(builderGroup, geometry, style, feature) {
|
|||||||
}
|
}
|
||||||
const textStyle = style.getText();
|
const textStyle = style.getText();
|
||||||
if (textStyle) {
|
if (textStyle) {
|
||||||
const textReplay = builderGroup.getBuilder(
|
const textReplay = (opt_declutterBuilderGroup || builderGroup).getBuilder(
|
||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.TEXT
|
BuilderType.TEXT
|
||||||
);
|
);
|
||||||
@@ -261,8 +300,15 @@ function renderMultiLineStringGeometry(builderGroup, geometry, style, feature) {
|
|||||||
* @param {import("../geom/MultiPolygon.js").default} geometry Geometry.
|
* @param {import("../geom/MultiPolygon.js").default} geometry Geometry.
|
||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {import("../Feature.js").default} feature Feature.
|
* @param {import("../Feature.js").default} feature Feature.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderMultiPolygonGeometry(builderGroup, geometry, style, feature) {
|
function renderMultiPolygonGeometry(
|
||||||
|
builderGroup,
|
||||||
|
geometry,
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
) {
|
||||||
const fillStyle = style.getFill();
|
const fillStyle = style.getFill();
|
||||||
const strokeStyle = style.getStroke();
|
const strokeStyle = style.getStroke();
|
||||||
if (strokeStyle || fillStyle) {
|
if (strokeStyle || fillStyle) {
|
||||||
@@ -275,7 +321,7 @@ function renderMultiPolygonGeometry(builderGroup, geometry, style, feature) {
|
|||||||
}
|
}
|
||||||
const textStyle = style.getText();
|
const textStyle = style.getText();
|
||||||
if (textStyle) {
|
if (textStyle) {
|
||||||
const textReplay = builderGroup.getBuilder(
|
const textReplay = (opt_declutterBuilderGroup || builderGroup).getBuilder(
|
||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.TEXT
|
BuilderType.TEXT
|
||||||
);
|
);
|
||||||
@@ -289,9 +335,22 @@ function renderMultiPolygonGeometry(builderGroup, geometry, style, feature) {
|
|||||||
* @param {import("../geom/Point.js").default|import("../render/Feature.js").default} geometry Geometry.
|
* @param {import("../geom/Point.js").default|import("../render/Feature.js").default} geometry Geometry.
|
||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderPointGeometry(builderGroup, geometry, style, feature) {
|
function renderPointGeometry(
|
||||||
|
builderGroup,
|
||||||
|
geometry,
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
) {
|
||||||
const imageStyle = style.getImage();
|
const imageStyle = style.getImage();
|
||||||
|
const textStyle = style.getText();
|
||||||
|
let sharedData;
|
||||||
|
if (opt_declutterBuilderGroup) {
|
||||||
|
builderGroup = opt_declutterBuilderGroup;
|
||||||
|
sharedData = imageStyle && textStyle ? {} : undefined;
|
||||||
|
}
|
||||||
if (imageStyle) {
|
if (imageStyle) {
|
||||||
if (imageStyle.getImageState() != ImageState.LOADED) {
|
if (imageStyle.getImageState() != ImageState.LOADED) {
|
||||||
return;
|
return;
|
||||||
@@ -300,16 +359,15 @@ function renderPointGeometry(builderGroup, geometry, style, feature) {
|
|||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.IMAGE
|
BuilderType.IMAGE
|
||||||
);
|
);
|
||||||
imageReplay.setImageStyle(imageStyle);
|
imageReplay.setImageStyle(imageStyle, sharedData);
|
||||||
imageReplay.drawPoint(geometry, feature);
|
imageReplay.drawPoint(geometry, feature);
|
||||||
}
|
}
|
||||||
const textStyle = style.getText();
|
|
||||||
if (textStyle) {
|
if (textStyle) {
|
||||||
const textReplay = builderGroup.getBuilder(
|
const textReplay = builderGroup.getBuilder(
|
||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.TEXT
|
BuilderType.TEXT
|
||||||
);
|
);
|
||||||
textReplay.setTextStyle(textStyle);
|
textReplay.setTextStyle(textStyle, sharedData);
|
||||||
textReplay.drawText(geometry, feature);
|
textReplay.drawText(geometry, feature);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -319,9 +377,22 @@ function renderPointGeometry(builderGroup, geometry, style, feature) {
|
|||||||
* @param {import("../geom/MultiPoint.js").default|import("../render/Feature.js").default} geometry Geometry.
|
* @param {import("../geom/MultiPoint.js").default|import("../render/Feature.js").default} geometry Geometry.
|
||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderMultiPointGeometry(builderGroup, geometry, style, feature) {
|
function renderMultiPointGeometry(
|
||||||
|
builderGroup,
|
||||||
|
geometry,
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
) {
|
||||||
const imageStyle = style.getImage();
|
const imageStyle = style.getImage();
|
||||||
|
const textStyle = style.getText();
|
||||||
|
let sharedData;
|
||||||
|
if (opt_declutterBuilderGroup) {
|
||||||
|
builderGroup = opt_declutterBuilderGroup;
|
||||||
|
sharedData = imageStyle && textStyle ? {} : undefined;
|
||||||
|
}
|
||||||
if (imageStyle) {
|
if (imageStyle) {
|
||||||
if (imageStyle.getImageState() != ImageState.LOADED) {
|
if (imageStyle.getImageState() != ImageState.LOADED) {
|
||||||
return;
|
return;
|
||||||
@@ -330,16 +401,15 @@ function renderMultiPointGeometry(builderGroup, geometry, style, feature) {
|
|||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.IMAGE
|
BuilderType.IMAGE
|
||||||
);
|
);
|
||||||
imageReplay.setImageStyle(imageStyle);
|
imageReplay.setImageStyle(imageStyle, sharedData);
|
||||||
imageReplay.drawMultiPoint(geometry, feature);
|
imageReplay.drawMultiPoint(geometry, feature);
|
||||||
}
|
}
|
||||||
const textStyle = style.getText();
|
|
||||||
if (textStyle) {
|
if (textStyle) {
|
||||||
const textReplay = builderGroup.getBuilder(
|
const textReplay = (opt_declutterBuilderGroup || builderGroup).getBuilder(
|
||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.TEXT
|
BuilderType.TEXT
|
||||||
);
|
);
|
||||||
textReplay.setTextStyle(textStyle);
|
textReplay.setTextStyle(textStyle, sharedData);
|
||||||
textReplay.drawText(geometry, feature);
|
textReplay.drawText(geometry, feature);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -349,8 +419,15 @@ function renderMultiPointGeometry(builderGroup, geometry, style, feature) {
|
|||||||
* @param {import("../geom/Polygon.js").default|import("../render/Feature.js").default} geometry Geometry.
|
* @param {import("../geom/Polygon.js").default|import("../render/Feature.js").default} geometry Geometry.
|
||||||
* @param {import("../style/Style.js").default} style Style.
|
* @param {import("../style/Style.js").default} style Style.
|
||||||
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
* @param {import("../Feature.js").FeatureLike} feature Feature.
|
||||||
|
* @param {import("../render/canvas/BuilderGroup.js").default=} opt_declutterBuilderGroup Builder for decluttering.
|
||||||
*/
|
*/
|
||||||
function renderPolygonGeometry(builderGroup, geometry, style, feature) {
|
function renderPolygonGeometry(
|
||||||
|
builderGroup,
|
||||||
|
geometry,
|
||||||
|
style,
|
||||||
|
feature,
|
||||||
|
opt_declutterBuilderGroup
|
||||||
|
) {
|
||||||
const fillStyle = style.getFill();
|
const fillStyle = style.getFill();
|
||||||
const strokeStyle = style.getStroke();
|
const strokeStyle = style.getStroke();
|
||||||
if (fillStyle || strokeStyle) {
|
if (fillStyle || strokeStyle) {
|
||||||
@@ -363,7 +440,7 @@ function renderPolygonGeometry(builderGroup, geometry, style, feature) {
|
|||||||
}
|
}
|
||||||
const textStyle = style.getText();
|
const textStyle = style.getText();
|
||||||
if (textStyle) {
|
if (textStyle) {
|
||||||
const textReplay = builderGroup.getBuilder(
|
const textReplay = (opt_declutterBuilderGroup || builderGroup).getBuilder(
|
||||||
style.getZIndex(),
|
style.getZIndex(),
|
||||||
BuilderType.TEXT
|
BuilderType.TEXT
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -549,6 +549,7 @@ class RasterSource extends ImageSource {
|
|||||||
this.frameState_ = {
|
this.frameState_ = {
|
||||||
animate: false,
|
animate: false,
|
||||||
coordinateToPixelTransform: createTransform(),
|
coordinateToPixelTransform: createTransform(),
|
||||||
|
declutterTree: null,
|
||||||
extent: null,
|
extent: null,
|
||||||
index: 0,
|
index: 0,
|
||||||
layerIndex: 0,
|
layerIndex: 0,
|
||||||
|
|||||||
Reference in New Issue
Block a user