maplibre-gl
Version:
BSD licensed community fork of mapbox-gl, a WebGL interactive maps library
1,354 lines (1,130 loc) • 75.4 kB
text/typescript
import {SourceCache} from './source_cache';
import {Map} from '../ui/map';
import {Source, addSourceType} from './source';
import {Tile} from './tile';
import {OverscaledTileID} from './tile_id';
import {Transform} from '../geo/transform';
import {LngLat} from '../geo/lng_lat';
import Point from '@mapbox/point-geometry';
import {Event, ErrorEvent, Evented} from '../util/evented';
import {extend} from '../util/util';
import {browser} from '../util/browser';
import {Dispatcher} from '../util/dispatcher';
import {TileBounds} from './tile_bounds';
import {sleep} from '../util/test/util';
import {TileCache} from './tile_cache';
class SourceMock extends Evented implements Source {
id: string;
minzoom: number;
maxzoom: number;
hasTile: (tileID: OverscaledTileID) => boolean;
sourceOptions: any;
type: string;
tileSize: number;
constructor(id: string, sourceOptions: any, _dispatcher: Dispatcher, eventedParent: Evented) {
super();
this.id = id;
this.minzoom = 0;
this.maxzoom = 22;
extend(this, sourceOptions);
this.sourceOptions = sourceOptions;
this.setEventedParent(eventedParent);
if (sourceOptions.hasTile) {
this.hasTile = sourceOptions.hasTile;
}
}
loadTile(tile: Tile): Promise<void> {
if (this.sourceOptions.expires) {
tile.setExpiryData({
expires: this.sourceOptions.expires
});
}
return new Promise(resolve => setTimeout(resolve, 0));
}
loaded() {
return true;
}
onAdd() {
if (this.sourceOptions.noLoad) return;
if (this.sourceOptions.error) {
this.fire(new ErrorEvent(this.sourceOptions.error));
} else {
this.fire(new Event('data', {dataType: 'source', sourceDataType: 'metadata'}));
}
}
async abortTile() {}
async unloadTile() {}
serialize() {}
hasTransition(): boolean {
return false;
}
}
// Add a mocked source type for use in these tests
function createSource(id: string, sourceOptions: any, _dispatcher: any, eventedParent: Evented) {
// allow tests to override mocked methods/properties by providing
// them in the source definition object that's given to Source.create()
const source = new SourceMock(id, sourceOptions, _dispatcher, eventedParent);
return source;
}
addSourceType('mock-source-type', createSource as any);
function createSourceCache(options?, used?) {
const sc = new SourceCache('id', extend({
tileSize: 512,
minzoom: 0,
maxzoom: 14,
type: 'mock-source-type'
}, options), {} as Dispatcher);
const scWithTestLogic = extend(sc, {
used: typeof used === 'boolean' ? used : true,
addTile(tileID: OverscaledTileID): Tile {
return this._addTile(tileID);
},
getCache(): TileCache {
return this._cache;
},
getTiles(): { [_: string]: Tile } {
return this._tiles;
},
updateLoadedSiblingTileCache(): void {
this._updateLoadedSiblingTileCache();
}
});
return scWithTestLogic;
}
afterEach(() => {
jest.clearAllMocks();
});
describe('SourceCache#addTile', () => {
test('loads tile when uncached', done => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
expect(tile.tileID).toEqual(tileID);
expect(tile.uses).toBe(0);
done();
};
sourceCache.onAdd(undefined);
sourceCache._addTile(tileID);
});
test('adds tile when uncached', done => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache({}).on('dataloading', (data) => {
expect(data.tile.tileID).toEqual(tileID);
expect(data.tile.uses).toBe(1);
done();
});
sourceCache.onAdd(undefined);
sourceCache._addTile(tileID);
});
test('updates feature state on added uncached tile', done => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
let updateFeaturesSpy;
const sourceCache = createSourceCache({});
sourceCache._source.loadTile = async (tile) => {
sourceCache.on('data', () => {
expect(updateFeaturesSpy).toHaveBeenCalledTimes(1);
done();
});
updateFeaturesSpy = jest.spyOn(tile, 'setFeatureState');
tile.state = 'loaded';
};
sourceCache.onAdd(undefined);
sourceCache._addTile(tileID);
});
test('uses cached tile', () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
let load = 0,
add = 0;
const sourceCache = createSourceCache({});
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loaded';
load++;
};
sourceCache.on('dataloading', () => { add++; });
const tr = new Transform();
tr.width = 512;
tr.height = 512;
sourceCache.updateCacheSize(tr);
sourceCache._addTile(tileID);
sourceCache._removeTile(tileID.key);
sourceCache._addTile(tileID);
expect(load).toBe(1);
expect(add).toBe(1);
});
test('updates feature state on cached tile', () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache({});
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loaded';
};
const tr = new Transform();
tr.width = 512;
tr.height = 512;
sourceCache.updateCacheSize(tr);
const tile = sourceCache._addTile(tileID);
const updateFeaturesSpy = jest.spyOn(tile, 'setFeatureState');
sourceCache._removeTile(tileID.key);
sourceCache._addTile(tileID);
expect(updateFeaturesSpy).toHaveBeenCalledTimes(1);
});
test('moves timers when adding tile from cache', () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const time = new Date();
time.setSeconds(time.getSeconds() + 5);
const sourceCache = createSourceCache();
sourceCache._setTileReloadTimer = (id) => {
sourceCache._timers[id] = setTimeout(() => {}, 0);
};
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loaded';
tile.getExpiryTimeout = () => 1000 * 60;
sourceCache._setTileReloadTimer(tileID.key, tile);
};
const tr = new Transform();
tr.width = 512;
tr.height = 512;
sourceCache.updateCacheSize(tr);
const id = tileID.key;
expect(sourceCache._timers[id]).toBeFalsy();
expect(sourceCache._cache.has(tileID)).toBeFalsy();
sourceCache._addTile(tileID);
expect(sourceCache._timers[id]).toBeTruthy();
expect(sourceCache._cache.has(tileID)).toBeFalsy();
sourceCache._removeTile(tileID.key);
expect(sourceCache._timers[id]).toBeFalsy();
expect(sourceCache._cache.has(tileID)).toBeTruthy();
sourceCache._addTile(tileID);
expect(sourceCache._timers[id]).toBeTruthy();
expect(sourceCache._cache.has(tileID)).toBeFalsy();
});
test('does not reuse wrapped tile', () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
let load = 0,
add = 0;
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loaded';
load++;
};
sourceCache.on('dataloading', () => { add++; });
const t1 = sourceCache._addTile(tileID);
const t2 = sourceCache._addTile(new OverscaledTileID(0, 1, 0, 0, 0));
expect(load).toBe(2);
expect(add).toBe(2);
expect(t1).not.toBe(t2);
});
test('should load tiles with identical overscaled Z but different canonical Z', () => {
const sourceCache = createSourceCache();
const tileIDs = [
new OverscaledTileID(1, 0, 0, 0, 0),
new OverscaledTileID(1, 0, 1, 0, 0),
new OverscaledTileID(1, 0, 1, 1, 0),
new OverscaledTileID(1, 0, 1, 0, 1),
new OverscaledTileID(1, 0, 1, 1, 1)
];
for (let i = 0; i < tileIDs.length; i++)
sourceCache._addTile(tileIDs[i]);
for (let i = 0; i < tileIDs.length; i++) {
const id = tileIDs[i];
const key = id.key;
expect(sourceCache._tiles[key]).toBeTruthy();
expect(sourceCache._tiles[key].tileID).toEqual(id);
}
});
});
describe('SourceCache#removeTile', () => {
test('removes tile', done => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache({});
sourceCache._addTile(tileID);
sourceCache.on('data', () => {
sourceCache._removeTile(tileID.key);
expect(sourceCache._tiles[tileID.key]).toBeFalsy();
done();
});
});
test('caches (does not unload) loaded tile', () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loaded';
};
sourceCache._source.unloadTile = jest.fn();
const tr = new Transform();
tr.width = 512;
tr.height = 512;
sourceCache.updateCacheSize(tr);
sourceCache._addTile(tileID);
sourceCache._removeTile(tileID.key);
expect(sourceCache._source.unloadTile).not.toHaveBeenCalled();
});
test('aborts and unloads unfinished tile', () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
let abort = 0,
unload = 0;
const sourceCache = createSourceCache();
sourceCache._source.abortTile = async (tile) => {
expect(tile.tileID).toEqual(tileID);
abort++;
};
sourceCache._source.unloadTile = async (tile) => {
expect(tile.tileID).toEqual(tileID);
unload++;
};
sourceCache._addTile(tileID);
sourceCache._removeTile(tileID.key);
expect(abort).toBe(1);
expect(unload).toBe(1);
});
test('_tileLoaded after _removeTile skips tile.added', () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async () => {
sourceCache._removeTile(tileID.key);
};
sourceCache.map = {painter: {crossTileSymbolIndex: '', tileExtentVAO: {}}} as any;
sourceCache._addTile(tileID);
});
test('fires dataabort event', async () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = () => {
// Do not call back in order to make sure the tile is removed before it is loaded.
return new Promise(() => {});
};
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const tile = sourceCache._addTile(tileID);
const abortPromise = sourceCache.once('dataabort');
sourceCache._removeTile(tileID.key);
const event = await abortPromise;
expect(event.dataType).toBe('source');
expect(event.tile).toBe(tile);
expect(event.coord).toBe(tileID);
});
test('does not fire dataabort event when the tile has already been loaded', () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loaded';
};
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
sourceCache._addTile(tileID);
const onAbort = jest.fn();
sourceCache.once('dataabort', onAbort);
sourceCache._removeTile(tileID.key);
expect(onAbort).toHaveBeenCalledTimes(0);
});
test('does not fire data event when the tile has already been aborted', () => {
const onData = jest.fn();
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
sourceCache.once('dataabort', () => {
tile.state = 'loaded';
expect(onData).toHaveBeenCalledTimes(0);
});
};
sourceCache.once('data', onData);
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
sourceCache._addTile(tileID);
sourceCache._removeTile(tileID.key);
});
});
describe('SourceCache / Source lifecycle', () => {
test('does not fire load or change before source load event', done => {
const sourceCache = createSourceCache({noLoad: true})
.on('data', () => done('test failed: data event fired'));
sourceCache.onAdd(undefined);
setTimeout(() => done(), 1);
});
test('forward load event', done => {
const sourceCache = createSourceCache({}).on('data', (e) => {
if (e.sourceDataType === 'metadata') done();
});
sourceCache.onAdd(undefined);
});
test('forward change event', done => {
const sourceCache = createSourceCache().on('data', (e) => {
if (e.sourceDataType === 'metadata') done();
});
sourceCache.onAdd(undefined);
sourceCache.getSource().fire(new Event('data'));
});
test('forward error event', done => {
const sourceCache = createSourceCache({error: 'Error loading source'}).on('error', (err) => {
expect(err.error).toBe('Error loading source');
done();
});
sourceCache.onAdd(undefined);
});
test('suppress 404 errors', done => {
const sourceCache = createSourceCache({status: 404, message: 'Not found'})
.on('error', () => done('test failed: error event fired'));
sourceCache.onAdd(undefined);
done();
});
test('loaded() true after source error', done => {
const sourceCache = createSourceCache({error: 'Error loading source'}).on('error', () => {
expect(sourceCache.loaded()).toBeTruthy();
done();
});
sourceCache.onAdd(undefined);
});
test('loaded() true after tile error', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 0;
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async () => {
throw new Error('Error loading tile');
};
sourceCache.on('data', (e) => {
if (e.dataType === 'source' && e.sourceDataType === 'metadata') {
sourceCache.update(transform);
}
}).on('error', () => {
expect(sourceCache.loaded()).toBeTruthy();
done();
});
sourceCache.onAdd(undefined);
});
test('loaded() false after source begins loading following error', done => {
const sourceCache = createSourceCache({error: 'Error loading source'}).on('error', () => {
sourceCache.on('dataloading', () => {
expect(sourceCache.loaded()).toBeFalsy();
done();
});
sourceCache.getSource().fire(new Event('dataloading'));
});
sourceCache.onAdd(undefined);
});
test('loaded() false when error occurs while source is not loaded', done => {
const sourceCache = createSourceCache({
error: 'Error loading source',
loaded() {
return false;
}
}).on('error', () => {
expect(sourceCache.loaded()).toBeFalsy();
done();
});
sourceCache.onAdd(undefined);
});
test('reloads tiles after a data event where source is updated', () => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 0;
const expected = [new OverscaledTileID(0, 0, 0, 0, 0).key, new OverscaledTileID(0, 0, 0, 0, 0).key];
expect.assertions(expected.length);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
expect(tile.tileID.key).toBe(expected.shift());
tile.state = 'loaded';
};
sourceCache.on('data', (e) => {
if (e.dataType === 'source' && e.sourceDataType === 'metadata') {
sourceCache.update(transform);
sourceCache.getSource().fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));
}
});
sourceCache.onAdd(undefined);
});
test('does not reload errored tiles', () => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 1;
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
// this transform will try to load the four tiles at z1 and a single z0 tile
// we only expect _reloadTile to be called with the 'loaded' z0 tile
tile.state = tile.tileID.canonical.z === 1 ? 'errored' : 'loaded';
};
const reloadTileSpy = jest.spyOn(sourceCache, '_reloadTile');
sourceCache.on('data', (e) => {
if (e.dataType === 'source' && e.sourceDataType === 'metadata') {
sourceCache.update(transform);
sourceCache.getSource().fire(new Event('data', {dataType: 'source', sourceDataType: 'content'}));
}
});
sourceCache.onAdd(undefined);
// we expect the source cache to have five tiles, but only to have reloaded one
expect(Object.keys(sourceCache._tiles)).toHaveLength(5);
expect(reloadTileSpy).toHaveBeenCalledTimes(1);
});
});
describe('SourceCache#update', () => {
test('loads no tiles if used is false', done => {
const transform = new Transform();
transform.resize(512, 512);
transform.zoom = 0;
const sourceCache = createSourceCache({}, false);
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([]);
done();
}
});
sourceCache.onAdd(undefined);
});
test('loads covering tiles', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 0;
const sourceCache = createSourceCache({});
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]);
done();
}
});
sourceCache.onAdd(undefined);
});
test('respects Source#hasTile method if it is present', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 1;
const sourceCache = createSourceCache({
hasTile: (coord) => (coord.canonical.x !== 0)
});
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getIds().sort()).toEqual([
new OverscaledTileID(1, 0, 1, 1, 0).key,
new OverscaledTileID(1, 0, 1, 1, 1).key
].sort());
done();
}
});
sourceCache.onAdd(undefined);
});
test('removes unused tiles', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 0;
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loaded';
};
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]);
transform.zoom = 1;
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([
new OverscaledTileID(1, 0, 1, 1, 1).key,
new OverscaledTileID(1, 0, 1, 0, 1).key,
new OverscaledTileID(1, 0, 1, 1, 0).key,
new OverscaledTileID(1, 0, 1, 0, 0).key
]);
done();
}
});
sourceCache.onAdd(undefined);
});
test('retains parent tiles for pending children', done => {
const transform = new Transform();
(transform as any)._test = 'retains';
transform.resize(511, 511);
transform.zoom = 0;
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = (tile.tileID.key === new OverscaledTileID(0, 0, 0, 0, 0).key) ? 'loaded' : 'loading';
};
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]);
transform.zoom = 1;
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([
new OverscaledTileID(0, 0, 0, 0, 0).key,
new OverscaledTileID(1, 0, 1, 1, 1).key,
new OverscaledTileID(1, 0, 1, 0, 1).key,
new OverscaledTileID(1, 0, 1, 1, 0).key,
new OverscaledTileID(1, 0, 1, 0, 0).key
]);
done();
}
});
sourceCache.onAdd(undefined);
});
test('retains parent tiles for pending children (wrapped)', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 0;
transform.center = new LngLat(360, 0);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = (tile.tileID.key === new OverscaledTileID(0, 1, 0, 0, 0).key) ? 'loaded' : 'loading';
};
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 1, 0, 0, 0).key]);
transform.zoom = 1;
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([
new OverscaledTileID(0, 1, 0, 0, 0).key,
new OverscaledTileID(1, 1, 1, 1, 1).key,
new OverscaledTileID(1, 1, 1, 0, 1).key,
new OverscaledTileID(1, 1, 1, 1, 0).key,
new OverscaledTileID(1, 1, 1, 0, 0).key
]);
done();
}
});
sourceCache.onAdd(undefined);
});
test('retains covered child tiles while parent tile is fading in', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 2;
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.timeAdded = Infinity;
tile.state = 'loaded';
tile.registerFadeDuration(100);
};
(sourceCache._source as any).type = 'raster';
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([
new OverscaledTileID(2, 0, 2, 2, 2).key,
new OverscaledTileID(2, 0, 2, 1, 2).key,
new OverscaledTileID(2, 0, 2, 2, 1).key,
new OverscaledTileID(2, 0, 2, 1, 1).key
]);
transform.zoom = 0;
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toHaveLength(5);
done();
}
});
sourceCache.onAdd(undefined);
});
test('retains a parent tile for fading even if a tile is partially covered by children', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 0;
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.timeAdded = Infinity;
tile.state = 'loaded';
tile.registerFadeDuration(100);
};
(sourceCache._source as any).type = 'raster';
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
transform.zoom = 2;
sourceCache.update(transform);
transform.zoom = 1;
sourceCache.update(transform);
expect(sourceCache._coveredTiles[(new OverscaledTileID(0, 0, 0, 0, 0).key)]).toBe(true);
done();
}
});
sourceCache.onAdd(undefined);
});
test('retain children for fading fadeEndTime is 0 (added but registerFadeDuration() is not called yet)', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 1;
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
// not setting fadeEndTime because class Tile default is 0, and need to be tested
tile.timeAdded = Date.now();
tile.state = 'loaded';
};
(sourceCache._source as any).type = 'raster';
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
transform.zoom = 0;
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toHaveLength(5);
done();
}
});
sourceCache.onAdd(undefined);
});
test('retains children when tile.fadeEndTime is in the future', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 1;
const fadeTime = 100;
const start = Date.now();
let time = start;
jest.spyOn(browser, 'now').mockImplementation(() => time);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.timeAdded = browser.now();
tile.state = 'loaded';
tile.fadeEndTime = browser.now() + fadeTime;
};
(sourceCache._source as any).type = 'raster';
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
// load children
sourceCache.update(transform);
transform.zoom = 0;
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toHaveLength(5);
time = start + 98;
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toHaveLength(5);
time = start + fadeTime + 1;
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toHaveLength(1);
done();
}
});
sourceCache.onAdd(undefined);
});
test('retains overscaled loaded children', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 16;
// use slightly offset center so that sort order is better defined
transform.center = new LngLat(-0.001, 0.001);
const sourceCache = createSourceCache({reparseOverscaled: true});
sourceCache._source.loadTile = async (tile) => {
tile.state = tile.tileID.overscaledZ === 16 ? 'loaded' : 'loading';
};
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toEqual([
new OverscaledTileID(16, 0, 14, 8192, 8192).key,
new OverscaledTileID(16, 0, 14, 8191, 8192).key,
new OverscaledTileID(16, 0, 14, 8192, 8191).key,
new OverscaledTileID(16, 0, 14, 8191, 8191).key
]);
transform.zoom = 15;
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toEqual([
new OverscaledTileID(16, 0, 14, 8192, 8192).key,
new OverscaledTileID(16, 0, 14, 8191, 8192).key,
new OverscaledTileID(16, 0, 14, 8192, 8191).key,
new OverscaledTileID(16, 0, 14, 8191, 8191).key
]);
done();
}
});
sourceCache.onAdd(undefined);
});
test('reassigns tiles for large jumps in longitude', done => {
const transform = new Transform();
transform.resize(511, 511);
transform.zoom = 0;
const sourceCache = createSourceCache({});
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
transform.center = new LngLat(360, 0);
const tileID = new OverscaledTileID(0, 1, 0, 0, 0);
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([tileID.key]);
const tile = sourceCache.getTile(tileID);
transform.center = new LngLat(0, 0);
const wrappedTileID = new OverscaledTileID(0, 0, 0, 0, 0);
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([wrappedTileID.key]);
expect(sourceCache.getTile(wrappedTileID)).toBe(tile);
done();
}
});
sourceCache.onAdd(undefined);
});
});
describe('SourceCache#_updateRetainedTiles', () => {
test('loads ideal tiles if they exist', () => {
const stateCache = {};
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = stateCache[tile.tileID.key] || 'errored';
};
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
const idealTile = new OverscaledTileID(1, 0, 1, 1, 1);
stateCache[idealTile.key] = 'loaded';
sourceCache._updateRetainedTiles([idealTile], 1);
expect(getTileSpy).not.toHaveBeenCalled();
expect(sourceCache.getIds()).toEqual([idealTile.key]);
});
test('retains all loaded children ', () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'errored';
};
const idealTile = new OverscaledTileID(3, 0, 3, 1, 2);
sourceCache._tiles[idealTile.key] = new Tile(idealTile, undefined);
sourceCache._tiles[idealTile.key].state = 'errored';
const loadedChildren = [
new OverscaledTileID(4, 0, 4, 2, 4),
new OverscaledTileID(4, 0, 4, 3, 4),
new OverscaledTileID(4, 0, 4, 2, 5),
new OverscaledTileID(5, 0, 5, 6, 10),
new OverscaledTileID(5, 0, 5, 7, 10),
new OverscaledTileID(5, 0, 5, 6, 11),
new OverscaledTileID(5, 0, 5, 7, 11)
];
for (const t of loadedChildren) {
sourceCache._tiles[t.key] = new Tile(t, undefined);
sourceCache._tiles[t.key].state = 'loaded';
}
const retained = sourceCache._updateRetainedTiles([idealTile], 3);
expect(Object.keys(retained).sort()).toEqual([
// parents are requested because ideal ideal tile is not completely covered by
// loaded child tiles
new OverscaledTileID(0, 0, 0, 0, 0),
new OverscaledTileID(2, 0, 2, 0, 1),
new OverscaledTileID(1, 0, 1, 0, 0),
idealTile
].concat(loadedChildren).map(t => t.key).sort());
});
test('adds parent tile if ideal tile errors and no child tiles are loaded', () => {
const stateCache = {};
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = stateCache[tile.tileID.key] || 'errored';
};
jest.spyOn(sourceCache, '_addTile');
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
const idealTiles = [new OverscaledTileID(1, 0, 1, 1, 1), new OverscaledTileID(1, 0, 1, 0, 1)];
stateCache[idealTiles[0].key] = 'loaded';
const retained = sourceCache._updateRetainedTiles(idealTiles, 1);
expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([
// when child tiles aren't found, check and request parent tile
new OverscaledTileID(0, 0, 0, 0, 0)
]);
// retained tiles include all ideal tiles and any parents that were loaded to cover
// non-existant tiles
expect(retained).toEqual({
// 1/0/1
'211': new OverscaledTileID(1, 0, 1, 0, 1),
// 1/1/1
'311': new OverscaledTileID(1, 0, 1, 1, 1),
// parent
'000': new OverscaledTileID(0, 0, 0, 0, 0)
});
});
test('don\'t use wrong parent tile', () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'errored';
};
const idealTile = new OverscaledTileID(2, 0, 2, 0, 0);
sourceCache._tiles[idealTile.key] = new Tile(idealTile, undefined);
sourceCache._tiles[idealTile.key].state = 'errored';
sourceCache._tiles[new OverscaledTileID(1, 0, 1, 1, 0).key] = new Tile(new OverscaledTileID(1, 0, 1, 1, 0), undefined);
sourceCache._tiles[new OverscaledTileID(1, 0, 1, 1, 0).key].state = 'loaded';
const addTileSpy = jest.spyOn(sourceCache, '_addTile');
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
sourceCache._updateRetainedTiles([idealTile], 2);
expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([
// parents
new OverscaledTileID(1, 0, 1, 0, 0), // not found
new OverscaledTileID(0, 0, 0, 0, 0) // not found
]);
expect(addTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([
// ideal tile
new OverscaledTileID(2, 0, 2, 0, 0),
// parents
new OverscaledTileID(1, 0, 1, 0, 0), // not found
new OverscaledTileID(0, 0, 0, 0, 0) // not found
]);
});
test('use parent tile when ideal tile is not loaded', () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
const idealTile = new OverscaledTileID(1, 0, 1, 0, 1);
const parentTile = new OverscaledTileID(0, 0, 0, 0, 0);
sourceCache._tiles[idealTile.key] = new Tile(idealTile, undefined);
sourceCache._tiles[idealTile.key].state = 'loading';
sourceCache._tiles[parentTile.key] = new Tile(parentTile, undefined);
sourceCache._tiles[parentTile.key].state = 'loaded';
const addTileSpy = jest.spyOn(sourceCache, '_addTile');
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
const retained = sourceCache._updateRetainedTiles([idealTile], 1);
expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([
// parents
new OverscaledTileID(0, 0, 0, 0, 0), // found
]);
expect(retained).toEqual({
// parent of ideal tile 0/0/0
'000': new OverscaledTileID(0, 0, 0, 0, 0),
// ideal tile id 1/0/1
'211': new OverscaledTileID(1, 0, 1, 0, 1)
});
addTileSpy.mockClear();
getTileSpy.mockClear();
// now make sure we don't retain the parent tile when the ideal tile is loaded
sourceCache._tiles[idealTile.key].state = 'loaded';
const retainedLoaded = sourceCache._updateRetainedTiles([idealTile], 1);
expect(getTileSpy).not.toHaveBeenCalled();
expect(retainedLoaded).toEqual({
// only ideal tile retained
'211': new OverscaledTileID(1, 0, 1, 0, 1)
});
});
test('don\'t load parent if all immediate children are loaded', () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
const idealTile = new OverscaledTileID(2, 0, 2, 1, 1);
const loadedTiles = [new OverscaledTileID(3, 0, 3, 2, 2), new OverscaledTileID(3, 0, 3, 3, 2), new OverscaledTileID(3, 0, 3, 2, 3), new OverscaledTileID(3, 0, 3, 3, 3)];
loadedTiles.forEach(t => {
sourceCache._tiles[t.key] = new Tile(t, undefined);
sourceCache._tiles[t.key].state = 'loaded';
});
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
const retained = sourceCache._updateRetainedTiles([idealTile], 2);
// parent tile isn't requested because all covering children are loaded
expect(getTileSpy).not.toHaveBeenCalled();
expect(Object.keys(retained)).toEqual([idealTile.key].concat(loadedTiles.map(t => t.key)));
});
test('prefer loaded child tiles to parent tiles', () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
const idealTile = new OverscaledTileID(1, 0, 1, 0, 0);
const loadedTiles = [new OverscaledTileID(0, 0, 0, 0, 0), new OverscaledTileID(2, 0, 2, 0, 0)];
loadedTiles.forEach(t => {
sourceCache._tiles[t.key] = new Tile(t, undefined);
sourceCache._tiles[t.key].state = 'loaded';
});
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
let retained = sourceCache._updateRetainedTiles([idealTile], 1);
expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([
// parent
new OverscaledTileID(0, 0, 0, 0, 0)
]);
expect(retained).toEqual({
// parent of ideal tile (0, 0, 0) (only partially covered by loaded child
// tiles, so we still need to load the parent)
'000': new OverscaledTileID(0, 0, 0, 0, 0),
// ideal tile id (1, 0, 0)
'011': new OverscaledTileID(1, 0, 1, 0, 0),
// loaded child tile (2, 0, 0)
'022': new OverscaledTileID(2, 0, 2, 0, 0)
});
getTileSpy.mockClear();
// remove child tile and check that it only uses parent tile
delete sourceCache._tiles['022'];
retained = sourceCache._updateRetainedTiles([idealTile], 1);
expect(retained).toEqual({
// parent of ideal tile (0, 0, 0) (only partially covered by loaded child
// tiles, so we still need to load the parent)
'000': new OverscaledTileID(0, 0, 0, 0, 0),
// ideal tile id (1, 0, 0)
'011': new OverscaledTileID(1, 0, 1, 0, 0)
});
});
test('don\'t use tiles below minzoom', () => {
const sourceCache = createSourceCache({minzoom: 2});
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
const idealTile = new OverscaledTileID(2, 0, 2, 0, 0);
const loadedTiles = [new OverscaledTileID(1, 0, 1, 0, 0)];
loadedTiles.forEach(t => {
sourceCache._tiles[t.key] = new Tile(t, undefined);
sourceCache._tiles[t.key].state = 'loaded';
});
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
const retained = sourceCache._updateRetainedTiles([idealTile], 2);
sleep(10);
expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([]);
expect(retained).toEqual({
// ideal tile id (2, 0, 0)
'022': new OverscaledTileID(2, 0, 2, 0, 0)
});
});
test('use overzoomed tile above maxzoom', () => {
const sourceCache = createSourceCache({maxzoom: 2});
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
const idealTile = new OverscaledTileID(2, 0, 2, 0, 0);
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
const retained = sourceCache._updateRetainedTiles([idealTile], 2);
expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([
// overzoomed child
new OverscaledTileID(3, 0, 2, 0, 0),
// parents
new OverscaledTileID(1, 0, 1, 0, 0),
new OverscaledTileID(0, 0, 0, 0, 0)
]);
expect(retained).toEqual({
// ideal tile id (2, 0, 0)
'022': new OverscaledTileID(2, 0, 2, 0, 0)
});
});
test('don\'t ascend multiple times if a tile is not found', () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
const idealTiles = [new OverscaledTileID(8, 0, 8, 0, 0), new OverscaledTileID(8, 0, 8, 1, 0)];
const getTileSpy = jest.spyOn(sourceCache, 'getTile');
sourceCache._updateRetainedTiles(idealTiles, 8);
expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([
// parent tile ascent
new OverscaledTileID(7, 0, 7, 0, 0),
new OverscaledTileID(6, 0, 6, 0, 0),
new OverscaledTileID(5, 0, 5, 0, 0),
new OverscaledTileID(4, 0, 4, 0, 0),
new OverscaledTileID(3, 0, 3, 0, 0),
new OverscaledTileID(2, 0, 2, 0, 0),
new OverscaledTileID(1, 0, 1, 0, 0),
new OverscaledTileID(0, 0, 0, 0, 0),
]);
getTileSpy.mockClear();
const loadedTiles = [new OverscaledTileID(4, 0, 4, 0, 0)];
loadedTiles.forEach(t => {
sourceCache._tiles[t.key] = new Tile(t, undefined);
sourceCache._tiles[t.key].state = 'loaded';
});
sourceCache._updateRetainedTiles(idealTiles, 8);
expect(getTileSpy.mock.calls.map((c) => { return c[0]; })).toEqual([
// parent tile ascent
new OverscaledTileID(7, 0, 7, 0, 0),
new OverscaledTileID(6, 0, 6, 0, 0),
new OverscaledTileID(5, 0, 5, 0, 0),
new OverscaledTileID(4, 0, 4, 0, 0), // tile is loaded, stops ascent
]);
});
test('Retain, then cancel loading tiles when zooming in', () => {
const sourceCache = createSourceCache();
// Disabling pending tile canceling (thus retaining) in Map mock:
const map = {cancelPendingTileRequestsWhileZooming: false} as Map;
sourceCache.onAdd(map);
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
let idealTiles = [new OverscaledTileID(9, 0, 9, 0, 0), new OverscaledTileID(9, 0, 9, 1, 0)];
sourceCache._updateRetainedTiles(idealTiles, 9);
idealTiles = [new OverscaledTileID(10, 0, 10, 0, 0), new OverscaledTileID(10, 0, 10, 1, 0)];
let retained = sourceCache._updateRetainedTiles(idealTiles, 10);
expect(Object.keys(retained).sort()).toEqual([
new OverscaledTileID(9, 0, 9, 0, 0).key, // retained
new OverscaledTileID(10, 0, 10, 0, 0).key,
new OverscaledTileID(10, 0, 10, 1, 0).key
]);
// Canceling pending tiles now via runtime map property:
map.cancelPendingTileRequestsWhileZooming = true;
retained = sourceCache._updateRetainedTiles(idealTiles, 10);
// Parent loading tiles from z=9 not retained:
expect(Object.keys(retained).sort()).toEqual([
new OverscaledTileID(10, 0, 10, 0, 0).key,
new OverscaledTileID(10, 0, 10, 1, 0).key
]);
});
test('Cancel, then retain, then cancel loading tiles when zooming in', () => {
const sourceCache = createSourceCache();
// Applying tile canceling default behavior in Map mock:
const map = {cancelPendingTileRequestsWhileZooming: true} as Map;
sourceCache.onAdd(map);
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
let idealTiles = [new OverscaledTileID(9, 0, 9, 0, 0), new OverscaledTileID(9, 0, 9, 1, 0)];
let retained = sourceCache._updateRetainedTiles(idealTiles, 9);
// Parent loading tiles from z=8 not retained
expect(Object.keys(retained).sort()).toEqual(
idealTiles.map((tile) => tile.key).sort()
);
idealTiles = [new OverscaledTileID(10, 0, 10, 0, 0), new OverscaledTileID(10, 0, 10, 1, 0)];
retained = sourceCache._updateRetainedTiles(idealTiles, 10);
// Parent loading tiles from z=9 not retained
expect(Object.keys(retained).sort()).toEqual(
idealTiles.map((tile) => tile.key).sort()
);
// Stopping tile canceling via runtime map property:
map.cancelPendingTileRequestsWhileZooming = false;
retained = sourceCache._updateRetainedTiles(idealTiles, 10);
expect(Object.keys(retained).sort()).toEqual([
new OverscaledTileID(9, 0, 9, 0, 0).key, // retained
new OverscaledTileID(10, 0, 10, 0, 0).key,
new OverscaledTileID(10, 0, 10, 1, 0).key
]);
// Resuming tile canceling via runtime map property:
map.cancelPendingTileRequestsWhileZooming = true;
const loadedTiles = idealTiles;
loadedTiles.forEach(t => {
sourceCache._tiles[t.key] = new Tile(t, undefined);
sourceCache._tiles[t.key].state = 'loaded';
});
idealTiles = [new OverscaledTileID(11, 0, 11, 0, 0), new OverscaledTileID(11, 0, 11, 1, 0)];
retained = sourceCache._updateRetainedTiles(idealTiles, 11);
// Parent loaded tile in the view port from z=10 was retained
expect(Object.keys(retained).sort()).toEqual([
new OverscaledTileID(10, 0, 10, 0, 0).key, // Parent loaded tile
new OverscaledTileID(11, 0, 11, 0, 0).key,
new OverscaledTileID(11, 0, 11, 1, 0).key
].sort());
});
test('Only retain loaded child tile when zooming out', () => {
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
let idealTiles = [new OverscaledTileID(7, 0, 7, 0, 0), new OverscaledTileID(7, 0, 7, 1, 0)];
let retained = sourceCache._updateRetainedTiles(idealTiles, 7);
// Client tiles from z=6 not retained
expect(Object.keys(retained).sort()).toEqual(
idealTiles.map((tile) => tile.key).sort()
);
idealTiles = [new OverscaledTileID(6, 0, 6, 0, 0), new OverscaledTileID(6, 0, 6, 1, 0)];
retained = sourceCache._updateRetainedTiles(idealTiles, 6);
// Client tiles from z=6 not retained
expect(Object.keys(retained).sort()).toEqual(
idealTiles.map((tile) => tile.key).sort()
);
const loadedTiles = idealTiles;
loadedTiles.forEach(t => {
sourceCache._tiles[t.key] = new Tile(t, undefined);
sourceCache._tiles[t.key].state = 'loaded';
});
idealTiles = [new OverscaledTileID(5, 0, 5, 0, 0), new OverscaledTileID(5, 0, 5, 1, 0)];
retained = sourceCache._updateRetainedTiles(idealTiles, 5);
// Child loaded tile in the view port from z=6 was retained
expect(Object.keys(retained).sort()).toEqual([
new OverscaledTileID(6, 0, 6, 0, 0).key,
new OverscaledTileID(6, 0, 6, 1, 0).key,
new OverscaledTileID(5, 0, 5, 0, 0).key,
new OverscaledTileID(5, 0, 5, 1, 0).key
].sort());
});
test('adds correct loaded parent tiles for overzoomed tiles', () => {
const sourceCache = createSourceCache({maxzoom: 7});
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loading';
};
const loadedTiles = [new OverscaledTileID(7, 0, 7, 0, 0), new OverscaledTileID(7, 0, 7, 1, 0)];
loadedTiles.forEach(t => {
sourceCache._tiles[t.key] = new Tile(t, undefined);
sourceCache._tiles[t.key].state = 'loaded';
});
const idealTiles = [new OverscaledTileID(8, 0, 7, 0, 0), new OverscaledTileID(8, 0, 7, 1, 0)];
const retained = sourceCache._updateRetainedTiles(idealTiles, 8);
expect(Object.keys(retained)).toEqual([
new OverscaledTileID(7, 0, 7, 1, 0).key,
new OverscaledTileID(8, 0, 7, 1, 0).key,
new OverscaledTileID(8, 0, 7, 0, 0).key,
new OverscaledTileID(7, 0, 7, 0, 0).key
]);
});
});
describe('SourceCache#clearTiles', () => {
test('unloads tiles', () => {
const coord = new OverscaledTileID(0, 0, 0, 0, 0);
let abort = 0,
unload = 0;
const sourceCache = createSourceCache();
sourceCache._source.abo