maplibre-gl
Version:
BSD licensed community fork of mapbox-gl, a WebGL interactive maps library
1,316 lines (1,092 loc) • 94 kB
text/typescript
import {describe, afterEach, test, expect, vi} from 'vitest';
import {SourceCache} from './source_cache';
import {type Map} from '../ui/map';
import {type Source, addSourceType} from './source';
import {Tile} from './tile';
import {CanonicalTileID, OverscaledTileID} from './tile_id';
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 {type Dispatcher} from '../util/dispatcher';
import {TileBounds} from './tile_bounds';
import {sleep, waitForEvent} from '../util/test/util';
import {type TileCache} from './tile_cache';
import {MercatorTransform} from '../geo/projection/mercator_transform';
import {GlobeTransform} from '../geo/projection/globe_transform';
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 sleep(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(() => {
vi.clearAllMocks();
});
describe('SourceCache.addTile', () => {
test('loads tile when uncached', () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache();
const spy = vi.fn();
sourceCache._source.loadTile = spy;
sourceCache.onAdd(undefined);
sourceCache._addTile(tileID);
expect(spy).toHaveBeenCalledTimes(1);
expect(spy.mock.calls[0][0].tileID).toEqual(tileID);
expect(spy.mock.calls[0][0].uses).toBe(1);
});
test('adds tile when uncached', async () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache({});
const dataLoadingPromise = sourceCache.once('dataloading');
sourceCache.onAdd(undefined);
sourceCache._addTile(tileID);
const data = await dataLoadingPromise;
expect(data.tile.tileID).toEqual(tileID);
expect(data.tile.uses).toBe(1);
});
test('updates feature state on added uncached tile', async () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
let updateFeaturesSpy;
const sourceCache = createSourceCache({});
let dataPromise: any;
sourceCache._source.loadTile = async (tile) => {
dataPromise = sourceCache.once('data');
updateFeaturesSpy = vi.spyOn(tile, 'setFeatureState');
tile.state = 'loaded';
};
sourceCache.onAdd(undefined);
sourceCache._addTile(tileID);
await dataPromise;
expect(updateFeaturesSpy).toHaveBeenCalledTimes(1);
});
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 MercatorTransform();
tr.resize(512, 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 MercatorTransform();
tr.resize(512, 512);
sourceCache.updateCacheSize(tr);
const tile = sourceCache._addTile(tileID);
const updateFeaturesSpy = vi.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 MercatorTransform();
tr.resize(512, 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', async () => {
const tileID = new OverscaledTileID(0, 0, 0, 0, 0);
const sourceCache = createSourceCache({});
sourceCache._addTile(tileID);
await sourceCache.once('data');
sourceCache._removeTile(tileID.key);
expect(sourceCache._tiles[tileID.key]).toBeFalsy();
});
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 = vi.fn();
const tr = new MercatorTransform();
tr.resize(512, 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 = vi.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 = vi.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', async () => {
const sourceCache = createSourceCache({noLoad: true});
const spy = vi.fn();
sourceCache.on('data', spy);
sourceCache.onAdd(undefined);
await sleep(1);
expect(spy).not.toHaveBeenCalled();
});
test('forward load event', async () => {
const sourceCache = createSourceCache({});
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await expect(dataPromise).resolves.toBeDefined();
});
test('forward change event', async () => {
const sourceCache = createSourceCache();
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
sourceCache.getSource().fire(new Event('data'));
await expect(dataPromise).resolves.toBeDefined();
});
test('forward error event', async () => {
const sourceCache = createSourceCache({error: 'Error loading source'});
const errorPromise = sourceCache.once('error');
sourceCache.onAdd(undefined);
const err = await errorPromise;
expect(err.error).toBe('Error loading source');
});
test('suppress 404 errors', () => {
const sourceCache = createSourceCache({status: 404, message: 'Not found'});
sourceCache.on('error', () => { throw new Error('test failed: error event fired'); });
sourceCache.onAdd(undefined);
});
test('loaded() true after source error', async () => {
const sourceCache = createSourceCache({error: 'Error loading source'});
const errorPromise = sourceCache.once('error');
sourceCache.onAdd(undefined);
await errorPromise;
expect(sourceCache.loaded()).toBeTruthy();
});
test('loaded() true after tile error', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(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);
}
});
const errorPromise = sourceCache.once('error');
sourceCache.onAdd(undefined);
await errorPromise;
expect(sourceCache.loaded()).toBeTruthy();
});
test('loaded() false after source begins loading following error', async () => {
const sourceCache = createSourceCache({error: 'Error loading source'});
const errorPromise = sourceCache.once('error');
sourceCache.onAdd(undefined);
await errorPromise;
const dataLoadingProimse = sourceCache.once('dataloading');
sourceCache.getSource().fire(new Event('dataloading'));
await dataLoadingProimse;
expect(sourceCache.loaded()).toBeFalsy();
});
test('loaded() false when error occurs while source is not loaded', async () => {
const sourceCache = createSourceCache({
error: 'Error loading source',
loaded() {
return false;
}
});
const errorPromise = sourceCache.once('error');
sourceCache.onAdd(undefined);
await errorPromise;
expect(sourceCache.loaded()).toBeFalsy();
});
test('reloads tiles after a data event where source is updated', () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(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 MercatorTransform();
transform.resize(511, 511);
transform.setZoom(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 = vi.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);
});
test('does reload errored tiles, if event is source data change', () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(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 = vi.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', sourceDataChanged: true}));
}
});
sourceCache.onAdd(undefined);
// We expect the source cache to have five tiles, and for all of them
// to be reloaded
expect(Object.keys(sourceCache._tiles)).toHaveLength(5);
expect(reloadTileSpy).toHaveBeenCalledTimes(5);
});
});
describe('SourceCache.update', () => {
test('loads no tiles if used is false', async () => {
const transform = new MercatorTransform();
transform.resize(512, 512);
transform.setZoom(0);
const sourceCache = createSourceCache({}, false);
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([]);
});
test('loads covering tiles', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(0);
const sourceCache = createSourceCache({});
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]);
});
test('respects Source.hasTile method if it is present', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(1);
const sourceCache = createSourceCache({
hasTile: (coord) => (coord.canonical.x !== 0)
});
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
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());
});
test('removes unused tiles', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(0);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = 'loaded';
};
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]);
transform.setZoom(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
]);
});
test('retains parent tiles for pending children', async () => {
const transform = new MercatorTransform();
(transform as any)._test = 'retains';
transform.resize(511, 511);
transform.setZoom(0);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.state = (tile.tileID.key === new OverscaledTileID(0, 0, 0, 0, 0).key) ? 'loaded' : 'loading';
};
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 0, 0, 0, 0).key]);
transform.setZoom(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
]);
});
test('retains parent tiles for pending children (wrapped)', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(0);
transform.setCenter(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';
};
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([new OverscaledTileID(0, 1, 0, 0, 0).key]);
transform.setZoom(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
]);
});
test('retains covered child tiles while parent tile is fading in', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(2);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.timeAdded = Infinity;
tile.state = 'loaded';
tile.registerFadeDuration(100);
};
(sourceCache._source as any).type = 'raster';
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
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.setZoom(0);
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toHaveLength(5);
});
test('retains a parent tile for fading even if a tile is partially covered by children', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(0);
const sourceCache = createSourceCache();
sourceCache._source.loadTile = async (tile) => {
tile.timeAdded = Infinity;
tile.state = 'loaded';
tile.registerFadeDuration(100);
};
(sourceCache._source as any).type = 'raster';
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
sourceCache.update(transform);
transform.setZoom(2);
sourceCache.update(transform);
transform.setZoom(1);
sourceCache.update(transform);
expect(sourceCache._coveredTiles[(new OverscaledTileID(0, 0, 0, 0, 0).key)]).toBe(true);
});
test('retain children for fading fadeEndTime is 0 (added but registerFadeDuration() is not called yet)', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(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';
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
sourceCache.update(transform);
transform.setZoom(0);
sourceCache.update(transform);
expect(sourceCache.getRenderableIds()).toHaveLength(5);
});
test('retains children when tile.fadeEndTime is in the future', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(1);
const fadeTime = 100;
const start = Date.now();
let time = start;
vi.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';
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
// load children
sourceCache.update(transform);
transform.setZoom(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);
});
test('retains children tiles for pending parents', () => {
const transform = new GlobeTransform();
transform.resize(511, 511);
transform.setZoom(1);
transform.setCenter(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) ? 'loading' : 'loaded';
};
sourceCache.on('data', (e) => {
if (e.sourceDataType === 'metadata') {
sourceCache.update(transform);
expect(sourceCache.getIds()).toEqual([
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
]);
transform.setZoom(0);
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
]);
}
});
sourceCache.onAdd(undefined);
});
test('retains overscaled loaded children', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(16);
// use slightly offset center so that sort order is better defined
transform.setCenter(new LngLat(-0.001, 0.001));
const sourceCache = createSourceCache({reparseOverscaled: true});
sourceCache._source.loadTile = async (tile) => {
tile.state = tile.tileID.overscaledZ === 16 ? 'loaded' : 'loading';
};
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
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.setZoom(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
]);
});
test('reassigns tiles for large jumps in longitude', async () => {
const transform = new MercatorTransform();
transform.resize(511, 511);
transform.setZoom(0);
const sourceCache = createSourceCache({});
const dataPromise = waitForEvent(sourceCache, 'data', e => e.sourceDataType === 'metadata');
sourceCache.onAdd(undefined);
await dataPromise;
transform.setCenter(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.setCenter(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);
});
});
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 = vi.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';
};
vi.spyOn(sourceCache, '_addTile');
const getTileSpy = vi.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 = vi.spyOn(sourceCache, '_addTile');
const getTileSpy = vi.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 = vi.spyOn(sourceCache, '_addTile');
const getTileSpy = vi.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 = vi.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 = vi.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 = vi.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 = vi.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 = vi.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