Files
openlayers/test/browser/spec/ol/source/vectortile.test.js
2021-04-28 09:23:33 -07:00

472 lines
14 KiB
JavaScript

import Feature from '../../../../../src/ol/Feature.js';
import GeoJSON from '../../../../../src/ol/format/GeoJSON.js';
import MVT from '../../../../../src/ol/format/MVT.js';
import Map from '../../../../../src/ol/Map.js';
import TileGrid from '../../../../../src/ol/tilegrid/TileGrid.js';
import TileState from '../../../../../src/ol/TileState.js';
import VectorRenderTile from '../../../../../src/ol/VectorRenderTile.js';
import VectorTile from '../../../../../src/ol/VectorTile.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 {createXYZ} from '../../../../../src/ol/tilegrid.js';
import {fromExtent} from '../../../../../src/ol/geom/Polygon.js';
import {get, get as getProjection} from '../../../../../src/ol/proj.js';
import {getCenter} from '../../../../../src/ol/extent.js';
import {listen, unlistenByKey} from '../../../../../src/ol/events.js';
import {unByKey} from '../../../../../src/ol/Observable.js';
describe('ol.source.VectorTile', function () {
let format, source;
beforeEach(function () {
format = new MVT();
source = new VectorTileSource({
format: format,
url: 'spec/ol/data/{z}-{x}-{y}.vector.pbf',
});
});
describe('constructor', function () {
it('sets the format on the instance', function () {
expect(source.format_).to.equal(format);
});
it('sets the default zDirection on the instance', function () {
expect(source.zDirection).to.be(1);
});
it('uses ol.VectorTile as default tileClass', function () {
expect(source.tileClass).to.equal(VectorTile);
});
it('creates a 512 XYZ tilegrid by default', function () {
const tileGrid = createXYZ({tileSize: 512});
expect(source.tileGrid.tileSize_).to.equal(tileGrid.tileSize_);
expect(source.tileGrid.extent_).to.equal(tileGrid.extent_);
});
});
describe('#getTile()', function () {
it('creates a tile with the correct tile class', function () {
const tile = source.getTile(0, 0, 0, 1, getProjection('EPSG:3857'));
expect(tile).to.be.a(VectorRenderTile);
expect(tile.getTileCoord()).to.eql([0, 0, 0]);
expect(source.getTile(0, 0, 0, 1, getProjection('EPSG:3857'))).to.equal(
tile
);
});
it('loads source tiles', function (done) {
const source = new VectorTileSource({
format: new GeoJSON(),
url: 'spec/ol/data/point.json',
});
const tile = source.getTile(0, 0, 0, 1, source.getProjection());
tile.load();
const key = listen(tile, 'change', function (e) {
if (tile.getState() === TileState.LOADED) {
const sourceTile = source.getSourceTiles(
1,
source.getProjection(),
tile
)[0];
expect(sourceTile.getFeatures().length).to.be.greaterThan(0);
unlistenByKey(key);
done();
}
});
});
it('handles empty tiles', function () {
const source = new VectorTileSource({
format: new GeoJSON(),
url: '',
});
const tile = source.getTile(0, 0, 0, 1, source.getProjection());
expect(tile.getState()).to.be(TileState.EMPTY);
});
it('creates empty tiles outside the source extent', function () {
const fullExtent = get('EPSG:3857').getExtent();
const source = new VectorTileSource({
extent: [fullExtent[0], fullExtent[1], 0, 0],
});
const tile = source.getTile(1, 1, 1, 1, source.getProjection());
expect(tile.getState()).to.be(TileState.EMPTY);
});
it('creates empty tiles outside the world extent when wrapX === false', function () {
const source = new VectorTileSource({
wrapX: false,
});
const tile = source.getTile(0, -1, 0, 1, source.getProjection());
expect(tile.getState()).to.be(TileState.EMPTY);
});
it('creates empty tiles when the tileUrlFunction returns undefined', function () {
const source = new VectorTileSource({
tileUrlFunction: function (tileCoord) {
return;
},
});
const tile = source.getTile(1, 1, 1, 1, source.getProjection());
expect(tile.getState()).to.be(TileState.EMPTY);
});
it('creates non-empty tiles outside the world extent when wrapX === true', function () {
const source = new VectorTileSource({
url: '{z}/{x}/{y}.pbf',
});
const tile = source.getTile(0, -1, 0, 1, source.getProjection());
expect(tile.getState()).to.be(TileState.IDLE);
});
it('creates non-empty tiles for overzoomed resolutions', function () {
const source = new VectorTileSource({
url: '{z}/{x}/{y}.pbf',
tileLoadFunction: function (tile) {
tile.setLoader(function () {});
},
maxZoom: 16,
});
const tile = source.getTile(
24,
9119385,
5820434,
1,
source.getProjection()
);
tile.load();
expect(tile.getState()).to.be(TileState.LOADING);
});
it('creates new tile when source key changes', function () {
source.setKey('key1');
const tile1 = source.getTile(0, 0, 0, 1, getProjection('EPSG:3857'));
const tile2 = source.getTile(0, 0, 0, 1, getProjection('EPSG:3857'));
source.setKey('key2');
const tile3 = source.getTile(0, 0, 0, 1, getProjection('EPSG:3857'));
expect(tile1).to.equal(tile2);
expect(tile1.key).to.be('key1');
expect(tile3.key).to.be('key2');
});
});
describe('#getTileGridForProjection', function () {
it("creates a tile grid with the source tile grid's tile size", function () {
const tileGrid = source.getTileGridForProjection(
getProjection('EPSG:3857')
);
expect(tileGrid.getTileSize(0)).to.be(512);
});
});
describe('Tile load events', function () {
it('triggers tileloadstart and tileloadend with ol.VectorTile', function (done) {
const tile = source.getTile(
14,
8938,
5680,
1,
getProjection('EPSG:3857')
);
let started = false;
source.on('tileloadstart', function () {
started = true;
});
source.on('tileloadend', function (e) {
expect(started).to.be(true);
expect(e.tile).to.be.a(VectorTile);
expect(e.tile.getFeatures().length).to.be(1327);
done();
});
tile.load();
});
});
describe('different source and render tile grids', function () {
let source, map, loaded, requested, target;
beforeEach(function () {
loaded = [];
requested = 0;
function tileUrlFunction(tileCoord) {
++requested;
if (tileCoord.toString() == '5,13,-29') {
return tileCoord.join('/');
}
}
function tileLoadFunction(tile, src) {
tile.setLoader(function () {});
loaded.push(src);
}
const extent = [
665584.2026596286,
7033250.839875697,
667162.0221431496,
7035280.378636755,
];
source = new VectorTileSource({
tileGrid: new TileGrid({
origin: [218128, 6126002],
resolutions: [
4096,
2048,
1024,
512,
256,
128,
64,
32,
16,
8,
4,
2,
1,
0.5,
],
}),
tileUrlFunction: tileUrlFunction,
tileLoadFunction: tileLoadFunction,
});
target = document.createElement('div');
target.style.width = '100px';
target.style.height = '100px';
document.body.appendChild(target);
map = new Map({
layers: [
new VectorTileLayer({
extent: extent,
source: source,
}),
],
target: target,
view: new View({
zoom: 11,
center: [666373.1624999996, 7034265.3572],
}),
});
});
afterEach(function () {
document.body.removeChild(target);
});
it('loads available tiles', function (done) {
map.renderSync();
setTimeout(function () {
expect(requested).to.be.greaterThan(1);
expect(loaded).to.eql(['5/13/-29']);
done();
}, 0);
});
});
it('does not fill up the tile queue', function (done) {
const target = document.createElement('div');
target.style.width = '100px';
target.style.height = '100px';
document.body.appendChild(target);
const urls = [
'spec/ol/data/14-8938-5680.vector.pbf?num=0&coord={z},{x},{y}',
'spec/ol/data/14-8938-5680.vector.pbf?num=1&coord={z},{x},{y}',
'spec/ol/data/14-8938-5680.vector.pbf?num=2&coord={z},{x},{y}',
'spec/ol/data/14-8938-5680.vector.pbf?num=3&coord={z},{x},{y}',
];
const source = new VectorTileSource({
format: new MVT(),
url: urls[0],
});
const map = new Map({
target: target,
layers: [
new VectorTileLayer({
source: source,
}),
],
view: new View({
center: [0, 0],
zoom: 0,
}),
});
map.renderSync();
const max = urls.length + 3;
let count = 0;
let tile = source.getTile(0, 0, 0, 1, map.getView().getProjection());
tile.addEventListener('change', function onTileChange(e) {
if (e.target.getState() !== TileState.LOADED && !e.target.hifi) {
return;
}
e.target.removeEventListener('change', onTileChange);
map.once('rendercomplete', function () {
expect(map.tileQueue_.getTilesLoading()).to.be(0);
++count;
if (count === max) {
document.body.removeChild(target);
map.dispose();
done();
return;
}
source.setUrl(urls[count % urls.length]);
tile = source.getTile(0, 0, 0, 1, map.getView().getProjection());
tile.addEventListener('change', onTileChange);
});
});
});
describe('interim tile handling', function () {
let map, source, target;
beforeEach(function () {
target = document.createElement('div');
target.style.width = '100px';
target.style.height = '100px';
document.body.appendChild(target);
const extent = [
1824704.739223726,
6141868.096770482,
1827150.7241288517,
6144314.081675608,
];
source = new VectorTileSource({
format: new MVT(),
url: 'spec/ol/data/14-8938-5680.vector.pbf',
minZoom: 14,
maxZoom: 14,
});
map = new Map({
pixelRatio: 1,
target: target,
layers: [
new VectorTileLayer({
extent: extent,
source: source,
}),
],
view: new View({
center: getCenter(extent),
zoom: 15,
}),
});
});
afterEach(function () {
map.setTarget(null);
document.body.removeChild(target);
});
it('re-renders when source changes', function (done) {
map.once('rendercomplete', function () {
const key = map.on('rendercomplete', function () {
const tile = source.getTile(
14,
8938,
5680,
1,
map.getView().getProjection()
);
expect(tile.getKey()).to.be(
'spec/ol/data/14-8938-5680.vector.pbf?new/14,8938,5680'
);
expect(tile.interimTile).to.be.a(VectorRenderTile);
expect(tile.interimTile.getKey()).to.be(
'spec/ol/data/14-8938-5680.vector.pbf/14,8938,5680'
);
const sourceTiles = source.getSourceTiles(
1,
map.getView().getProjection(),
tile
);
if (sourceTiles) {
expect(sourceTiles[0].getKey()).to.be(
'spec/ol/data/14-8938-5680.vector.pbf?new/14,8938,5680'
);
unByKey(key);
done();
}
});
source.setUrl('spec/ol/data/14-8938-5680.vector.pbf?new');
});
});
});
describe('getFeatuersInExtent', function () {
let map, source, target;
beforeEach(function () {
source = new VectorTileSource({
maxZoom: 15,
tileSize: 256,
url: '{z}/{x}/{y}',
tileLoadFunction: function (tile) {
const extent = source
.getTileGrid()
.getTileCoordExtent(tile.tileCoord);
const feature = new Feature(fromExtent(extent));
feature.set('z', tile.tileCoord[0]);
tile.setFeatures([feature]);
},
});
target = document.createElement('div');
target.style.width = '100px';
target.style.height = '100px';
document.body.appendChild(target);
map = new Map({
target: target,
layers: [
new VectorTileLayer({
source: source,
}),
],
view: new View({
center: [0, 0],
zoom: 0,
}),
});
});
afterEach(function () {
map.setTarget(null);
document.body.removeChild(target);
});
it('returns an empty array when no tiles are in the cache', function () {
source.tileCache.clear();
const extent = map.getView().calculateExtent(map.getSize());
expect(source.getFeaturesInExtent(extent).length).to.be(0);
});
it('returns features in extent for the last rendered z', function (done) {
map.getView().setZoom(15);
map.once('rendercomplete', function () {
const extent = map.getView().calculateExtent(map.getSize());
const features = source.getFeaturesInExtent(extent);
expect(features.length).to.be(4);
expect(features[0].get('z')).to.be(15);
map.getView().setZoom(0);
map.once('rendercomplete', function () {
const extent = map.getView().calculateExtent(map.getSize());
const features = source.getFeaturesInExtent(extent);
expect(features.length).to.be(1);
expect(features[0].get('z')).to.be(0);
done();
});
});
});
});
});