maplibre-gl
Version:
BSD licensed community fork of mapbox-gl, a WebGL interactive maps library
251 lines (215 loc) • 9.69 kB
text/typescript
import {describe, beforeEach, test, expect, vi} from 'vitest';
import {ImageSource} from './image_source';
import {Evented} from '../util/evented';
import {type IReadonlyTransform} from '../geo/transform_interface';
import {extend, MAX_TILE_ZOOM} from '../util/util';
import {type FakeServer, fakeServer} from 'nise';
import {type RequestManager} from '../util/request_manager';
import {sleep, stubAjaxGetImage, waitForEvent} from '../util/test/util';
import {Tile} from './tile';
import {OverscaledTileID} from './tile_id';
import {type Texture} from '../render/texture';
import type {ImageSourceSpecification} from '@maplibre/maplibre-gl-style-spec';
import {MercatorTransform} from '../geo/projection/mercator_transform';
function createSource(options) {
options = extend({
coordinates: [[0, 0], [1, 0], [1, 1], [0, 1]]
}, options);
const source = new ImageSource('id', options, {} as any, options.eventedParent);
return source;
}
class StubMap extends Evented {
transform: IReadonlyTransform;
painter: any;
_requestManager: RequestManager;
constructor() {
super();
this.transform = new MercatorTransform();
this._requestManager = {
transformRequest: (url) => {
return {url};
}
} as any as RequestManager;
this.painter = {
context: {
gl: {}
}
};
}
}
describe('ImageSource', () => {
stubAjaxGetImage(undefined);
let server: FakeServer;
beforeEach(() => {
global.fetch = null;
server = fakeServer.create();
server.respondWith(new ArrayBuffer(1));
server.respondWith('/missing-image.png', [404, {}, '']);
});
test('constructor', () => {
const source = createSource({url: '/image.png'});
expect(source.minzoom).toBe(0);
expect(source.maxzoom).toBe(22);
expect(source.tileSize).toBe(512);
});
test('fires dataloading event', async () => {
const source = createSource({url: '/image.png'});
source.on('dataloading', (e) => {
expect(e.dataType).toBe('source');
});
source.onAdd(new StubMap() as any);
server.respond();
await sleep(0);
expect(source.image).toBeTruthy();
});
test('transforms url request', () => {
const source = createSource({url: '/image.png'});
const map = new StubMap() as any;
const spy = vi.spyOn(map._requestManager, 'transformRequest');
source.onAdd(map);
server.respond();
expect(spy).toHaveBeenCalledTimes(1);
expect(spy.mock.calls[0][0]).toBe('/image.png');
expect(spy.mock.calls[0][1]).toBe('Image');
});
test('updates url from updateImage', () => {
const source = createSource({url: '/image.png'});
const map = new StubMap() as any;
const spy = vi.spyOn(map._requestManager, 'transformRequest');
source.onAdd(map);
server.respond();
expect(spy).toHaveBeenCalledTimes(1);
expect(spy.mock.calls[0][0]).toBe('/image.png');
expect(spy.mock.calls[0][1]).toBe('Image');
source.updateImage({url: '/image2.png'});
server.respond();
expect(spy).toHaveBeenCalledTimes(2);
expect(spy.mock.calls[1][0]).toBe('/image2.png');
expect(spy.mock.calls[1][1]).toBe('Image');
});
test('sets coordinates', () => {
const source = createSource({url: '/image.png'});
const map = new StubMap() as any;
source.onAdd(map);
server.respond();
const beforeSerialized = source.serialize();
expect(beforeSerialized.coordinates).toEqual([[0, 0], [1, 0], [1, 1], [0, 1]]);
source.setCoordinates([[0, 0], [-1, 0], [-1, -1], [0, -1]]);
const afterSerialized = source.serialize();
expect(afterSerialized.coordinates).toEqual([[0, 0], [-1, 0], [-1, -1], [0, -1]]);
});
test('sets coordinates via updateImage', async () => {
const source = createSource({url: '/image.png'});
const map = new StubMap() as any;
source.onAdd(map);
server.respond();
const beforeSerialized = source.serialize();
expect(beforeSerialized.coordinates).toEqual([[0, 0], [1, 0], [1, 1], [0, 1]]);
source.updateImage({
url: '/image2.png',
coordinates: [[0, 0], [-1, 0], [-1, -1], [0, -1]]
});
server.respond();
await sleep(0);
const afterSerialized = source.serialize();
expect(afterSerialized.coordinates).toEqual([[0, 0], [-1, 0], [-1, -1], [0, -1]]);
});
test('fires data event when content is loaded', async () => {
const source = createSource({url: '/image.png'});
const promise = waitForEvent(source, 'data', (e) => e.dataType === 'source' && e.sourceDataType === 'content');
source.onAdd(new StubMap() as any);
server.respond();
await promise;
expect(typeof source.tileID == 'object').toBeTruthy();
});
test('fires data event when metadata is loaded', async () => {
const source = createSource({url: '/image.png'});
const promise = waitForEvent(source, 'data', (e) => e.dataType === 'source' && e.sourceDataType === 'metadata');
source.onAdd(new StubMap() as any);
server.respond();
await expect(promise).resolves.toBeDefined();
});
test('fires idle event on prepare call when there is at least one not loaded tile', async () => {
const source = createSource({url: '/image.png'});
const tile = new Tile(new OverscaledTileID(1, 0, 1, 0, 0), 512);
const promise = waitForEvent(source, 'data', (e) => e.dataType === 'source' && e.sourceDataType === 'idle');
source.onAdd(new StubMap() as any);
server.respond();
source.tiles[String(tile.tileID.wrap)] = tile;
source.image = new ImageBitmap();
// assign dummies directly so we don't need to stub the gl things
source.texture = {} as Texture;
source.prepare();
await promise;
expect(tile.state).toBe('loaded');
});
test('serialize url and coordinates', () => {
const source = createSource({url: '/image.png'});
const serialized = source.serialize() as ImageSourceSpecification;
expect(serialized.type).toBe('image');
expect(serialized.url).toBe('/image.png');
expect(serialized.coordinates).toEqual([[0, 0], [1, 0], [1, 1], [0, 1]]);
});
test('allows using updateImage before initial image is loaded', async () => {
const map = new StubMap() as any;
const source = createSource({url: '/image.png', eventedParent: map});
// Suppress errors because we're aborting when updating.
map.on('error', () => {});
source.onAdd(map);
expect(source.image).toBeUndefined();
source.updateImage({url: '/image2.png'});
server.respond();
await sleep(10);
expect(source.image).toBeTruthy();
});
test('cancels request if updateImage is used', () => {
const map = new StubMap() as any;
const source = createSource({url: '/image.png', eventedParent: map});
// Suppress errors because we're aborting.
map.on('error', () => {});
source.onAdd(map);
const spy = vi.spyOn(server.requests[0] as any, 'abort');
source.updateImage({url: '/image2.png'});
expect(spy).toHaveBeenCalled();
});
test('marks the source as loaded when the request has received a response', async () => {
const map = new StubMap() as any;
const source = createSource({url: '/image.png', eventedParent: map});
expect(source.loaded()).toBe(false);
source.onAdd(map);
server.respond();
await sleep(0);
expect(source.loaded()).toBe(true);
const missingImagesource = createSource({url: '/missing-image.png', eventedParent: map});
// Suppress errors as we're loading a missing image.
map.on('error', () => {});
expect(missingImagesource.loaded()).toBe(false);
missingImagesource.onAdd(map);
server.respond();
await sleep(0);
expect(missingImagesource.loaded()).toBe(true);
});
describe('terrainTileRanges', () => {
test('sets tile ranges for all zoom levels', () => {
const source = createSource({url: '/image.png'});
const map = new StubMap() as any;
source.onAdd(map);
server.respond();
source.setCoordinates([[-10, 10], [10, 10], [10, -10], [-10, -10]]);
for (let z = 0; z <= MAX_TILE_ZOOM; z++) {
expect(source.terrainTileRanges[z]).toBeDefined();
}
});
test('calculates tile ranges properly', () => {
const source = createSource({url: '/image.png'});
const map = new StubMap() as any;
source.onAdd(map);
server.respond();
source.setCoordinates([[11.39585,47.30074],[11.46585,47.30074],[11.46585,47.25074],[11.39585,47.25074]]);
expect(source.terrainTileRanges[9]).toEqual({minTileX: 272, minTileY: 179, maxTileX: 272, maxTileY: 179});
expect(source.terrainTileRanges[10]).toEqual({minTileX: 544, minTileY: 358, maxTileX: 544, maxTileY: 359});
expect(source.terrainTileRanges[11]).toEqual({minTileX: 1088, minTileY: 717, maxTileX: 1089, maxTileY: 718});
expect(source.terrainTileRanges[12]).toEqual({minTileX: 2177, minTileY: 1435, maxTileX: 2178, maxTileY: 1436});
});
});
});