blob: d4165d36beecd08a2962933d79be58ee6507a0dc [file] [log] [blame]
/**
* @license Angular v8.1.1
* (c) 2010-2019 Google LLC. https://angular.io/
* License: MIT
*/
import { EventEmitter, Injectable, InjectionToken, Inject, Optional } from '@angular/core';
import { LocationStrategy } from '@angular/common';
import { Subject } from 'rxjs';
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
/**
* A spy for {\@link Location} that allows tests to fire simulated location events.
*
* \@publicApi
*/
class SpyLocation {
constructor() {
this.urlChanges = [];
this._history = [new LocationState('', '', null)];
this._historyIndex = 0;
/**
* \@internal
*/
this._subject = new EventEmitter();
/**
* \@internal
*/
this._baseHref = '';
/**
* \@internal
*/
this._platformStrategy = (/** @type {?} */ (null));
/**
* \@internal
*/
this._platformLocation = (/** @type {?} */ (null));
/**
* \@internal
*/
this._urlChangeListeners = [];
}
/**
* @param {?} url
* @return {?}
*/
setInitialPath(url) { this._history[this._historyIndex].path = url; }
/**
* @param {?} url
* @return {?}
*/
setBaseHref(url) { this._baseHref = url; }
/**
* @return {?}
*/
path() { return this._history[this._historyIndex].path; }
/**
* @return {?}
*/
getState() { return this._history[this._historyIndex].state; }
/**
* @param {?} path
* @param {?=} query
* @return {?}
*/
isCurrentPathEqualTo(path, query = '') {
/** @type {?} */
const givenPath = path.endsWith('/') ? path.substring(0, path.length - 1) : path;
/** @type {?} */
const currPath = this.path().endsWith('/') ? this.path().substring(0, this.path().length - 1) : this.path();
return currPath == givenPath + (query.length > 0 ? ('?' + query) : '');
}
/**
* @param {?} pathname
* @return {?}
*/
simulateUrlPop(pathname) {
this._subject.emit({ 'url': pathname, 'pop': true, 'type': 'popstate' });
}
/**
* @param {?} pathname
* @return {?}
*/
simulateHashChange(pathname) {
// Because we don't prevent the native event, the browser will independently update the path
this.setInitialPath(pathname);
this.urlChanges.push('hash: ' + pathname);
this._subject.emit({ 'url': pathname, 'pop': true, 'type': 'hashchange' });
}
/**
* @param {?} url
* @return {?}
*/
prepareExternalUrl(url) {
if (url.length > 0 && !url.startsWith('/')) {
url = '/' + url;
}
return this._baseHref + url;
}
/**
* @param {?} path
* @param {?=} query
* @param {?=} state
* @return {?}
*/
go(path, query = '', state = null) {
path = this.prepareExternalUrl(path);
if (this._historyIndex > 0) {
this._history.splice(this._historyIndex + 1);
}
this._history.push(new LocationState(path, query, state));
this._historyIndex = this._history.length - 1;
/** @type {?} */
const locationState = this._history[this._historyIndex - 1];
if (locationState.path == path && locationState.query == query) {
return;
}
/** @type {?} */
const url = path + (query.length > 0 ? ('?' + query) : '');
this.urlChanges.push(url);
this._subject.emit({ 'url': url, 'pop': false });
}
/**
* @param {?} path
* @param {?=} query
* @param {?=} state
* @return {?}
*/
replaceState(path, query = '', state = null) {
path = this.prepareExternalUrl(path);
/** @type {?} */
const history = this._history[this._historyIndex];
if (history.path == path && history.query == query) {
return;
}
history.path = path;
history.query = query;
history.state = state;
/** @type {?} */
const url = path + (query.length > 0 ? ('?' + query) : '');
this.urlChanges.push('replace: ' + url);
}
/**
* @return {?}
*/
forward() {
if (this._historyIndex < (this._history.length - 1)) {
this._historyIndex++;
this._subject.emit({ 'url': this.path(), 'state': this.getState(), 'pop': true });
}
}
/**
* @return {?}
*/
back() {
if (this._historyIndex > 0) {
this._historyIndex--;
this._subject.emit({ 'url': this.path(), 'state': this.getState(), 'pop': true });
}
}
/**
* @param {?} fn
* @return {?}
*/
onUrlChange(fn) {
this._urlChangeListeners.push(fn);
this.subscribe((/**
* @param {?} v
* @return {?}
*/
v => { this._notifyUrlChangeListeners(v.url, v.state); }));
}
/**
* \@internal
* @param {?=} url
* @param {?=} state
* @return {?}
*/
_notifyUrlChangeListeners(url = '', state) {
this._urlChangeListeners.forEach((/**
* @param {?} fn
* @return {?}
*/
fn => fn(url, state)));
}
/**
* @param {?} onNext
* @param {?=} onThrow
* @param {?=} onReturn
* @return {?}
*/
subscribe(onNext, onThrow, onReturn) {
return this._subject.subscribe({ next: onNext, error: onThrow, complete: onReturn });
}
/**
* @param {?} url
* @return {?}
*/
normalize(url) { return (/** @type {?} */ (null)); }
}
SpyLocation.decorators = [
{ type: Injectable }
];
class LocationState {
/**
* @param {?} path
* @param {?} query
* @param {?} state
*/
constructor(path, query, state) {
this.path = path;
this.query = query;
this.state = state;
}
}
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
/**
* A mock implementation of {\@link LocationStrategy} that allows tests to fire simulated
* location events.
*
* \@publicApi
*/
class MockLocationStrategy extends LocationStrategy {
constructor() {
super();
this.internalBaseHref = '/';
this.internalPath = '/';
this.internalTitle = '';
this.urlChanges = [];
/**
* \@internal
*/
this._subject = new EventEmitter();
this.stateChanges = [];
}
/**
* @param {?} url
* @return {?}
*/
simulatePopState(url) {
this.internalPath = url;
this._subject.emit(new _MockPopStateEvent(this.path()));
}
/**
* @param {?=} includeHash
* @return {?}
*/
path(includeHash = false) { return this.internalPath; }
/**
* @param {?} internal
* @return {?}
*/
prepareExternalUrl(internal) {
if (internal.startsWith('/') && this.internalBaseHref.endsWith('/')) {
return this.internalBaseHref + internal.substring(1);
}
return this.internalBaseHref + internal;
}
/**
* @param {?} ctx
* @param {?} title
* @param {?} path
* @param {?} query
* @return {?}
*/
pushState(ctx, title, path, query) {
// Add state change to changes array
this.stateChanges.push(ctx);
this.internalTitle = title;
/** @type {?} */
const url = path + (query.length > 0 ? ('?' + query) : '');
this.internalPath = url;
/** @type {?} */
const externalUrl = this.prepareExternalUrl(url);
this.urlChanges.push(externalUrl);
}
/**
* @param {?} ctx
* @param {?} title
* @param {?} path
* @param {?} query
* @return {?}
*/
replaceState(ctx, title, path, query) {
// Reset the last index of stateChanges to the ctx (state) object
this.stateChanges[(this.stateChanges.length || 1) - 1] = ctx;
this.internalTitle = title;
/** @type {?} */
const url = path + (query.length > 0 ? ('?' + query) : '');
this.internalPath = url;
/** @type {?} */
const externalUrl = this.prepareExternalUrl(url);
this.urlChanges.push('replace: ' + externalUrl);
}
/**
* @param {?} fn
* @return {?}
*/
onPopState(fn) { this._subject.subscribe({ next: fn }); }
/**
* @return {?}
*/
getBaseHref() { return this.internalBaseHref; }
/**
* @return {?}
*/
back() {
if (this.urlChanges.length > 0) {
this.urlChanges.pop();
this.stateChanges.pop();
/** @type {?} */
const nextUrl = this.urlChanges.length > 0 ? this.urlChanges[this.urlChanges.length - 1] : '';
this.simulatePopState(nextUrl);
}
}
/**
* @return {?}
*/
forward() { throw 'not implemented'; }
/**
* @return {?}
*/
getState() { return this.stateChanges[(this.stateChanges.length || 1) - 1]; }
}
MockLocationStrategy.decorators = [
{ type: Injectable }
];
/** @nocollapse */
MockLocationStrategy.ctorParameters = () => [];
class _MockPopStateEvent {
/**
* @param {?} newUrl
*/
constructor(newUrl) {
this.newUrl = newUrl;
this.pop = true;
this.type = 'popstate';
}
}
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
/**
* Parser from https://tools.ietf.org/html/rfc3986#appendix-B
* ^(([^:/?#]+):)?(//([^/?#]*))?([^?#]*)(\?([^#]*))?(#(.*))?
* 12 3 4 5 6 7 8 9
*
* Example: http://www.ics.uci.edu/pub/ietf/uri/#Related
*
* Results in:
*
* $1 = http:
* $2 = http
* $3 = //www.ics.uci.edu
* $4 = www.ics.uci.edu
* $5 = /pub/ietf/uri/
* $6 = <undefined>
* $7 = <undefined>
* $8 = #Related
* $9 = Related
* @type {?}
*/
const urlParse = /^(([^:\/?#]+):)?(\/\/([^\/?#]*))?([^?#]*)(\?([^#]*))?(#(.*))?/;
/**
* @param {?} urlStr
* @param {?} baseHref
* @return {?}
*/
function parseUrl(urlStr, baseHref) {
/** @type {?} */
const verifyProtocol = /^((http[s]?|ftp):\/\/)/;
/** @type {?} */
let serverBase;
// URL class requires full URL. If the URL string doesn't start with protocol, we need to add
// an arbitrary base URL which can be removed afterward.
if (!verifyProtocol.test(urlStr)) {
serverBase = 'http://empty.com/';
}
/** @type {?} */
let parsedUrl;
try {
parsedUrl = new URL(urlStr, serverBase);
}
catch (e) {
/** @type {?} */
const result = urlParse.exec(serverBase || '' + urlStr);
if (!result) {
throw new Error(`Invalid URL: ${urlStr} with base: ${baseHref}`);
}
/** @type {?} */
const hostSplit = result[4].split(':');
parsedUrl = {
protocol: result[1],
hostname: hostSplit[0],
port: hostSplit[1] || '',
pathname: result[5],
search: result[6],
hash: result[8],
};
}
if (parsedUrl.pathname && parsedUrl.pathname.indexOf(baseHref) === 0) {
parsedUrl.pathname = parsedUrl.pathname.substring(baseHref.length);
}
return {
hostname: !serverBase && parsedUrl.hostname || '',
protocol: !serverBase && parsedUrl.protocol || '',
port: !serverBase && parsedUrl.port || '',
pathname: parsedUrl.pathname || '/',
search: parsedUrl.search || '',
hash: parsedUrl.hash || '',
};
}
/**
* Provider for mock platform location config
*
* \@publicApi
* @type {?}
*/
const MOCK_PLATFORM_LOCATION_CONFIG = new InjectionToken('MOCK_PLATFORM_LOCATION_CONFIG');
/**
* Mock implementation of URL state.
*
* \@publicApi
*/
class MockPlatformLocation {
/**
* @param {?=} config
*/
constructor(config) {
this.baseHref = '';
this.hashUpdate = new Subject();
this.urlChanges = [{ hostname: '', protocol: '', port: '', pathname: '/', search: '', hash: '', state: null }];
if (config) {
this.baseHref = config.appBaseHref || '';
/** @type {?} */
const parsedChanges = this.parseChanges(null, config.startUrl || 'http://<empty>/', this.baseHref);
this.urlChanges[0] = Object.assign({}, parsedChanges);
}
}
/**
* @return {?}
*/
get hostname() { return this.urlChanges[0].hostname; }
/**
* @return {?}
*/
get protocol() { return this.urlChanges[0].protocol; }
/**
* @return {?}
*/
get port() { return this.urlChanges[0].port; }
/**
* @return {?}
*/
get pathname() { return this.urlChanges[0].pathname; }
/**
* @return {?}
*/
get search() { return this.urlChanges[0].search; }
/**
* @return {?}
*/
get hash() { return this.urlChanges[0].hash; }
/**
* @return {?}
*/
get state() { return this.urlChanges[0].state; }
/**
* @return {?}
*/
getBaseHrefFromDOM() { return this.baseHref; }
/**
* @param {?} fn
* @return {?}
*/
onPopState(fn) {
// No-op: a state stack is not implemented, so
// no events will ever come.
}
/**
* @param {?} fn
* @return {?}
*/
onHashChange(fn) { this.hashUpdate.subscribe(fn); }
/**
* @return {?}
*/
get href() {
/** @type {?} */
let url = `${this.protocol}//${this.hostname}${this.port ? ':' + this.port : ''}`;
url += `${this.pathname === '/' ? '' : this.pathname}${this.search}${this.hash}`;
return url;
}
/**
* @return {?}
*/
get url() { return `${this.pathname}${this.search}${this.hash}`; }
/**
* @private
* @param {?} state
* @param {?} url
* @param {?=} baseHref
* @return {?}
*/
parseChanges(state, url, baseHref = '') {
// When the `history.state` value is stored, it is always copied.
state = JSON.parse(JSON.stringify(state));
return Object.assign({}, parseUrl(url, baseHref), { state });
}
/**
* @param {?} state
* @param {?} title
* @param {?} newUrl
* @return {?}
*/
replaceState(state, title, newUrl) {
const { pathname, search, state: parsedState, hash } = this.parseChanges(state, newUrl);
this.urlChanges[0] = Object.assign({}, this.urlChanges[0], { pathname, search, hash, state: parsedState });
}
/**
* @param {?} state
* @param {?} title
* @param {?} newUrl
* @return {?}
*/
pushState(state, title, newUrl) {
const { pathname, search, state: parsedState, hash } = this.parseChanges(state, newUrl);
this.urlChanges.unshift(Object.assign({}, this.urlChanges[0], { pathname, search, hash, state: parsedState }));
}
/**
* @return {?}
*/
forward() { throw new Error('Not implemented'); }
/**
* @return {?}
*/
back() {
/** @type {?} */
const oldUrl = this.url;
/** @type {?} */
const oldHash = this.hash;
this.urlChanges.shift();
/** @type {?} */
const newHash = this.hash;
if (oldHash !== newHash) {
scheduleMicroTask((/**
* @return {?}
*/
() => this.hashUpdate.next((/** @type {?} */ ({
type: 'hashchange', state: null, oldUrl, newUrl: this.url
})))));
}
}
/**
* @return {?}
*/
getState() { return this.state; }
}
MockPlatformLocation.decorators = [
{ type: Injectable }
];
/** @nocollapse */
MockPlatformLocation.ctorParameters = () => [
{ type: undefined, decorators: [{ type: Inject, args: [MOCK_PLATFORM_LOCATION_CONFIG,] }, { type: Optional }] }
];
/**
* @param {?} cb
* @return {?}
*/
function scheduleMicroTask(cb) {
Promise.resolve(null).then(cb);
}
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
/**
* Generated bundle index. Do not edit.
*/
export { SpyLocation, MockLocationStrategy, MOCK_PLATFORM_LOCATION_CONFIG, MockPlatformLocation };
//# sourceMappingURL=testing.js.map