Merge pull request #13626 from MoonE/rendercomplete-wait-for-icons
Wait for icons to be loaded before firing rendercomplete event
This commit is contained in:
@@ -196,7 +196,11 @@ class LayerRenderer extends Observable {
|
|||||||
*/
|
*/
|
||||||
renderIfReadyAndVisible() {
|
renderIfReadyAndVisible() {
|
||||||
const layer = this.getLayer();
|
const layer = this.getLayer();
|
||||||
if (layer.getVisible() && layer.getSourceState() == SourceState.READY) {
|
if (
|
||||||
|
layer &&
|
||||||
|
layer.getVisible() &&
|
||||||
|
layer.getSourceState() == SourceState.READY
|
||||||
|
) {
|
||||||
layer.changed();
|
layer.changed();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -62,12 +62,6 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
*/
|
*/
|
||||||
this.animatingOrInteracting_;
|
this.animatingOrInteracting_;
|
||||||
|
|
||||||
/**
|
|
||||||
* @private
|
|
||||||
* @type {boolean}
|
|
||||||
*/
|
|
||||||
this.dirty_ = false;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @type {ImageData}
|
* @type {ImageData}
|
||||||
*/
|
*/
|
||||||
@@ -523,7 +517,7 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
const updateWhileInteracting = vectorLayer.getUpdateWhileInteracting();
|
const updateWhileInteracting = vectorLayer.getUpdateWhileInteracting();
|
||||||
|
|
||||||
if (
|
if (
|
||||||
(!this.dirty_ && !updateWhileAnimating && animating) ||
|
(this.ready && !updateWhileAnimating && animating) ||
|
||||||
(!updateWhileInteracting && interacting)
|
(!updateWhileInteracting && interacting)
|
||||||
) {
|
) {
|
||||||
this.animatingOrInteracting_ = true;
|
this.animatingOrInteracting_ = true;
|
||||||
@@ -594,7 +588,7 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
!this.dirty_ &&
|
this.ready &&
|
||||||
this.renderedResolution_ == resolution &&
|
this.renderedResolution_ == resolution &&
|
||||||
this.renderedRevision_ == vectorLayerRevision &&
|
this.renderedRevision_ == vectorLayerRevision &&
|
||||||
this.renderedRenderOrder_ == vectorLayerRenderOrder &&
|
this.renderedRenderOrder_ == vectorLayerRenderOrder &&
|
||||||
@@ -611,8 +605,6 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
|
|
||||||
this.replayGroup_ = null;
|
this.replayGroup_ = null;
|
||||||
|
|
||||||
this.dirty_ = false;
|
|
||||||
|
|
||||||
const replayGroup = new CanvasBuilderGroup(
|
const replayGroup = new CanvasBuilderGroup(
|
||||||
getRenderTolerance(resolution, pixelRatio),
|
getRenderTolerance(resolution, pixelRatio),
|
||||||
extent,
|
extent,
|
||||||
@@ -650,7 +642,7 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const squaredTolerance = getSquaredRenderTolerance(resolution, pixelRatio);
|
const squaredTolerance = getSquaredRenderTolerance(resolution, pixelRatio);
|
||||||
|
let ready = true;
|
||||||
const render =
|
const render =
|
||||||
/**
|
/**
|
||||||
* @param {import("../../Feature.js").default} feature Feature.
|
* @param {import("../../Feature.js").default} feature Feature.
|
||||||
@@ -672,7 +664,7 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
userTransform,
|
userTransform,
|
||||||
declutterBuilderGroup
|
declutterBuilderGroup
|
||||||
);
|
);
|
||||||
this.dirty_ = this.dirty_ || dirty;
|
ready = ready && !dirty;
|
||||||
}
|
}
|
||||||
}.bind(this);
|
}.bind(this);
|
||||||
|
|
||||||
@@ -686,6 +678,7 @@ class CanvasVectorLayerRenderer extends CanvasLayerRenderer {
|
|||||||
render(features[i]);
|
render(features[i]);
|
||||||
}
|
}
|
||||||
this.renderedFeatures_ = features;
|
this.renderedFeatures_ = features;
|
||||||
|
this.ready = ready;
|
||||||
|
|
||||||
const replayGroupInstructions = replayGroup.finish();
|
const replayGroupInstructions = replayGroup.finish();
|
||||||
const executorGroup = new ExecutorGroup(
|
const executorGroup = new ExecutorGroup(
|
||||||
|
|||||||
@@ -85,12 +85,6 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
/** @private */
|
/** @private */
|
||||||
this.boundHandleStyleImageChange_ = this.handleStyleImageChange_.bind(this);
|
this.boundHandleStyleImageChange_ = this.handleStyleImageChange_.bind(this);
|
||||||
|
|
||||||
/**
|
|
||||||
* @private
|
|
||||||
* @type {boolean}
|
|
||||||
*/
|
|
||||||
this.dirty_ = false;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @private
|
* @private
|
||||||
* @type {number}
|
* @type {number}
|
||||||
@@ -194,10 +188,10 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
*/
|
*/
|
||||||
prepareFrame(frameState) {
|
prepareFrame(frameState) {
|
||||||
const layerRevision = this.getLayer().getRevision();
|
const layerRevision = this.getLayer().getRevision();
|
||||||
if (this.renderedLayerRevision_ != layerRevision) {
|
if (this.renderedLayerRevision_ !== layerRevision) {
|
||||||
|
this.renderedLayerRevision_ = layerRevision;
|
||||||
this.renderedTiles.length = 0;
|
this.renderedTiles.length = 0;
|
||||||
}
|
}
|
||||||
this.renderedLayerRevision_ = layerRevision;
|
|
||||||
return super.prepareFrame(frameState);
|
return super.prepareFrame(frameState);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -238,6 +232,7 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
if (declutter) {
|
if (declutter) {
|
||||||
tile.declutterExecutorGroups[layerUid] = [];
|
tile.declutterExecutorGroups[layerUid] = [];
|
||||||
}
|
}
|
||||||
|
builderState.dirty = false;
|
||||||
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) {
|
||||||
@@ -255,7 +250,6 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
const bufferedExtent = equals(sourceTileExtent, sharedExtent)
|
const bufferedExtent = equals(sourceTileExtent, sharedExtent)
|
||||||
? null
|
? null
|
||||||
: builderExtent;
|
: builderExtent;
|
||||||
builderState.dirty = false;
|
|
||||||
const builderGroup = new CanvasBuilderGroup(
|
const builderGroup = new CanvasBuilderGroup(
|
||||||
0,
|
0,
|
||||||
builderExtent,
|
builderExtent,
|
||||||
@@ -289,7 +283,6 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
builderGroup,
|
builderGroup,
|
||||||
declutterBuilderGroup
|
declutterBuilderGroup
|
||||||
);
|
);
|
||||||
this.dirty_ = this.dirty_ || dirty;
|
|
||||||
builderState.dirty = builderState.dirty || dirty;
|
builderState.dirty = builderState.dirty || dirty;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -673,10 +666,12 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
const tiles = this.renderedTiles;
|
const tiles = this.renderedTiles;
|
||||||
const clips = [];
|
const clips = [];
|
||||||
const clipZs = [];
|
const clipZs = [];
|
||||||
|
let ready = true;
|
||||||
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} */ (
|
const tile = /** @type {import("../../VectorRenderTile.js").default} */ (
|
||||||
tiles[i]
|
tiles[i]
|
||||||
);
|
);
|
||||||
|
ready = ready && !tile.getReplayState(layer).dirty;
|
||||||
const executorGroups = tile.executorGroups[getUid(layer)].filter(
|
const executorGroups = tile.executorGroups[getUid(layer)].filter(
|
||||||
(group) => group.hasExecutors(replayTypes)
|
(group) => group.hasExecutors(replayTypes)
|
||||||
);
|
);
|
||||||
@@ -741,6 +736,7 @@ class CanvasVectorTileLayerRenderer extends CanvasTileLayerRenderer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
context.globalAlpha = alpha;
|
context.globalAlpha = alpha;
|
||||||
|
this.ready = ready;
|
||||||
|
|
||||||
return this.container;
|
return this.container;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -124,14 +124,13 @@ export function renderFeature(
|
|||||||
let loading = false;
|
let loading = false;
|
||||||
const imageStyle = style.getImage();
|
const imageStyle = style.getImage();
|
||||||
if (imageStyle) {
|
if (imageStyle) {
|
||||||
let imageState = imageStyle.getImageState();
|
const imageState = imageStyle.getImageState();
|
||||||
if (imageState == ImageState.LOADED || imageState == ImageState.ERROR) {
|
if (imageState == ImageState.LOADED || imageState == ImageState.ERROR) {
|
||||||
imageStyle.unlistenImageChange(listener);
|
imageStyle.unlistenImageChange(listener);
|
||||||
} else {
|
} else {
|
||||||
if (imageState == ImageState.IDLE) {
|
if (imageState == ImageState.IDLE) {
|
||||||
imageStyle.load();
|
imageStyle.load();
|
||||||
}
|
}
|
||||||
imageState = imageStyle.getImageState();
|
|
||||||
imageStyle.listenImageChange(listener);
|
imageStyle.listenImageChange(listener);
|
||||||
loading = true;
|
loading = true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,9 +22,12 @@ import TileLayer from '../../../../src/ol/layer/Tile.js';
|
|||||||
import TileLayerRenderer from '../../../../src/ol/renderer/canvas/TileLayer.js';
|
import TileLayerRenderer from '../../../../src/ol/renderer/canvas/TileLayer.js';
|
||||||
import VectorLayer from '../../../../src/ol/layer/Vector.js';
|
import VectorLayer from '../../../../src/ol/layer/Vector.js';
|
||||||
import VectorSource from '../../../../src/ol/source/Vector.js';
|
import VectorSource from '../../../../src/ol/source/Vector.js';
|
||||||
|
import VectorTileLayer from '../../../../src/ol/layer/VectorTile.js';
|
||||||
|
import VectorTileSource from '../../../../src/ol/source/VectorTile.js';
|
||||||
import View from '../../../../src/ol/View.js';
|
import View from '../../../../src/ol/View.js';
|
||||||
import WebGLPointsLayer from '../../../../src/ol/layer/WebGLPoints.js';
|
import WebGLPointsLayer from '../../../../src/ol/layer/WebGLPoints.js';
|
||||||
import XYZ from '../../../../src/ol/source/XYZ.js';
|
import XYZ from '../../../../src/ol/source/XYZ.js';
|
||||||
|
import {Icon, Style} from '../../../../src/ol/style.js';
|
||||||
import {LineString, Point, Polygon} from '../../../../src/ol/geom.js';
|
import {LineString, Point, Polygon} from '../../../../src/ol/geom.js';
|
||||||
import {TRUE} from '../../../../src/ol/functions.js';
|
import {TRUE} from '../../../../src/ol/functions.js';
|
||||||
import {
|
import {
|
||||||
@@ -36,6 +39,7 @@ import {
|
|||||||
} from '../../../../src/ol/proj.js';
|
} from '../../../../src/ol/proj.js';
|
||||||
import {createXYZ} from '../../../../src/ol/tilegrid.js';
|
import {createXYZ} from '../../../../src/ol/tilegrid.js';
|
||||||
import {defaults as defaultInteractions} from '../../../../src/ol/interaction.js';
|
import {defaults as defaultInteractions} from '../../../../src/ol/interaction.js';
|
||||||
|
import {shared as iconImageCache} from '../../../../src/ol/style/IconImageCache.js';
|
||||||
import {tile as tileStrategy} from '../../../../src/ol/loadingstrategy.js';
|
import {tile as tileStrategy} from '../../../../src/ol/loadingstrategy.js';
|
||||||
|
|
||||||
describe('ol/Map', function () {
|
describe('ol/Map', function () {
|
||||||
@@ -403,105 +407,239 @@ describe('ol/Map', function () {
|
|||||||
});
|
});
|
||||||
|
|
||||||
describe('rendercomplete event', function () {
|
describe('rendercomplete event', function () {
|
||||||
let map;
|
let map, target;
|
||||||
|
|
||||||
beforeEach(function () {
|
beforeEach(function () {
|
||||||
const target = document.createElement('div');
|
target = document.createElement('div');
|
||||||
target.style.width = '100px';
|
target.style.width = '100px';
|
||||||
target.style.height = '100px';
|
target.style.height = '100px';
|
||||||
document.body.appendChild(target);
|
document.body.appendChild(target);
|
||||||
map = new Map({
|
|
||||||
target: target,
|
|
||||||
layers: [
|
|
||||||
new TileLayer({
|
|
||||||
opacity: 0.5,
|
|
||||||
source: new XYZ({
|
|
||||||
url: 'spec/ol/data/osm-{z}-{x}-{y}.png',
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
new ImageLayer({
|
|
||||||
source: new ImageStatic({
|
|
||||||
url: 'spec/ol/data/osm-0-0-0.png',
|
|
||||||
imageExtent: getProjection('EPSG:3857').getExtent(),
|
|
||||||
projection: 'EPSG:3857',
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
new VectorLayer({
|
|
||||||
source: new VectorSource({
|
|
||||||
url: 'spec/ol/data/point.json',
|
|
||||||
format: new GeoJSON(),
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
new VectorLayer({
|
|
||||||
source: new VectorSource({
|
|
||||||
url: 'spec/ol/data/point.json',
|
|
||||||
format: new GeoJSON(),
|
|
||||||
strategy: tileStrategy(createXYZ()),
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
new VectorLayer({
|
|
||||||
source: new VectorSource({
|
|
||||||
features: [new Feature(new Point([0, 0]))],
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
new VectorLayer({
|
|
||||||
source: new VectorSource({
|
|
||||||
loader: function (extent, resolution, projection) {
|
|
||||||
this.addFeature(new Feature(new Point([0, 0])));
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
new WebGLPointsLayer({
|
|
||||||
source: new VectorSource({
|
|
||||||
features: [new Feature(new Point([0, 0]))],
|
|
||||||
}),
|
|
||||||
style: {
|
|
||||||
symbol: {
|
|
||||||
color: 'red',
|
|
||||||
symbolType: 'circle',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
],
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(function () {
|
afterEach(function () {
|
||||||
document.body.removeChild(map.getTargetElement());
|
disposeMap(map);
|
||||||
map.setTarget(null);
|
|
||||||
map.dispose();
|
|
||||||
map.getLayers().forEach((layer) => layer.dispose());
|
map.getLayers().forEach((layer) => layer.dispose());
|
||||||
});
|
});
|
||||||
|
|
||||||
it('triggers when all tiles and sources are loaded and faded in', function (done) {
|
describe('renderer ready property', function () {
|
||||||
const layers = map.getLayers().getArray();
|
beforeEach(function () {
|
||||||
expect(layers[6].getRenderer().ready).to.be(false);
|
map = new Map({
|
||||||
map.once('rendercomplete', function () {
|
target: target,
|
||||||
expect(map.tileQueue_.getTilesLoading()).to.be(0);
|
layers: [
|
||||||
expect(layers[1].getSource().image_.getState()).to.be(
|
new TileLayer({
|
||||||
ImageState.LOADED
|
opacity: 0.5,
|
||||||
|
source: new XYZ({
|
||||||
|
url: 'spec/ol/data/osm-{z}-{x}-{y}.png',
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
new ImageLayer({
|
||||||
|
source: new ImageStatic({
|
||||||
|
url: 'spec/ol/data/osm-0-0-0.png',
|
||||||
|
imageExtent: getProjection('EPSG:3857').getExtent(),
|
||||||
|
projection: 'EPSG:3857',
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
new VectorLayer({
|
||||||
|
source: new VectorSource({
|
||||||
|
url: 'spec/ol/data/point.json',
|
||||||
|
format: new GeoJSON(),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
new VectorLayer({
|
||||||
|
source: new VectorSource({
|
||||||
|
url: 'spec/ol/data/point.json',
|
||||||
|
format: new GeoJSON(),
|
||||||
|
strategy: tileStrategy(createXYZ()),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
new VectorLayer({
|
||||||
|
source: new VectorSource({
|
||||||
|
features: [new Feature(new Point([0, 0]))],
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
new VectorLayer({
|
||||||
|
source: new VectorSource({
|
||||||
|
loader: function (extent, resolution, projection) {
|
||||||
|
this.addFeature(new Feature(new Point([0, 0])));
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
new WebGLPointsLayer({
|
||||||
|
source: new VectorSource({
|
||||||
|
features: [new Feature(new Point([0, 0]))],
|
||||||
|
}),
|
||||||
|
style: {
|
||||||
|
symbol: {
|
||||||
|
color: 'red',
|
||||||
|
symbolType: 'circle',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('triggers when all tiles and sources are loaded and faded in', function (done) {
|
||||||
|
const layers = map.getLayers().getArray();
|
||||||
|
expect(layers[6].getRenderer().ready).to.be(false);
|
||||||
|
map.once('rendercomplete', function () {
|
||||||
|
expect(map.tileQueue_.getTilesLoading()).to.be(0);
|
||||||
|
expect(layers[1].getSource().image_.getState()).to.be(
|
||||||
|
ImageState.LOADED
|
||||||
|
);
|
||||||
|
expect(layers[2].getSource().getFeatures().length).to.be(1);
|
||||||
|
expect(layers[6].getRenderer().ready).to.be(true);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
map.setView(
|
||||||
|
new View({
|
||||||
|
center: [0, 0],
|
||||||
|
zoom: 0,
|
||||||
|
})
|
||||||
);
|
);
|
||||||
expect(layers[2].getSource().getFeatures().length).to.be(1);
|
|
||||||
expect(layers[6].getRenderer().ready).to.be(true);
|
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
map.setView(
|
|
||||||
new View({
|
it('ignores invisible layers', function (done) {
|
||||||
center: [0, 0],
|
map.getLayers().forEach((layer, i) => layer.setVisible(i === 4));
|
||||||
zoom: 0,
|
map.setView(
|
||||||
})
|
new View({
|
||||||
);
|
center: [0, 0],
|
||||||
|
zoom: 0,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
map.once('rendercomplete', () => done());
|
||||||
|
});
|
||||||
});
|
});
|
||||||
it('ignores invisible layers', function (done) {
|
|
||||||
map.getLayers().forEach(function (layer, i) {
|
describe('with icons', function () {
|
||||||
layer.setVisible(i === 4);
|
/** @type {Icon} */
|
||||||
|
let icon;
|
||||||
|
beforeEach(function () {
|
||||||
|
iconImageCache.clear();
|
||||||
|
icon = new Icon({
|
||||||
|
src: 'spec/ol/data/dot.png?delayed',
|
||||||
|
});
|
||||||
|
|
||||||
|
const delay = 100;
|
||||||
|
// Delay icon change events
|
||||||
|
let states = [{state: icon.getImageState()}];
|
||||||
|
icon.listenImageChange = function (listener) {
|
||||||
|
if (!listener._delay) {
|
||||||
|
listener._delay = (e) => {
|
||||||
|
const key = setTimeout(() => {
|
||||||
|
states.shift();
|
||||||
|
listener.call(this, e);
|
||||||
|
}, delay);
|
||||||
|
Object.assign(states[states.length - 1], {key, listener});
|
||||||
|
states.push({
|
||||||
|
state: Icon.prototype.getImageState.call(this),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return Icon.prototype.listenImageChange.call(this, listener._delay);
|
||||||
|
};
|
||||||
|
icon.unlistenImageChange = function (listener) {
|
||||||
|
states = states.filter((state) => {
|
||||||
|
if (state.listener !== listener) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
clearTimeout(listener.key);
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
const addedListener = listener._delay;
|
||||||
|
delete listener._delay;
|
||||||
|
return Icon.prototype.unlistenImageChange.call(this, addedListener);
|
||||||
|
};
|
||||||
|
icon.getImageState = function () {
|
||||||
|
return states[0].state;
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
it('waits for icons to be loaded with ol/renderer/canvas/VectorTileLayer', function (done) {
|
||||||
|
const delayIconAtTile = 1;
|
||||||
|
let tilesRequested = 0;
|
||||||
|
const tileSize = 64;
|
||||||
|
const tileGrid = createXYZ({tileSize: tileSize});
|
||||||
|
map = new Map({
|
||||||
|
target: target,
|
||||||
|
view: new View({
|
||||||
|
center: [0, 0],
|
||||||
|
resolution: 1,
|
||||||
|
}),
|
||||||
|
layers: [
|
||||||
|
new VectorTileLayer({
|
||||||
|
source: new VectorTileSource({
|
||||||
|
tileSize: tileSize,
|
||||||
|
tileUrlFunction: (tileCoord) => tileCoord.join('/'),
|
||||||
|
tileLoadFunction: function (tile, url) {
|
||||||
|
const coordinate = tileGrid.getTileCoordCenter(
|
||||||
|
tile.getTileCoord()
|
||||||
|
);
|
||||||
|
const feature = new Feature(new Point(coordinate));
|
||||||
|
tile.setFeatures([feature]);
|
||||||
|
if (tilesRequested++ === delayIconAtTile) {
|
||||||
|
feature.setStyle(new Style({image: icon}));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
style: new Style({
|
||||||
|
image: new Icon({
|
||||||
|
src: 'spec/ol/data/dot.png',
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
});
|
||||||
|
let iconLoaded = false;
|
||||||
|
icon.listenImageChange(function (e) {
|
||||||
|
if (e.target.getImageState() === ImageState.LOADED) {
|
||||||
|
iconLoaded = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
map.once('rendercomplete', function () {
|
||||||
|
try {
|
||||||
|
expect(tilesRequested).to.be.greaterThan(delayIconAtTile);
|
||||||
|
expect(iconLoaded).to.be(true);
|
||||||
|
done();
|
||||||
|
} catch (e) {
|
||||||
|
done(e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('waits for icons to be loaded with ol/renderer/canvas/VectorLayer', function (done) {
|
||||||
|
map = new Map({
|
||||||
|
target: target,
|
||||||
|
view: new View({
|
||||||
|
center: [0, 0],
|
||||||
|
resolution: 1,
|
||||||
|
}),
|
||||||
|
layers: [
|
||||||
|
new VectorLayer({
|
||||||
|
source: new VectorSource({
|
||||||
|
features: [new Feature(new Point([0, 0]))],
|
||||||
|
}),
|
||||||
|
style: new Style({
|
||||||
|
image: icon,
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
});
|
||||||
|
let iconLoaded = false;
|
||||||
|
icon.listenImageChange(function (e) {
|
||||||
|
if (e.target.getImageState() === ImageState.LOADED) {
|
||||||
|
iconLoaded = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
map.once('rendercomplete', function () {
|
||||||
|
try {
|
||||||
|
expect(iconLoaded).to.be(true);
|
||||||
|
done();
|
||||||
|
} catch (e) {
|
||||||
|
done(e);
|
||||||
|
}
|
||||||
|
});
|
||||||
});
|
});
|
||||||
map.setView(
|
|
||||||
new View({
|
|
||||||
center: [0, 0],
|
|
||||||
zoom: 0,
|
|
||||||
})
|
|
||||||
);
|
|
||||||
map.once('rendercomplete', () => done());
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,17 @@
|
|||||||
|
import Feature from '../../../../../src/ol/Feature.js';
|
||||||
import GeoJSON from '../../../../../src/ol/format/GeoJSON.js';
|
import GeoJSON from '../../../../../src/ol/format/GeoJSON.js';
|
||||||
|
import ImageState from '../../../../../src/ol/ImageState.js';
|
||||||
import Map from '../../../../../src/ol/Map.js';
|
import Map from '../../../../../src/ol/Map.js';
|
||||||
import VectorTileLayer from '../../../../../src/ol/layer/VectorTile.js';
|
import VectorTileLayer from '../../../../../src/ol/layer/VectorTile.js';
|
||||||
import VectorTileSource from '../../../../../src/ol/source/VectorTile.js';
|
import VectorTileSource from '../../../../../src/ol/source/VectorTile.js';
|
||||||
import View from '../../../../../src/ol/View.js';
|
import View from '../../../../../src/ol/View.js';
|
||||||
import {fromLonLat} from '../../../../../src/ol/proj.js';
|
import {Icon, Style} from '../../../../../src/ol/style.js';
|
||||||
|
import {Point} from '../../../../../src/ol/geom.js';
|
||||||
|
import {create as createTransform} from '../../../../../src/ol/transform.js';
|
||||||
|
import {createXYZ} from '../../../../../src/ol/tilegrid.js';
|
||||||
|
import {fromLonLat, get as getProjection} from '../../../../../src/ol/proj.js';
|
||||||
|
import {getUid} from '../../../../../src/ol/util.js';
|
||||||
|
import {isEmpty} from '../../../../../src/ol/obj.js';
|
||||||
|
|
||||||
describe('ol.layer.VectorTile', function () {
|
describe('ol.layer.VectorTile', function () {
|
||||||
describe('constructor (defaults)', function () {
|
describe('constructor (defaults)', function () {
|
||||||
@@ -150,4 +158,94 @@ describe('ol.layer.VectorTile', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('#renderFrame', function () {
|
||||||
|
/** @type {VectorTileLayer} */ let layer;
|
||||||
|
|
||||||
|
afterEach(function () {
|
||||||
|
layer.dispose();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('sets ready property to false when icons are loading', function (done) {
|
||||||
|
const zoom = 1;
|
||||||
|
const tileSize = 32;
|
||||||
|
const projection = getProjection('EPSG:3857');
|
||||||
|
const tileGrid = createXYZ({tileSize: tileSize});
|
||||||
|
const resolution = tileGrid.getResolution(zoom);
|
||||||
|
layer = new VectorTileLayer({
|
||||||
|
renderBuffer: 0,
|
||||||
|
source: new VectorTileSource({
|
||||||
|
tileSize: tileSize,
|
||||||
|
tileUrlFunction: (tileCoord) => tileCoord.join('/'),
|
||||||
|
tileLoadFunction: function (tile, url) {
|
||||||
|
const coordinate = tileGrid.getTileCoordCenter(tile.getTileCoord());
|
||||||
|
tile.setFeatures([new Feature(new Point(coordinate))]);
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
style: new Style({
|
||||||
|
image: new Icon({
|
||||||
|
src:
|
||||||
|
'data:image/svg+xml;base64,' +
|
||||||
|
window.btoa(`<svg width="10" height="10" viewBox="0 0 1 1" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<rect x="0" y="0" width="1" height="1"/>
|
||||||
|
</svg>`),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
const renderer = layer.getRenderer();
|
||||||
|
const frameState =
|
||||||
|
/** @type {import("../../../../../src/ol/PluggableMap.js").FrameState} */ ({
|
||||||
|
pixelRatio: 1,
|
||||||
|
viewState: {
|
||||||
|
zoom: zoom,
|
||||||
|
resolution: resolution,
|
||||||
|
center: [0, 0],
|
||||||
|
rotation: 0,
|
||||||
|
projection: projection,
|
||||||
|
},
|
||||||
|
size: [2 * tileSize, 2 * tileSize],
|
||||||
|
extent: [-tileSize, -tileSize, tileSize, tileSize].map(
|
||||||
|
(n) => n * resolution
|
||||||
|
),
|
||||||
|
viewHints: [0, 0],
|
||||||
|
layerStatesArray: layer.getLayerStatesArray(),
|
||||||
|
layerIndex: 0,
|
||||||
|
wantedTiles: {},
|
||||||
|
usedTiles: {},
|
||||||
|
tileQueue: {isKeyQueued: () => true},
|
||||||
|
pixelToCoordinateTransform: createTransform(),
|
||||||
|
});
|
||||||
|
|
||||||
|
renderer.renderFrame(frameState);
|
||||||
|
// Tiles not yet loaded, no icon queued
|
||||||
|
expect(renderer.ready).to.be(true);
|
||||||
|
const source = layer.getSource();
|
||||||
|
const wantedTiles = frameState.wantedTiles[getUid(source)];
|
||||||
|
expect(isEmpty(wantedTiles)).to.be(false);
|
||||||
|
|
||||||
|
// Tiles are loaded synchronously
|
||||||
|
source.tileCache.forEach((tile) => tile.load());
|
||||||
|
|
||||||
|
renderer.renderFrame(frameState);
|
||||||
|
// Tiles loaded, waiting for icon
|
||||||
|
expect(renderer.ready).to.be(false);
|
||||||
|
|
||||||
|
layer
|
||||||
|
.getStyle()
|
||||||
|
.getImage()
|
||||||
|
.listenImageChange(function (evt) {
|
||||||
|
if (evt.target.getImageState() !== ImageState.LOADED) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
renderer.renderFrame(frameState);
|
||||||
|
// Tiles and icon loaded
|
||||||
|
expect(renderer.ready).to.be(true);
|
||||||
|
done();
|
||||||
|
} catch (e) {
|
||||||
|
done(e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user