Files
iD/test/spec/services/panoramax.js
2025-05-06 11:25:32 +02:00

200 lines
7.6 KiB
JavaScript

import { setTimeout } from 'node:timers/promises';
describe('iD.servicePanoramax', function() {
const dimensions = [64, 64];
var context, panoramax;
const data = {
images:[{
loc: [10,0],
capture_time: '2020-01-01',
id: 'abc',
account_id: '123',
sequence_id: 'a1b2',
heading: 0,
image_path: '',
isPano: true,
model: 'camera',
}, {
loc: [10,1],
capture_time: '2020-02-01',
id: 'def',
account_id: 'c3d4',
sequence_id: '',
heading: 0,
image_path: '',
isPano: true,
model: 'camera',
}, {
loc: [10,2],
capture_time: '2020-02-01',
id: 'ghi',
account_id: '789',
sequence_id: 'e5f6',
heading: 0,
image_path: '',
isPano: true,
model: 'camera',
}],
};
before(function() {
iD.services.panoramax = iD.servicePanoramax;
fetchMock.reset();
});
after(function() {
delete iD.services.panoramax;
});
beforeEach(function() {
context = iD.coreContext().assetPath('../dist/').init();
context.projection
.scale(iD.geoZoomToScale(14))
.translate([-116508, 0]) // 10,0
.clipExtent([[0,0], dimensions]);
panoramax = iD.services.panoramax;
panoramax.reset();
fetchMock.reset();
// never resolve
fetchMock.mock(/api\.panoramax\.xyz/, new Promise(() => {}));
});
afterEach(function() {
fetchMock.reset();
});
describe('#init', function() {
it('Initializes cache one time', function() {
var cache = panoramax.cache();
expect(cache).to.have.property('images');
expect(cache).to.have.property('sequences');
panoramax.init();
var cache2 = panoramax.cache();
expect(cache).to.equal(cache2);
});
});
describe('#reset', function() {
it('resets cache', function() {
panoramax.cache().foo = 'bar';
panoramax.setActiveImage(context, {key: 'baz'});
panoramax.reset();
expect(panoramax.cache()).to.not.have.property('foo');
});
});
describe('#loadImages', function() {
it('does not load images around null island', async () => {
var spy = sinon.spy();
fetchMock.reset();
fetchMock.mock(new RegExp('/api\.panoramax\.xyz/'), {
body: JSON.stringify(data),
status: 200,
headers: { 'Content-Type': 'application/json' }
});
context.projection
.scale(iD.geoZoomToScale(15))
.translate([0, 0]);
panoramax.on('loadedImages', spy);
panoramax.loadImages(context.projection);
await setTimeout(200);
expect(spy).to.have.been.not.called;
expect(fetchMock.calls().length).to.eql(0); // no tile requests of any kind
});
it('handle API error response', async ({ expect }) => {
fetchMock.reset();
fetchMock.mock('/api\.panoramax\.xyz/', 500);
const promise = panoramax.getImageData('collection1', 'image1');
await expect(promise).rejects.toThrowError();
});
});
describe('#images', function() {
it('returns images in the visible map area', function() {
var features = [
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '0', loc: [10,0], heading: 90, sequence_id: '100', account_id: '0' } },
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '1', loc: [10,0], heading: 90, sequence_id: '100', account_id: '1' } },
{ minX: 10, minY: 1, maxX: 10, maxY: 1, data: { id: '2', loc: [10,1], heading: 90, sequence_id: '100', account_id: '2' } }
];
panoramax.cache().images.rtree.load(features);
var res = panoramax.images(context.projection);
expect(res).to.deep.eql([
{ id: '0', loc: [10,0], heading: 90, sequence_id: '100', account_id: '0' },
{ id: '1', loc: [10,0], heading: 90, sequence_id: '100', account_id: '1' }
]);
});
it('limits results no more than 5 stacked images in one spot', function() {
var features = [
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '0', loc: [10,0], heading: 90, sequence_id: '100', account_id: '0' } },
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '1', loc: [10,0], heading: 90, sequence_id: '100', account_id: '1' } },
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '2', loc: [10,0], heading: 90, sequence_id: '100', account_id: '2' } },
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '3', loc: [10,0], heading: 90, sequence_id: '100', account_id: '3' } },
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '4', loc: [10,0], heading: 90, sequence_id: '100', account_id: '4' } },
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '5', loc: [10,0], heading: 90, sequence_id: '100', account_id: '5' } }
];
panoramax.cache().images.rtree.load(features);
var res = panoramax.images(context.projection);
expect(res).to.have.length.of.at.most(5);
});
it('handle invalid image data', function() {
const invalidImage = { id: null, sequence_id: null };
panoramax.setActiveImage(invalidImage);
expect(panoramax.getActiveImage()).to.be.null;
});
it('return empty array when no images are available', function() {
const result = panoramax.images(context.projection);
expect(result).to.deep.equal([]);
});
it('load images quickly under normal conditions', function() {
const start = performance.now();
panoramax.loadImages(context.projection);
const duration = performance.now() - start;
expect(duration).to.be.lessThan(1000);
});
});
describe('#sequences', function() {
it('returns sequence linestrings in the visible map area', function() {
var features = [
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '0', loc: [10,0], heading: 90, sequence_id: '100', account_id: '0' } },
{ minX: 10, minY: 0, maxX: 10, maxY: 0, data: { id: '1', loc: [10,0], heading: 90, sequence_id: '100', account_id: '1' } },
{ minX: 10, minY: 1, maxX: 10, maxY: 1, data: { id: '2', loc: [10,1], heading: 90, sequence_id: '100', account_id: '2' } }
];
panoramax.cache().images.rtree.load(features);
panoramax.cache().sequences.lineString['100'] = { rotation: 0, images: [ features[0].data, features[1].data, features[2].data ] };
var res = panoramax.sequences(context.projection, 15);
expect(res).to.deep.eql([{
rotation: 0, images: [features[0].data, features[1].data, features[2].data]
}]);
});
});
describe('#selectedImage', function() {
it('sets and gets selected image', function() {
const photo = { id: 'foo', sequence_id: '100'};
panoramax.cache().images = { forImageId: { foo: photo }};
panoramax.selectImage(context, 'foo');
expect(panoramax.getActiveImage().id).to.eql(photo.id);
});
});
});