From a1df915da44ab2824aef8d8ed70e26da6db7e0bb Mon Sep 17 00:00:00 2001 From: Oscar Lorentzon Date: Wed, 9 Jun 2021 21:23:05 +0200 Subject: [PATCH] fix: viewer configuration as observables Avoid setting static properties which affects all future Viewers created in the same app. Use observables instead. --- .../attribution/AttributionComponent.ts | 25 ++-- src/component/cover/CoverComponent.ts | 17 ++- src/component/image/ImageComponent.ts | 117 ++++++++++-------- src/component/slider/SliderComponent.ts | 108 +++++++++------- src/viewer/ConfigurationService.ts | 29 +++++ src/viewer/Container.ts | 19 +-- src/viewer/PlayService.ts | 1 - src/viewer/Viewer.ts | 3 - src/viewer/ViewerConfiguration.ts | 40 ------ styles/cover.css | 19 ++- .../helper/ConfigurationServiceMockCreator.ts | 26 ++++ test/helper/ContainerMockCreator.ts | 6 + test/util/ViewerConfiguration.test.ts | 26 ---- test/viewer/ConfigurationService.test.ts | 89 +++++++++++++ 14 files changed, 327 insertions(+), 198 deletions(-) create mode 100644 src/viewer/ConfigurationService.ts delete mode 100644 src/viewer/ViewerConfiguration.ts create mode 100644 test/helper/ConfigurationServiceMockCreator.ts delete mode 100644 test/util/ViewerConfiguration.test.ts create mode 100644 test/viewer/ConfigurationService.test.ts diff --git a/src/component/attribution/AttributionComponent.ts b/src/component/attribution/AttributionComponent.ts index 2d3f3a6e6..8d19b6552 100644 --- a/src/component/attribution/AttributionComponent.ts +++ b/src/component/attribution/AttributionComponent.ts @@ -6,7 +6,6 @@ import { map } from "rxjs/operators"; import { Image } from "../../graph/Image"; import { ViewportSize } from "../../render/interfaces/ViewportSize"; import { VirtualNodeHash } from "../../render/interfaces/VirtualNodeHash"; -import { ViewerConfiguration } from "../../viewer/ViewerConfiguration"; import { Component } from "../Component"; import { ComponentConfiguration } from "../interfaces/ComponentConfiguration"; @@ -18,13 +17,15 @@ export class AttributionComponent extends Component { protected _activate(): void { this._subscriptions.push( observableCombineLatest( + this._container.configurationService.exploreUrl$, this._navigator.stateService.currentImage$, this._container.renderService.size$).pipe( map( - ([image, size]: [Image, ViewportSize]): VirtualNodeHash => { + ([exploreUrl, image, size]: [string, Image, ViewportSize]): VirtualNodeHash => { const attribution = this._makeAttribution( image.creatorUsername, + exploreUrl, image.id, image.capturedAt, size.width); @@ -44,8 +45,13 @@ export class AttributionComponent extends Component { return {}; } + private makeImageUrl(exploreUrl: string, id: string): string { + return `${exploreUrl}/app/?pKey=${id}&focus=photo`; + } + private _makeAttribution( creatorUsername: string, + exploreUrl: string, imageId: string, capturedAt: number, viewportWidth: number) @@ -53,7 +59,7 @@ export class AttributionComponent extends Component { const compact = viewportWidth <= 640; const date = this._makeDate(capturedAt, compact); - const by = this._makeBy(creatorUsername, imageId, compact); + const by = this._makeBy(creatorUsername, exploreUrl, imageId, compact); const compactClass = compact ? ".mapillary-attribution-compact" : ""; @@ -66,6 +72,7 @@ export class AttributionComponent extends Component { private _makeBy( creatorUsername: string, + exploreUrl: string, imageId: string, compact: boolean): vd.VNode[] { @@ -73,18 +80,19 @@ export class AttributionComponent extends Component { "div.mapillary-attribution-logo", []); return creatorUsername ? - this._makeCreatorBy(icon, creatorUsername, imageId, compact) : - this._makeGeneralBy(icon, imageId, compact); + this._makeCreatorBy(icon, creatorUsername, exploreUrl, imageId, compact) : + this._makeGeneralBy(icon, exploreUrl, imageId, compact); } private _makeCreatorBy( icon: vd.VNode, creatorUsername: string, + exploreUrl: string, imageId: string, compact: boolean): vd.VNode[] { const mapillary = vd.h( "a.mapillary-attribution-icon-container", - { href: ViewerConfiguration.explore, rel: "noreferrer", target: "_blank" }, + { href: exploreUrl, rel: "noreferrer", target: "_blank" }, [icon]); const content = compact ? @@ -97,7 +105,7 @@ export class AttributionComponent extends Component { const image = vd.h( "a.mapillary-attribution-image-container", { - href: ViewerConfiguration.exploreImage(imageId), + href: this.makeImageUrl(exploreUrl, imageId), rel: "noreferrer", target: "_blank", }, @@ -108,6 +116,7 @@ export class AttributionComponent extends Component { private _makeGeneralBy( icon: vd.VNode, + exploreUrl: string, imageId: string, compact: boolean): vd.VNode[] { @@ -134,7 +143,7 @@ export class AttributionComponent extends Component { const image = vd.h( "a.mapillary-attribution-image-container", { - href: ViewerConfiguration.exploreImage(imageId), + href: this.makeImageUrl(exploreUrl, imageId), rel: "noreferrer", target: "_blank", }, diff --git a/src/component/cover/CoverComponent.ts b/src/component/cover/CoverComponent.ts index 8ec2731ec..cbff843c4 100644 --- a/src/component/cover/CoverComponent.ts +++ b/src/component/cover/CoverComponent.ts @@ -26,7 +26,6 @@ import { MapillaryError } from "../../error/MapillaryError"; import { Image as MImage } from "../../graph/Image"; import { ViewportSize } from "../../render/interfaces/ViewportSize"; import { VirtualNodeHash } from "../../render/interfaces/VirtualNodeHash"; -import { ViewerConfiguration } from "../../viewer/ViewerConfiguration"; import { Container } from "../../viewer/Container"; import { Navigator } from "../../viewer/Navigator"; import { ImagesContract } from "../../api/contracts/ImagesContract"; @@ -128,9 +127,11 @@ export class CoverComponent extends Component { subs.push(observableCombineLatest( this._configuration$, + this._container.configurationService.exploreUrl$, this._container.renderService.size$).pipe( map( - ([configuration, size]: [CoverConfiguration, ViewportSize]): VirtualNodeHash => { + ([configuration, exploreUrl, size]: + [CoverConfiguration, string, ViewportSize]): VirtualNodeHash => { if (!configuration.src) { return { name: this._name, vNode: vd.h("div", []) }; } @@ -147,7 +148,7 @@ export class CoverComponent extends Component { const container: vd.VNode = vd.h( "div.mapillary-cover-container" + compactClass, - [this._getCoverButtonVNode(configuration)]); + [this._getCoverButtonVNode(configuration, exploreUrl)]); return { name: this._name, vNode: container }; })) @@ -162,13 +163,19 @@ export class CoverComponent extends Component { return { state: CoverState.Visible }; } - private _getCoverButtonVNode(configuration: CoverConfiguration): vd.VNode { + private _getCoverButtonVNode( + configuration: CoverConfiguration, + exploreUrl: string): vd.VNode { + const cover: string = configuration.state === CoverState.Loading ? "div.mapillary-cover.mapillary-cover-loading" : "div.mapillary-cover"; const coverButton: vd.VNode = vd.h( "div.mapillary-cover-button", [vd.h("div.mapillary-cover-button-icon", [])]); - const coverLogo: vd.VNode = vd.h("a.mapillary-cover-logo", { href: ViewerConfiguration.explore, target: "_blank" }, []); + const coverLogo: vd.VNode = vd.h( + "a.mapillary-cover-logo", + { href: exploreUrl, target: "_blank" }, + []); const coverIndicator: vd.VNode = vd.h( "div.mapillary-cover-indicator", { onclick: (): void => { this.configure({ state: CoverState.Loading }); } }, diff --git a/src/component/image/ImageComponent.ts b/src/component/image/ImageComponent.ts index 4e16c4cbf..49f88c7f7 100644 --- a/src/component/image/ImageComponent.ts +++ b/src/component/image/ImageComponent.ts @@ -46,10 +46,8 @@ import { RegionOfInterestCalculator } import { TextureProvider } from "../../tile/TextureProvider"; import { ComponentConfiguration } from "../interfaces/ComponentConfiguration"; import { Transform } from "../../geo/Transform"; -import { ViewerConfiguration } from "../../viewer/ViewerConfiguration"; import { ComponentName } from "../ComponentName"; import { State } from "../../state/State"; -import { Camera } from "three"; interface ImageGLRendererOperation { (renderer: ImageGLRenderer): ImageGLRenderer; @@ -170,9 +168,14 @@ export class ImageComponent extends Component { })) .subscribe(this._rendererOperation$)); - const textureProvider$ = this._navigator.stateService.currentState$ - .pipe( - filter(() => ViewerConfiguration.imageTiling), + const textureProvider$ = + this._container.configurationService.imageTiling$.pipe( + switchMap( + (active): Observable => { + return active ? + this._navigator.stateService.currentState$ : + new Subject(); + }), distinctUntilChanged( undefined, (frame: AnimationFrame): string => { @@ -220,56 +223,60 @@ export class ImageComponent extends Component { previous.abort(); })); - const roiTrigger$ = ViewerConfiguration.imageTiling ? - observableCombineLatest( - this._navigator.stateService.state$, - this._navigator.stateService.inTranslation$) - .pipe( - switchMap( - ([state, inTranslation]: [State, boolean]) => { - const streetState = - state === State.Traversing || - state === State.Waiting || - state === State.WaitingInteractively; - const active = streetState && !inTranslation; - return active ? - this._container.renderService.renderCameraFrame$ : - observableEmpty(); - }), - map( - (camera: RenderCamera): PositionLookat => { - return { - camera, - height: camera.size.height.valueOf(), - lookat: camera.camera.lookat.clone(), - width: camera.size.width.valueOf(), - zoom: camera.zoom.valueOf(), - }; - }), - pairwise(), - map( - ([pl0, pl1]: [PositionLookat, PositionLookat]) - : StalledCamera => { - const stalled = - pl0.width === pl1.width && - pl0.height === pl1.height && - pl0.zoom === pl1.zoom && - pl0.lookat.equals(pl1.lookat); - - return { camera: pl1.camera, stalled }; - }), - distinctUntilChanged( - (x, y): boolean => { - return x.stalled === y.stalled; - }), - filter( - (camera: StalledCamera): boolean => { - return camera.stalled; - }), - withLatestFrom( - this._container.renderService.size$, - this._navigator.stateService.currentTransform$)) : - observableEmpty(); + const roiTrigger$ = + this._container.configurationService.imageTiling$.pipe( + switchMap( + (active): Observable<[State, boolean]> => { + return active ? + observableCombineLatest( + this._navigator.stateService.state$, + this._navigator.stateService.inTranslation$) : + new Subject(); + }), + switchMap( + ([state, inTranslation]: [State, boolean]) => { + const streetState = + state === State.Traversing || + state === State.Waiting || + state === State.WaitingInteractively; + const active = streetState && !inTranslation; + return active ? + this._container.renderService.renderCameraFrame$ : + observableEmpty(); + }), + map( + (camera: RenderCamera): PositionLookat => { + return { + camera, + height: camera.size.height.valueOf(), + lookat: camera.camera.lookat.clone(), + width: camera.size.width.valueOf(), + zoom: camera.zoom.valueOf(), + }; + }), + pairwise(), + map( + ([pl0, pl1]: [PositionLookat, PositionLookat]) + : StalledCamera => { + const stalled = + pl0.width === pl1.width && + pl0.height === pl1.height && + pl0.zoom === pl1.zoom && + pl0.lookat.equals(pl1.lookat); + + return { camera: pl1.camera, stalled }; + }), + distinctUntilChanged( + (x, y): boolean => { + return x.stalled === y.stalled; + }), + filter( + (camera: StalledCamera): boolean => { + return camera.stalled; + }), + withLatestFrom( + this._container.renderService.size$, + this._navigator.stateService.currentTransform$)); subs.push(textureProvider$.pipe( switchMap( diff --git a/src/component/slider/SliderComponent.ts b/src/component/slider/SliderComponent.ts index f842f92aa..b8de183f4 100644 --- a/src/component/slider/SliderComponent.ts +++ b/src/component/slider/SliderComponent.ts @@ -64,7 +64,6 @@ import { SliderGLRenderer } from "./SliderGLRenderer"; import { Transform } from "../../geo/Transform"; import { SliderDOMRenderer } from "./SliderDOMRenderer"; import { isSpherical } from "../../geo/Geo"; -import { ViewerConfiguration } from "../../viewer/ViewerConfiguration"; import { ComponentName } from "../ComponentName"; /** @@ -396,9 +395,14 @@ export class SliderComponent extends Component { })); - const textureProvider$ = this._navigator.stateService.currentState$ - .pipe( - filter(() => ViewerConfiguration.imageTiling), + const textureProvider$ = + this._container.configurationService.imageTiling$.pipe( + switchMap( + (active): Observable => { + return active ? + this._navigator.stateService.currentState$ : + new Subject(); + }), distinctUntilChanged( undefined, (frame: AnimationFrame): string => { @@ -449,10 +453,16 @@ export class SliderComponent extends Component { previous.abort(); })); - const roiTrigger$ = observableCombineLatest( - this._container.renderService.renderCameraFrame$, - this._container.renderService.size$.pipe(debounceTime(250))).pipe( - filter(() => ViewerConfiguration.imageTiling), + const roiTrigger$ = + this._container.configurationService.imageTiling$.pipe( + switchMap( + (active): Observable<[RenderCamera, ViewportSize]> => { + return active ? + observableCombineLatest( + this._container.renderService.renderCameraFrame$, + this._container.renderService.size$.pipe(debounceTime(250))) : + new Subject(); + }), map( ([camera, size]: [RenderCamera, ViewportSize]): PositionLookat => { return [ @@ -527,37 +537,43 @@ export class SliderComponent extends Component { subs.push(hasTexture$.subscribe(() => { /*noop*/ })); - const textureProviderPrev$ = this._navigator.stateService.currentState$.pipe( - filter(() => ViewerConfiguration.imageTiling), - filter( - (frame: AnimationFrame): boolean => { - return !!frame.state.previousImage; - }), - distinctUntilChanged( - undefined, - (frame: AnimationFrame): string => { - return frame.state.previousImage.id; - }), - withLatestFrom( - this._container.glRenderer.webGLRenderer$, - this._container.renderService.size$), - map( - ([frame, renderer, size]: [AnimationFrame, THREE.WebGLRenderer, ViewportSize]): TextureProvider => { - const state = frame.state; - const previousImage = state.previousImage; - const previousTransform = state.previousTransform; - - return new TextureProvider( - previousImage.id, - previousTransform.basicWidth, - previousTransform.basicHeight, - previousImage.image, - this._imageTileLoader, - new TileStore(), - renderer); - }), - publishReplay(1), - refCount()); + const textureProviderPrev$ = + this._container.configurationService.imageTiling$.pipe( + switchMap( + (active): Observable => { + return active ? + this._navigator.stateService.currentState$ : + new Subject(); + }), + filter( + (frame: AnimationFrame): boolean => { + return !!frame.state.previousImage; + }), + distinctUntilChanged( + undefined, + (frame: AnimationFrame): string => { + return frame.state.previousImage.id; + }), + withLatestFrom( + this._container.glRenderer.webGLRenderer$, + this._container.renderService.size$), + map( + ([frame, renderer, size]: [AnimationFrame, THREE.WebGLRenderer, ViewportSize]): TextureProvider => { + const state = frame.state; + const previousImage = state.previousImage; + const previousTransform = state.previousTransform; + + return new TextureProvider( + previousImage.id, + previousTransform.basicWidth, + previousTransform.basicHeight, + previousImage.image, + this._imageTileLoader, + new TileStore(), + renderer); + }), + publishReplay(1), + refCount()); subs.push(textureProviderPrev$.subscribe(() => { /*noop*/ })); @@ -580,10 +596,16 @@ export class SliderComponent extends Component { previous.abort(); })); - const roiTriggerPrev$ = observableCombineLatest( - this._container.renderService.renderCameraFrame$, - this._container.renderService.size$.pipe(debounceTime(250))).pipe( - filter(() => ViewerConfiguration.imageTiling), + const roiTriggerPrev$ = + this._container.configurationService.imageTiling$.pipe( + switchMap( + (active): Observable<[RenderCamera, ViewportSize]> => { + return active ? + observableCombineLatest( + this._container.renderService.renderCameraFrame$, + this._container.renderService.size$.pipe(debounceTime(250))) : + new Subject(); + }), map( ([camera, size]: [RenderCamera, ViewportSize]): PositionLookat => { return [ diff --git a/src/viewer/ConfigurationService.ts b/src/viewer/ConfigurationService.ts new file mode 100644 index 000000000..992fbc815 --- /dev/null +++ b/src/viewer/ConfigurationService.ts @@ -0,0 +1,29 @@ +import { + Observable, + of as observableOf, +} from "rxjs"; + +import { ViewerOptions } from "../external/viewer"; + +export class ConfigurationService { + private _imageTiling$: Observable; + private _exploreUrl$: Observable; + + constructor(options: ViewerOptions) { + const host = options?.url?.exploreHost ?? "www.mapillary.com"; + const scheme = options?.url?.scheme ?? "https"; + const exploreUrl = `${scheme}://${host}`; + this._exploreUrl$ = observableOf(exploreUrl); + + const imageTiling = options?.imageTiling === false ? false : true; + this._imageTiling$ = observableOf(imageTiling); + } + + public get exploreUrl$(): Observable { + return this._exploreUrl$; + } + + public get imageTiling$(): Observable { + return this._imageTiling$; + } +} diff --git a/src/viewer/Container.ts b/src/viewer/Container.ts index 854912464..98357f874 100644 --- a/src/viewer/Container.ts +++ b/src/viewer/Container.ts @@ -1,15 +1,16 @@ -import { KeyboardService } from "./KeyboardService"; -import { MouseService } from "./MouseService"; -import { SpriteService } from "./SpriteService"; -import { TouchService } from "./TouchService"; -import { ViewerOptions } from "./options/ViewerOptions"; - import { DOMRenderer } from "../render/DOMRenderer"; import { GLRenderer } from "../render/GLRenderer"; import { RenderService } from "../render/RenderService"; import { StateService } from "../state/StateService"; import { DOM } from "../util/DOM"; +import { ViewerOptions } from "./options/ViewerOptions"; +import { KeyboardService } from "./KeyboardService"; +import { MouseService } from "./MouseService"; +import { SpriteService } from "./SpriteService"; +import { TouchService } from "./TouchService"; +import { ConfigurationService } from "./ConfigurationService"; + export class Container { public id: string; @@ -24,6 +25,8 @@ export class Container { public spriteService: SpriteService; + public readonly configurationService: ConfigurationService; + private _canvasContainer: HTMLDivElement; private _canvas: HTMLCanvasElement; private _container: HTMLElement; @@ -87,6 +90,8 @@ export class Container { "mapillary-dom", this._container); + this.configurationService = new ConfigurationService(options); + this.renderService = new RenderService( this._container, @@ -166,7 +171,7 @@ export class Container { if (this._trackResize) { this.renderService.resize$.next(); } - } + }; private _removeNode(node: Node): void { if (node.parentNode) { diff --git a/src/viewer/PlayService.ts b/src/viewer/PlayService.ts index 5b31b7dfe..6cbbcd769 100644 --- a/src/viewer/PlayService.ts +++ b/src/viewer/PlayService.ts @@ -27,7 +27,6 @@ import { withLatestFrom, } from "rxjs/operators"; -import { GraphCalculator } from "../graph/GraphCalculator"; import { GraphMode } from "../graph/GraphMode"; import { GraphService } from "../graph/GraphService"; import { Image } from '../graph/Image'; diff --git a/src/viewer/Viewer.ts b/src/viewer/Viewer.ts index bddff03eb..154d83e16 100644 --- a/src/viewer/Viewer.ts +++ b/src/viewer/Viewer.ts @@ -19,7 +19,6 @@ import { RenderCamera } from "../render/RenderCamera"; import { RenderMode } from "../render/RenderMode"; import { TransitionMode } from "../state/TransitionMode"; import { EventEmitter } from "../util/EventEmitter"; -import { ViewerConfiguration } from "./ViewerConfiguration"; import { ICustomRenderer } from "./interfaces/ICustomRenderer"; import { PointOfView } from "./interfaces/PointOfView"; import { ViewerOptions } from "./options/ViewerOptions"; @@ -141,8 +140,6 @@ export class Viewer extends EventEmitter implements IViewer { constructor(options: ViewerOptions) { super(); - ViewerConfiguration.setOptions(options); - this._navigator = new Navigator(options); diff --git a/src/viewer/ViewerConfiguration.ts b/src/viewer/ViewerConfiguration.ts deleted file mode 100644 index d02601db8..000000000 --- a/src/viewer/ViewerConfiguration.ts +++ /dev/null @@ -1,40 +0,0 @@ -import { ViewerOptions } from "./options/ViewerOptions"; - -export class ViewerConfiguration { - private static _exploreHost: string = "www.mapillary.com"; - private static _scheme: string = "https"; - private static _imageTiling: boolean = true; - - public static get explore(): string { - const scheme = ViewerConfiguration._scheme; - const host = ViewerConfiguration._exploreHost; - return `${scheme}://${host}`; - } - - public static get imageTiling(): boolean { - return ViewerConfiguration._imageTiling; - } - - public static exploreImage(id: string): string { - return `${ViewerConfiguration.explore}/app/?pKey=${id}&focus=photo`; - } - - public static exploreUser(username: string): string { - return `${ViewerConfiguration.explore}/app/user/${username}`; - } - - public static setOptions(options: ViewerOptions): void { - if (!options) { return; } - if (options.imageTiling === false) { - ViewerConfiguration._imageTiling = false; - } - - if (!options.url) { return; } - if (!!options.url.exploreHost) { - ViewerConfiguration._exploreHost = options.url.exploreHost; - } - if (!!options.url.scheme) { - ViewerConfiguration._scheme = options.url.scheme; - } - } -} diff --git a/styles/cover.css b/styles/cover.css index 73bfd1012..301ecf0c3 100644 --- a/styles/cover.css +++ b/styles/cover.css @@ -30,7 +30,7 @@ top: 0; z-index: 0; background-color: rgba(0, 0, 0, 0.2); - transition: opacity .1s ease-in-out; + transition: opacity 0.1s ease-in-out; cursor: pointer; } @@ -49,7 +49,7 @@ margin-left: -64px; margin-top: -64px; position: absolute; - transition: opacity .2s ease-in-out; + transition: opacity 0.2s ease-in-out; z-index: 1; pointer-events: none; } @@ -64,7 +64,7 @@ .mapillary-cover-button-icon { background-size: contain; background-repeat: no-repeat; - background-image: url(''); + background-image: url(""); height: 100%; width: 100%; position: absolute; @@ -79,16 +79,15 @@ .mapillary-cover-logo { background-size: contain; background-repeat: no-repeat; - background-image: url(''); + background-image: url(""); bottom: 0px; height: 52px; left: 50%; margin-left: -73px; position: absolute; - transition: opacity .2s ease-in-out; + transition: opacity 0.2s ease-in-out; width: 157px; z-index: 1; - pointer-events: none; } .mapillary-cover-compact .mapillary-cover-logo { @@ -111,7 +110,7 @@ animation: rotate 1s linear infinite; background-repeat: no-repeat; background-size: 100%; - background-image: url(''); + background-image: url(""); height: 32px; left: 50%; margin-left: -16px; @@ -123,7 +122,7 @@ } @keyframes rotate { - 100% { - transform: rotate(360deg); - } + 100% { + transform: rotate(360deg); + } } diff --git a/test/helper/ConfigurationServiceMockCreator.ts b/test/helper/ConfigurationServiceMockCreator.ts new file mode 100644 index 000000000..619cd9e5e --- /dev/null +++ b/test/helper/ConfigurationServiceMockCreator.ts @@ -0,0 +1,26 @@ +import { Subject } from "rxjs"; + +import { MockCreator } from "./MockCreator"; +import { MockCreatorBase } from "./MockCreatorBase"; + +import { ConfigurationService } from "../../src/viewer/ConfigurationService"; + +export class ConfigurationServiceMockCreator extends + MockCreatorBase { + public create(): ConfigurationService { + const mock: ConfigurationService = + new MockCreator() + .create(ConfigurationService, "ConfigurationService"); + + this._mockProperty( + mock, + "exploreUrl$", + new Subject()); + this._mockProperty( + mock, + "imageTiling$", + new Subject()); + + return mock; + } +} diff --git a/test/helper/ContainerMockCreator.ts b/test/helper/ContainerMockCreator.ts index 9ecd9ab0d..6e90151c6 100644 --- a/test/helper/ContainerMockCreator.ts +++ b/test/helper/ContainerMockCreator.ts @@ -1,4 +1,6 @@ import { Container } from "../../src/viewer/Container"; +import { ConfigurationServiceMockCreator } + from "./ConfigurationServiceMockCreator"; import { DOMRendererMockCreator } from "./DOMRendererMockCreator"; import { GLRendererMockCreator } from "./GLRendererMockCreator"; import { KeyboardServiceMockCreator } from "./KeyboardServiceMockCreator"; @@ -14,6 +16,10 @@ export class ContainerMockCreator extends MockCreatorBase { const mock: Container = new MockCreator().create(Container, "Container"); this._mockProperty(mock, "canvasContainer", document.createElement("canvas")); + this._mockProperty( + mock, + "configurationService", + new ConfigurationServiceMockCreator().create()); this._mockProperty(mock, "domRenderer", new DOMRendererMockCreator().create()); this._mockProperty(mock, "container", document.createElement("div")); this._mockProperty(mock, "glRenderer", new GLRendererMockCreator().create()); diff --git a/test/util/ViewerConfiguration.test.ts b/test/util/ViewerConfiguration.test.ts deleted file mode 100644 index ff229100a..000000000 --- a/test/util/ViewerConfiguration.test.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { ViewerConfiguration } from "../../src/viewer/ViewerConfiguration"; -import { ViewerOptions } from "../../src/viewer/options/ViewerOptions"; - -describe("Urls.setOptions", () => { - it("should set all option properties", () => { - const options: ViewerOptions = { - container: "container-id", - imageTiling: false, - url: { - exploreHost: "test-explore", - scheme: "test-scheme", - }, - }; - - expect(ViewerConfiguration.imageTiling).toBe(true); - expect(ViewerConfiguration.explore).toBe("https://www.mapillary.com"); - - ViewerConfiguration.setOptions(options); - - expect(ViewerConfiguration.imageTiling).toBe(false); - expect(ViewerConfiguration.explore).not.toContain("https"); - expect(ViewerConfiguration.explore).not.toContain("mapillary"); - expect(ViewerConfiguration.explore).toContain(options.url.scheme); - expect(ViewerConfiguration.explore).toContain(options.url.exploreHost); - }); -}); diff --git a/test/viewer/ConfigurationService.test.ts b/test/viewer/ConfigurationService.test.ts new file mode 100644 index 000000000..349f07549 --- /dev/null +++ b/test/viewer/ConfigurationService.test.ts @@ -0,0 +1,89 @@ +import { ConfigurationService } from "../../src/viewer/ConfigurationService"; +import { ViewerOptions } from "../../src/viewer/options/ViewerOptions"; + +describe("ConfigurationService.ctor", () => { + it("should set all option properties", () => { + const options: ViewerOptions = { container: "container-id" }; + const service = new ConfigurationService(options); + + expect(service).toBeDefined(); + + }); +}); + +describe("ConfigurationService.exploreHost$", () => { + it("should emit default URL on each subscription", () => { + const options: ViewerOptions = { container: "container-id" }; + + const service = new ConfigurationService(options); + + let count = 0; + service.exploreUrl$.subscribe( + url => { + count++; + expect(url).toBe("https://www.mapillary.com"); + }); + service.exploreUrl$.subscribe( + url => { + count++; + expect(url).toBe("https://www.mapillary.com"); + }); + + expect(count).toBe(2); + }); + + it("should emit configured URL", (done: Function) => { + const options: ViewerOptions = { + container: "container-id", + url: { + exploreHost: "test-explore", + scheme: "test-scheme", + }, + }; + + const service = new ConfigurationService(options); + + service.exploreUrl$.subscribe( + url => { + expect(url).toBe("test-scheme://test-explore"); + done(); + }); + }); +}); + +describe("ConfigurationService.imageTiling$", () => { + it("should emit default value on each subscription", () => { + const options: ViewerOptions = { container: "container-id" }; + + const service = new ConfigurationService(options); + + let count = 0; + service.imageTiling$.subscribe( + active => { + count++; + expect(active).toBe(true); + }); + service.imageTiling$.subscribe( + active => { + count++; + expect(active).toBe(true); + }); + + expect(count).toBe(2); + }); + + it("should emit configured value", (done: Function) => { + const options: ViewerOptions = { + container: "container-id", + imageTiling: false, + }; + + const service = new ConfigurationService(options); + + service.imageTiling$.subscribe( + active => { + expect(active).toBe(false); + done(); + }); + }); +});