/** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import {CommonModule, Location, LocationStrategy, PathLocationStrategy, PlatformLocation} from '@angular/common'; import {MockLocationStrategy, MockPlatformLocation} from '@angular/common/testing'; import {TestBed} from '@angular/core/testing'; const baseUrl = '/base'; describe('Location Class', () => { describe('stripTrailingSlash', () => { it('should strip single character slash', () => { const input = '/'; expect(Location.stripTrailingSlash(input)).toBe(''); }); it('should normalize strip a trailing slash', () => { const input = baseUrl + '/'; expect(Location.stripTrailingSlash(input)).toBe(baseUrl); }); it('should ignore query params when stripping a slash', () => { const input = baseUrl + '/?param=1'; expect(Location.stripTrailingSlash(input)).toBe(baseUrl + '?param=1'); }); it('should not remove slashes inside query params', () => { const input = baseUrl + '?test/?=3'; expect(Location.stripTrailingSlash(input)).toBe(input); }); it('should not remove slashes after a pound sign', () => { const input = baseUrl + '#test/?=3'; expect(Location.stripTrailingSlash(input)).toBe(input); }); }); describe('location.getState()', () => { let location: Location; beforeEach(() => { TestBed.configureTestingModule({ imports: [CommonModule], providers: [ {provide: LocationStrategy, useClass: PathLocationStrategy}, { provide: PlatformLocation, useFactory: () => { return new MockPlatformLocation(); } }, {provide: Location, useClass: Location, deps: [LocationStrategy, PlatformLocation]}, ] }); location = TestBed.inject(Location); }); it('should get the state object', () => { expect(location.getState()).toBe(null); location.go('/test', '', {foo: 'bar'}); expect(location.getState()).toEqual({foo: 'bar'}); }); it('should work after using back button', () => { expect(location.getState()).toBe(null); location.go('/test1', '', {url: 'test1'}); location.go('/test2', '', {url: 'test2'}); expect(location.getState()).toEqual({url: 'test2'}); location.back(); expect(location.getState()).toEqual({url: 'test1'}); }); it('should work after using forward button', () => { expect(location.getState()).toBe(null); location.go('/test1', '', {url: 'test1'}); location.go('/test2', '', {url: 'test2'}); expect(location.getState()).toEqual({url: 'test2'}); location.back(); expect(location.getState()).toEqual({url: 'test1'}); location.forward(); expect(location.getState()).toEqual({url: 'test2'}); }); it('should work after using location.historyGo()', () => { expect(location.getState()).toBe(null); location.go('/test1', '', {url: 'test1'}); location.go('/test2', '', {url: 'test2'}); location.go('/test3', '', {url: 'test3'}); expect(location.getState()).toEqual({url: 'test3'}); location.historyGo(-2); expect(location.getState()).toEqual({url: 'test1'}); location.historyGo(2); expect(location.getState()).toEqual({url: 'test3'}); location.go('/test3', '', {url: 'test4'}); location.historyGo(0); expect(location.getState()).toEqual({url: 'test4'}); location.historyGo(); expect(location.getState()).toEqual({url: 'test4'}); // we are testing the behaviour of the `historyGo` method at the moment when the value of // the relativePosition goes out of bounds. // The result should be that the locationState does not change. location.historyGo(100); expect(location.getState()).toEqual({url: 'test4'}); location.historyGo(-100); expect(location.getState()).toEqual({url: 'test4'}); location.back(); expect(location.getState()).toEqual({url: 'test3'}); }); }); describe('location.onUrlChange()', () => { let location: Location; let locationStrategy: MockLocationStrategy; beforeEach(() => { TestBed.configureTestingModule({ imports: [CommonModule], providers: [ {provide: LocationStrategy, useClass: MockLocationStrategy}, { provide: PlatformLocation, useFactory: () => { return new MockPlatformLocation(); } }, {provide: Location, useClass: Location, deps: [LocationStrategy, PlatformLocation]}, ] }); location = TestBed.inject(Location); locationStrategy = TestBed.inject(LocationStrategy) as MockLocationStrategy; }); it('should have onUrlChange method', () => { expect(typeof location.onUrlChange).toBe('function'); }); it('should add registered functions to urlChangeListeners', () => { function changeListener(url: string, state: unknown) { return undefined; } expect((location as any)._urlChangeListeners.length).toBe(0); location.onUrlChange(changeListener); expect((location as any)._urlChangeListeners.length).toBe(1); expect((location as any)._urlChangeListeners[0]).toEqual(changeListener); }); it('should only notify listeners once when multiple listeners are registered', () => { let notificationCount = 0; function incrementChangeListener(url: string, state: unknown) { notificationCount += 1; return undefined; } function noopChangeListener(url: string, state: unknown) { return undefined; } location.onUrlChange(incrementChangeListener); location.onUrlChange(noopChangeListener); expect(notificationCount).toBe(0); locationStrategy.simulatePopState('/test'); expect(notificationCount).toBe(1); }); }); });