single-spa-angular
Advanced tools
Comparing version 4.3.2 to 4.4.1
(function (global, factory) { | ||
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) : | ||
typeof define === 'function' && define.amd ? define('single-spa-angular/internals', ['exports'], factory) : | ||
(global = global || self, factory((global['single-spa-angular'] = global['single-spa-angular'] || {}, global['single-spa-angular'].internals = {}))); | ||
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory((global['single-spa-angular'] = global['single-spa-angular'] || {}, global['single-spa-angular'].internals = {}))); | ||
}(this, (function (exports) { 'use strict'; | ||
function removeApplicationFromDOMIfIvyEnabled(opts, props) { | ||
function removeApplicationFromDOMIfIvyEnabled(options, props) { | ||
if (ivyEnabled()) { | ||
var domElementGetter = chooseDomElementGetter(opts, props); | ||
var domElement = getContainerEl(domElementGetter); | ||
var domElementGetter = chooseDomElementGetter(options, props); | ||
var domElement = getContainerElement(domElementGetter); | ||
// View Engine removes all nodes automatically when calling `NgModuleRef.destroy()`, | ||
@@ -18,3 +18,12 @@ // which calls `ComponentRef.destroy()`. | ||
} | ||
function getContainerEl(domElementGetter) { | ||
function getContainerElementAndSetTemplate(options, props) { | ||
var domElementGetter = chooseDomElementGetter(options, props); | ||
if (!domElementGetter) { | ||
throw Error("Cannot mount angular application '" + (props.name || props.appName) + "' without a domElementGetter provided either as an opt or a prop"); | ||
} | ||
var containerElement = getContainerElement(domElementGetter); | ||
containerElement.innerHTML = options.template; | ||
return containerElement; | ||
} | ||
function getContainerElement(domElementGetter) { | ||
var element = domElementGetter(); | ||
@@ -69,4 +78,7 @@ if (!element) { | ||
exports.chooseDomElementGetter = chooseDomElementGetter; | ||
exports.getContainerEl = getContainerEl; | ||
/** | ||
* Generated bundle index. Do not edit. | ||
*/ | ||
exports.getContainerElementAndSetTemplate = getContainerElementAndSetTemplate; | ||
exports.removeApplicationFromDOMIfIvyEnabled = removeApplicationFromDOMIfIvyEnabled; | ||
@@ -73,0 +85,0 @@ |
@@ -1,2 +0,2 @@ | ||
!function(e,n){"object"==typeof exports&&"undefined"!=typeof module?n(exports):"function"==typeof define&&define.amd?define("single-spa-angular/internals",["exports"],n):n(((e=e||self)["single-spa-angular"]=e["single-spa-angular"]||{},e["single-spa-angular"].internals={}))}(this,(function(e){"use strict";function n(e){var n=e();if(!n)throw Error("domElementGetter did not return a valid dom element");return n}function t(e,n){var t,r;return(n=null!==(t=null==n?void 0:n.customProps)&&void 0!==t?t:n).domElement?function(){return n.domElement}:n.domElementGetter?n.domElementGetter:e.domElementGetter?e.domElementGetter:(r=n.name,function(){var e="single-spa-application:"+r,n=document.getElementById(e);return n||((n=document.createElement("div")).id=e,document.body.appendChild(n)),n})}e.chooseDomElementGetter=t,e.getContainerEl=n,e.removeApplicationFromDOMIfIvyEnabled=function(e,r){if(function(){try{return!!require("@angular/core").ɵivyEnabled}catch(e){return!1}}())for(var o=n(t(e,r));o.firstChild;)o.removeChild(o.firstChild)},Object.defineProperty(e,"__esModule",{value:!0})})); | ||
!function(e,n){"object"==typeof exports&&"undefined"!=typeof module?n(exports):"function"==typeof define&&define.amd?define("single-spa-angular/internals",["exports"],n):n(((e="undefined"!=typeof globalThis?globalThis:e||self)["single-spa-angular"]=e["single-spa-angular"]||{},e["single-spa-angular"].internals={}))}(this,(function(e){"use strict";function n(e){var n=e();if(!n)throw Error("domElementGetter did not return a valid dom element");return n}function t(e,n){var t,r;return(n=null!==(t=null==n?void 0:n.customProps)&&void 0!==t?t:n).domElement?function(){return n.domElement}:n.domElementGetter?n.domElementGetter:e.domElementGetter?e.domElementGetter:(r=n.name,function(){var e="single-spa-application:"+r,n=document.getElementById(e);return n||((n=document.createElement("div")).id=e,document.body.appendChild(n)),n})}e.getContainerElementAndSetTemplate=function(e,r){var o=t(e,r);if(!o)throw Error("Cannot mount angular application '"+(r.name||r.appName)+"' without a domElementGetter provided either as an opt or a prop");var i=n(o);return i.innerHTML=e.template,i},e.removeApplicationFromDOMIfIvyEnabled=function(e,r){if(function(){try{return!!require("@angular/core").ɵivyEnabled}catch(e){return!1}}())for(var o=n(t(e,r));o.firstChild;)o.removeChild(o.firstChild)},Object.defineProperty(e,"__esModule",{value:!0})})); | ||
//# sourceMappingURL=single-spa-angular-internals.umd.min.js.map |
(function (global, factory) { | ||
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('single-spa-angular/internals'), require('@angular/core'), require('@angular/common')) : | ||
typeof define === 'function' && define.amd ? define('single-spa-angular', ['exports', 'single-spa-angular/internals', '@angular/core', '@angular/common'], factory) : | ||
(global = global || self, factory(global['single-spa-angular'] = {}, global['single-spa-angular'].internals, global.ng.core, global.ng.common)); | ||
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global['single-spa-angular'] = {}, global['single-spa-angular'].internals, global.ng.core, global.ng.common)); | ||
}(this, (function (exports, internals, core, common) { 'use strict'; | ||
/*! ***************************************************************************** | ||
Copyright (c) Microsoft Corporation. All rights reserved. | ||
Licensed under the Apache License, Version 2.0 (the "License"); you may not use | ||
this file except in compliance with the License. You may obtain a copy of the | ||
License at http://www.apache.org/licenses/LICENSE-2.0 | ||
Copyright (c) Microsoft Corporation. | ||
THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | ||
KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED | ||
WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE, | ||
MERCHANTABLITY OR NON-INFRINGEMENT. | ||
Permission to use, copy, modify, and/or distribute this software for any | ||
purpose with or without fee is hereby granted. | ||
See the Apache Version 2.0 License for specific language governing permissions | ||
and limitations under the License. | ||
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH | ||
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY | ||
AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, | ||
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM | ||
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR | ||
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR | ||
PERFORMANCE OF THIS SOFTWARE. | ||
***************************************************************************** */ | ||
/* global Reflect, Promise */ | ||
var extendStatics = function(d, b) { | ||
var extendStatics = function (d, b) { | ||
extendStatics = Object.setPrototypeOf || | ||
({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || | ||
function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; | ||
function (d, b) { for (var p in b) | ||
if (Object.prototype.hasOwnProperty.call(b, p)) | ||
d[p] = b[p]; }; | ||
return extendStatics(d, b); | ||
}; | ||
function __extends(d, b) { | ||
@@ -35,8 +35,9 @@ extendStatics(d, b); | ||
} | ||
var __assign = function() { | ||
var __assign = function () { | ||
__assign = Object.assign || function __assign(t) { | ||
for (var s, i = 1, n = arguments.length; i < n; i++) { | ||
s = arguments[i]; | ||
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; | ||
for (var p in s) | ||
if (Object.prototype.hasOwnProperty.call(s, p)) | ||
t[p] = s[p]; | ||
} | ||
@@ -47,7 +48,7 @@ return t; | ||
}; | ||
function __rest(s, e) { | ||
var t = {}; | ||
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0) | ||
t[p] = s[p]; | ||
for (var p in s) | ||
if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0) | ||
t[p] = s[p]; | ||
if (s != null && typeof Object.getOwnPropertySymbols === "function") | ||
@@ -60,23 +61,34 @@ for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) { | ||
} | ||
function __decorate(decorators, target, key, desc) { | ||
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; | ||
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); | ||
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; | ||
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") | ||
r = Reflect.decorate(decorators, target, key, desc); | ||
else | ||
for (var i = decorators.length - 1; i >= 0; i--) | ||
if (d = decorators[i]) | ||
r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; | ||
return c > 3 && r && Object.defineProperty(target, key, r), r; | ||
} | ||
function __param(paramIndex, decorator) { | ||
return function (target, key) { decorator(target, key, paramIndex); } | ||
return function (target, key) { decorator(target, key, paramIndex); }; | ||
} | ||
function __metadata(metadataKey, metadataValue) { | ||
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(metadataKey, metadataValue); | ||
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") | ||
return Reflect.metadata(metadataKey, metadataValue); | ||
} | ||
function __awaiter(thisArg, _arguments, P, generator) { | ||
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } | ||
return new (P || (P = Promise))(function (resolve, reject) { | ||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } | ||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } | ||
function fulfilled(value) { try { | ||
step(generator.next(value)); | ||
} | ||
catch (e) { | ||
reject(e); | ||
} } | ||
function rejected(value) { try { | ||
step(generator["throw"](value)); | ||
} | ||
catch (e) { | ||
reject(e); | ||
} } | ||
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } | ||
@@ -86,64 +98,123 @@ step((generator = generator.apply(thisArg, _arguments || [])).next()); | ||
} | ||
function __generator(thisArg, body) { | ||
var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; | ||
return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; | ||
var _ = { label: 0, sent: function () { if (t[0] & 1) | ||
throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; | ||
return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function () { return this; }), g; | ||
function verb(n) { return function (v) { return step([n, v]); }; } | ||
function step(op) { | ||
if (f) throw new TypeError("Generator is already executing."); | ||
while (_) try { | ||
if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t; | ||
if (y = 0, t) op = [op[0] & 2, t.value]; | ||
switch (op[0]) { | ||
case 0: case 1: t = op; break; | ||
case 4: _.label++; return { value: op[1], done: false }; | ||
case 5: _.label++; y = op[1]; op = [0]; continue; | ||
case 7: op = _.ops.pop(); _.trys.pop(); continue; | ||
default: | ||
if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } | ||
if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } | ||
if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } | ||
if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } | ||
if (t[2]) _.ops.pop(); | ||
_.trys.pop(); continue; | ||
if (f) | ||
throw new TypeError("Generator is already executing."); | ||
while (_) | ||
try { | ||
if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) | ||
return t; | ||
if (y = 0, t) | ||
op = [op[0] & 2, t.value]; | ||
switch (op[0]) { | ||
case 0: | ||
case 1: | ||
t = op; | ||
break; | ||
case 4: | ||
_.label++; | ||
return { value: op[1], done: false }; | ||
case 5: | ||
_.label++; | ||
y = op[1]; | ||
op = [0]; | ||
continue; | ||
case 7: | ||
op = _.ops.pop(); | ||
_.trys.pop(); | ||
continue; | ||
default: | ||
if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { | ||
_ = 0; | ||
continue; | ||
} | ||
if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { | ||
_.label = op[1]; | ||
break; | ||
} | ||
if (op[0] === 6 && _.label < t[1]) { | ||
_.label = t[1]; | ||
t = op; | ||
break; | ||
} | ||
if (t && _.label < t[2]) { | ||
_.label = t[2]; | ||
_.ops.push(op); | ||
break; | ||
} | ||
if (t[2]) | ||
_.ops.pop(); | ||
_.trys.pop(); | ||
continue; | ||
} | ||
op = body.call(thisArg, _); | ||
} | ||
op = body.call(thisArg, _); | ||
} catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } | ||
if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; | ||
catch (e) { | ||
op = [6, e]; | ||
y = 0; | ||
} | ||
finally { | ||
f = t = 0; | ||
} | ||
if (op[0] & 5) | ||
throw op[1]; | ||
return { value: op[0] ? op[1] : void 0, done: true }; | ||
} | ||
} | ||
function __exportStar(m, exports) { | ||
for (var p in m) if (!exports.hasOwnProperty(p)) exports[p] = m[p]; | ||
var __createBinding = Object.create ? (function (o, m, k, k2) { | ||
if (k2 === undefined) | ||
k2 = k; | ||
Object.defineProperty(o, k2, { enumerable: true, get: function () { return m[k]; } }); | ||
}) : (function (o, m, k, k2) { | ||
if (k2 === undefined) | ||
k2 = k; | ||
o[k2] = m[k]; | ||
}); | ||
function __exportStar(m, o) { | ||
for (var p in m) | ||
if (p !== "default" && !Object.prototype.hasOwnProperty.call(o, p)) | ||
__createBinding(o, m, p); | ||
} | ||
function __values(o) { | ||
var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0; | ||
if (m) return m.call(o); | ||
if (o && typeof o.length === "number") return { | ||
next: function () { | ||
if (o && i >= o.length) o = void 0; | ||
return { value: o && o[i++], done: !o }; | ||
} | ||
}; | ||
if (m) | ||
return m.call(o); | ||
if (o && typeof o.length === "number") | ||
return { | ||
next: function () { | ||
if (o && i >= o.length) | ||
o = void 0; | ||
return { value: o && o[i++], done: !o }; | ||
} | ||
}; | ||
throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined."); | ||
} | ||
function __read(o, n) { | ||
var m = typeof Symbol === "function" && o[Symbol.iterator]; | ||
if (!m) return o; | ||
if (!m) | ||
return o; | ||
var i = m.call(o), r, ar = [], e; | ||
try { | ||
while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value); | ||
while ((n === void 0 || n-- > 0) && !(r = i.next()).done) | ||
ar.push(r.value); | ||
} | ||
catch (error) { e = { error: error }; } | ||
catch (error) { | ||
e = { error: error }; | ||
} | ||
finally { | ||
try { | ||
if (r && !r.done && (m = i["return"])) m.call(i); | ||
if (r && !r.done && (m = i["return"])) | ||
m.call(i); | ||
} | ||
finally { if (e) throw e.error; } | ||
finally { | ||
if (e) | ||
throw e.error; | ||
} | ||
} | ||
return ar; | ||
} | ||
function __spread() { | ||
@@ -154,5 +225,5 @@ for (var ar = [], i = 0; i < arguments.length; i++) | ||
} | ||
function __spreadArrays() { | ||
for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length; | ||
for (var s = 0, i = 0, il = arguments.length; i < il; i++) | ||
s += arguments[i].length; | ||
for (var r = Array(s), k = 0, i = 0; i < il; i++) | ||
@@ -162,20 +233,26 @@ for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++) | ||
return r; | ||
}; | ||
} | ||
; | ||
function __await(v) { | ||
return this instanceof __await ? (this.v = v, this) : new __await(v); | ||
} | ||
function __asyncGenerator(thisArg, _arguments, generator) { | ||
if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined."); | ||
if (!Symbol.asyncIterator) | ||
throw new TypeError("Symbol.asyncIterator is not defined."); | ||
var g = generator.apply(thisArg, _arguments || []), i, q = []; | ||
return i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i; | ||
function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; } | ||
function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } } | ||
function verb(n) { if (g[n]) | ||
i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; } | ||
function resume(n, v) { try { | ||
step(g[n](v)); | ||
} | ||
catch (e) { | ||
settle(q[0][3], e); | ||
} } | ||
function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); } | ||
function fulfill(value) { resume("next", value); } | ||
function reject(value) { resume("throw", value); } | ||
function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); } | ||
function settle(f, v) { if (f(v), q.shift(), q.length) | ||
resume(q[0][0], q[0][1]); } | ||
} | ||
function __asyncDelegator(o) { | ||
@@ -186,28 +263,39 @@ var i, p; | ||
} | ||
function __asyncValues(o) { | ||
if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined."); | ||
if (!Symbol.asyncIterator) | ||
throw new TypeError("Symbol.asyncIterator is not defined."); | ||
var m = o[Symbol.asyncIterator], i; | ||
return m ? m.call(o) : (o = typeof __values === "function" ? __values(o) : o[Symbol.iterator](), i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i); | ||
function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; } | ||
function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); } | ||
function settle(resolve, reject, d, v) { Promise.resolve(v).then(function (v) { resolve({ value: v, done: d }); }, reject); } | ||
} | ||
function __makeTemplateObject(cooked, raw) { | ||
if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; } | ||
if (Object.defineProperty) { | ||
Object.defineProperty(cooked, "raw", { value: raw }); | ||
} | ||
else { | ||
cooked.raw = raw; | ||
} | ||
return cooked; | ||
} | ||
; | ||
var __setModuleDefault = Object.create ? (function (o, v) { | ||
Object.defineProperty(o, "default", { enumerable: true, value: v }); | ||
}) : function (o, v) { | ||
o["default"] = v; | ||
}; | ||
function __importStar(mod) { | ||
if (mod && mod.__esModule) return mod; | ||
if (mod && mod.__esModule) | ||
return mod; | ||
var result = {}; | ||
if (mod != null) for (var k in mod) if (Object.hasOwnProperty.call(mod, k)) result[k] = mod[k]; | ||
result.default = mod; | ||
if (mod != null) | ||
for (var k in mod) | ||
if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) | ||
__createBinding(result, mod, k); | ||
__setModuleDefault(result, mod); | ||
return result; | ||
} | ||
function __importDefault(mod) { | ||
return (mod && mod.__esModule) ? mod : { default: mod }; | ||
} | ||
function __classPrivateFieldGet(receiver, privateMap) { | ||
@@ -219,3 +307,2 @@ if (!privateMap.has(receiver)) { | ||
} | ||
function __classPrivateFieldSet(receiver, privateMap, value) { | ||
@@ -232,10 +319,27 @@ if (!privateMap.has(receiver)) { | ||
function SingleSpaPlatformLocation() { | ||
var _this = _super !== null && _super.apply(this, arguments) || this; | ||
var _this = _super.apply(this, __spread(arguments)) || this; | ||
// This is a simple marker that helps us to ignore PopStateEvents | ||
// that was not dispatched by the browser. | ||
_this.skipNextPopState = false; | ||
_this.onPopStateListeners = []; | ||
// The key here is an actual forked `Zone` of some specific application. | ||
// We will be able to find the specific zone when application gets destroyed | ||
// by application `name`. | ||
// The reason of that the `onPopState` method is invoked during `bootstrapModule` | ||
// and we can't know what application has invoked it. Why should we know the application | ||
// that has invoked `onPopState`? When application gets destroyed in a `sharing dependencies mode` | ||
// (when there is a single platform per all applications) we want to remove application | ||
// specific `popstate` listeners. E.g. if there are 2 applications: | ||
// * shop application adds `popstate` listener | ||
// * navbar application adds `popstate` listener | ||
// When shop application gets destroyed we want to remove only its `popstate` listener. | ||
_this.zoneToOnPopStateListenersMap = new Map(); | ||
// This is used only to make `Zone.wrap` happy, since it requires 2 arguments | ||
// and the second argument is a unique string which `zone.js` uses for debugging purposes. | ||
// We might want to use the application name, but we're not able to get it when `onPopState` | ||
// method is called during module bootstrapping. | ||
_this.source = 0; | ||
return _this; | ||
} | ||
SingleSpaPlatformLocation.prototype.destroy = function () { | ||
SingleSpaPlatformLocation.prototype.destroyApplication = function (zoneIdentifier) { | ||
var e_1, _a; | ||
// TLDR: Angular adds `popstate` event listener and then doesn't remove it when application gets destroyed. | ||
@@ -245,21 +349,25 @@ // Basically, Angular has a potentional memory leak. The `ɵBrowserPlatformLocation` | ||
// https://github.com/angular/angular/blob/14be55c9facf3e47b8c97df4502dc3f0f897da03/packages/common/src/location/platform_location.ts#L126 | ||
var e_1, _a; | ||
try { | ||
for (var _b = __values(this.onPopStateListeners), _c = _b.next(); !_c.done; _c = _b.next()) { | ||
var onPopStateListener = _c.value; | ||
window.removeEventListener('popstate', onPopStateListener); | ||
} | ||
} | ||
catch (e_1_1) { e_1 = { error: e_1_1 }; } | ||
finally { | ||
var zone = __spread(this.zoneToOnPopStateListenersMap.keys()).find( | ||
// `getZoneWith` will return a zone which defines a `key` and in our case | ||
// we define a custom key in `single-spa-angular.ts` | ||
// via this line of code: | ||
// `_properties[zoneIdentifier] = true;` | ||
function (zone) { return zone.getZoneWith(zoneIdentifier) !== null; }); | ||
var onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone); | ||
if (Array.isArray(onPopStateListeners)) { | ||
try { | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
for (var onPopStateListeners_1 = __values(onPopStateListeners), onPopStateListeners_1_1 = onPopStateListeners_1.next(); !onPopStateListeners_1_1.done; onPopStateListeners_1_1 = onPopStateListeners_1.next()) { | ||
var onPopStateListener = onPopStateListeners_1_1.value; | ||
window.removeEventListener('popstate', onPopStateListener); | ||
} | ||
} | ||
finally { if (e_1) throw e_1.error; } | ||
catch (e_1_1) { e_1 = { error: e_1_1 }; } | ||
finally { | ||
try { | ||
if (onPopStateListeners_1_1 && !onPopStateListeners_1_1.done && (_a = onPopStateListeners_1.return)) _a.call(onPopStateListeners_1); | ||
} | ||
finally { if (e_1) throw e_1.error; } | ||
} | ||
} | ||
// We do this because the `SingleSpaPlatformLocation` is a part of PLATFORM_INJECTOR, | ||
// which means it's created only once and will not be garbage collected, since the PLATFORM_INJECTOR | ||
// will keep reference to its instance. | ||
// TODO: https://github.com/single-spa/single-spa-angular/issues/170 | ||
this.onPopStateListeners = []; | ||
this.zoneToOnPopStateListenersMap.delete(zone); | ||
}; | ||
@@ -276,2 +384,12 @@ SingleSpaPlatformLocation.prototype.pushState = function (state, title, url) { | ||
var _this = this; | ||
// `Zone.current` will reference the zone that serves as an execution context | ||
// to some specific application, especially when `onPopState` is called. | ||
var zone = Zone.current; | ||
// Wrap any event listener into zone that is specific to some application. | ||
// The main issue is `back/forward` buttons of browsers, because they invoke | ||
// `history.back|forward` which dispatch `popstate` event. Since `single-spa` | ||
// overrides `history.replaceState` Angular's zone cannot intercept this event. | ||
// Only the root zone is able to intercept all events. | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
fn = zone.wrap(fn, "" + this.source++); | ||
var onPopStateListener = function (event) { | ||
@@ -287,24 +405,22 @@ // The `LocationChangeEvent` doesn't have the `singleSpa` property, since it's added | ||
else { | ||
// Wrap any event listener into zone that is specific to some application. | ||
// The main issue is `back/forward` buttons of browsers, because they invoke | ||
// `history.back|forward` which dispatch `popstate` event. Since `single-spa` | ||
// overrides `history.replaceState` Angular's zone cannot intercept this event. | ||
// Only the root zone is able to intercept all events. | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
_this.ngZone.run(function () { return fn(event); }); | ||
fn(event); | ||
} | ||
}; | ||
this.storeOnPopStateListener(zone, onPopStateListener); | ||
_super.prototype.onPopState.call(this, onPopStateListener); | ||
}; | ||
SingleSpaPlatformLocation.prototype.storeOnPopStateListener = function (zone, onPopStateListener) { | ||
// All listeners should be stored inside an array because the `onPopState` can be called | ||
// multiple times thus we wanna reference all listeners to remove them further. | ||
this.onPopStateListeners.push(onPopStateListener); | ||
_super.prototype.onPopState.call(this, onPopStateListener); | ||
var onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone) || []; | ||
onPopStateListeners.push(onPopStateListener); | ||
if (!this.zoneToOnPopStateListenersMap.has(zone)) { | ||
this.zoneToOnPopStateListenersMap.set(zone, onPopStateListeners); | ||
} | ||
}; | ||
SingleSpaPlatformLocation.prototype.setNgZone = function (ngZone) { | ||
this.ngZone = ngZone; | ||
}; | ||
SingleSpaPlatformLocation = __decorate([ | ||
core.Injectable() | ||
], SingleSpaPlatformLocation); | ||
return SingleSpaPlatformLocation; | ||
}(common["ɵBrowserPlatformLocation"])); | ||
}(common.ɵBrowserPlatformLocation)); | ||
SingleSpaPlatformLocation.decorators = [ | ||
{ type: core.Injectable } | ||
]; | ||
/** | ||
@@ -329,8 +445,8 @@ * The `PlatformLocation` class is an "injectee" of the `PathLocationStrategy`, | ||
var defaultOpts = { | ||
// Required opts that will be set by the library consumer. | ||
var defaultOptions = { | ||
// Required options that will be set by the library consumer. | ||
NgZone: null, | ||
bootstrapFunction: null, | ||
template: null, | ||
// optional opts | ||
// Optional options | ||
Router: undefined, | ||
@@ -341,24 +457,24 @@ domElementGetter: undefined, | ||
}; | ||
function singleSpaAngular(userOpts) { | ||
if (typeof userOpts !== 'object') { | ||
function singleSpaAngular(userOptions) { | ||
if (typeof userOptions !== 'object') { | ||
throw Error('single-spa-angular requires a configuration object'); | ||
} | ||
var opts = __assign(__assign({}, defaultOpts), userOpts); | ||
if (typeof opts.bootstrapFunction !== 'function') { | ||
throw Error('single-spa-angular must be passed an opts.bootstrapFunction'); | ||
var options = Object.assign(Object.assign({}, defaultOptions), userOptions); | ||
if (typeof options.bootstrapFunction !== 'function') { | ||
throw Error('single-spa-angular must be passed an options.bootstrapFunction'); | ||
} | ||
if (typeof opts.template !== 'string') { | ||
throw Error('single-spa-angular must be passed opts.template string'); | ||
if (typeof options.template !== 'string') { | ||
throw Error('single-spa-angular must be passed options.template string'); | ||
} | ||
if (!opts.NgZone) { | ||
throw Error("single-spa-angular must be passed the NgZone opt"); | ||
if (!options.NgZone) { | ||
throw Error("single-spa-angular must be passed the NgZone option"); | ||
} | ||
return { | ||
bootstrap: bootstrap.bind(null, opts), | ||
mount: mount.bind(null, opts), | ||
unmount: unmount.bind(null, opts), | ||
update: opts.updateFunction, | ||
bootstrap: bootstrap.bind(null, options), | ||
mount: mount.bind(null, options), | ||
unmount: unmount.bind(null, options), | ||
update: options.updateFunction, | ||
}; | ||
} | ||
function bootstrap(opts, props) { | ||
function bootstrap(options, props) { | ||
return __awaiter(this, void 0, void 0, function () { | ||
@@ -369,7 +485,7 @@ return __generator(this, function (_a) { | ||
// See https://angular.io/guide/zone#noopzone | ||
if (opts.NgZone === 'noop') { | ||
if (options.NgZone === 'noop') { | ||
return [2 /*return*/]; | ||
} | ||
// In order for multiple Angular apps to work concurrently on a page, they each need a unique identifier. | ||
opts.zoneIdentifier = "single-spa-angular:" + (props.name || props.appName); | ||
options.zoneIdentifier = "single-spa-angular:" + (props.name || props.appName); | ||
// This is a hack, since NgZone doesn't allow you to configure the property that identifies your zone. | ||
@@ -380,8 +496,8 @@ // See https://github.com/PlaceMe-SAS/single-spa-angular-cli/issues/33, | ||
// and https://github.com/angular/angular/blob/a14dc2d7a4821a19f20a9547053a5734798f541e/packages/core/src/zone/ng_zone.ts#L257 | ||
opts.NgZone.isInAngularZone = function () { | ||
options.NgZone.isInAngularZone = function () { | ||
// @ts-ignore | ||
return window.Zone.current._properties[opts.zoneIdentifier] === true; | ||
return window.Zone.current._properties[options.zoneIdentifier] === true; | ||
}; | ||
opts.routingEventListener = function () { | ||
opts.bootstrappedNgZone.run(function () { | ||
options.routingEventListener = function () { | ||
options.bootstrappedNgZone.run(function () { | ||
// See https://github.com/single-spa/single-spa-angular/issues/86 | ||
@@ -396,17 +512,12 @@ // Zone is unaware of the single-spa navigation change and so Angular change detection doesn't work | ||
} | ||
function mount(opts, props) { | ||
function mount(options, props) { | ||
return __awaiter(this, void 0, void 0, function () { | ||
var domElementGetter, containerEl, bootstrapPromise, module, singleSpaPlatformLocation, ngZoneEnabled, bootstrappedOpts, ngZone; | ||
var bootstrapPromise, module, singleSpaPlatformLocation, ngZoneEnabled, bootstrappedOptions, ngZone, zoneIdentifier_1; | ||
return __generator(this, function (_a) { | ||
switch (_a.label) { | ||
case 0: | ||
domElementGetter = internals.chooseDomElementGetter(opts, props); | ||
if (!domElementGetter) { | ||
throw Error("cannot mount angular application '" + (props.name || props.appName) + "' without a domElementGetter provided either as an opt or a prop"); | ||
} | ||
containerEl = internals.getContainerEl(domElementGetter); | ||
containerEl.innerHTML = opts.template; | ||
bootstrapPromise = opts.bootstrapFunction(props); | ||
internals.getContainerElementAndSetTemplate(options, props); | ||
bootstrapPromise = options.bootstrapFunction(props); | ||
if (!(bootstrapPromise instanceof Promise)) { | ||
throw Error("single-spa-angular: the opts.bootstrapFunction must return a promise, but instead returned a '" + typeof bootstrapPromise + "' that is not a Promise"); | ||
throw Error("single-spa-angular: the options.bootstrapFunction must return a promise, but instead returned a '" + typeof bootstrapPromise + "' that is not a Promise"); | ||
} | ||
@@ -417,6 +528,6 @@ return [4 /*yield*/, bootstrapPromise]; | ||
if (!module || typeof module.destroy !== 'function') { | ||
throw Error("single-spa-angular: the opts.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?"); | ||
throw Error("single-spa-angular: the options.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?"); | ||
} | ||
singleSpaPlatformLocation = module.injector.get(SingleSpaPlatformLocation, null); | ||
ngZoneEnabled = opts.NgZone !== 'noop'; | ||
ngZoneEnabled = options.NgZone !== 'noop'; | ||
// The user has to provide `BrowserPlatformLocation` only if his application uses routing. | ||
@@ -426,21 +537,21 @@ // So if he provided `Router` but didn't provide `BrowserPlatformLocation` then we have to inform him. | ||
// `zone-less` change detection, if `NgZone` is `noop` then we can skip it. | ||
if (ngZoneEnabled && opts.Router && singleSpaPlatformLocation === null) { | ||
if (ngZoneEnabled && options.Router && singleSpaPlatformLocation === null) { | ||
throw new Error("\t\n single-spa-angular: could not retrieve extra providers from the platform injector. Did you call platformBrowserDynamic(getSingleSpaExtraProviders()).bootstrapModule()?\n "); | ||
} | ||
bootstrappedOpts = opts; | ||
bootstrappedOptions = options; | ||
if (ngZoneEnabled) { | ||
ngZone = module.injector.get(opts.NgZone); | ||
ngZone = module.injector.get(options.NgZone); | ||
zoneIdentifier_1 = bootstrappedOptions.zoneIdentifier; | ||
// `NgZone` can be enabled but routing may not be used thus `getSingleSpaExtraProviders()` | ||
// function was not called. | ||
if (singleSpaPlatformLocation !== null) { | ||
singleSpaPlatformLocation.setNgZone(ngZone); | ||
// Cleanup resources, especially remove event listeners thus they will not be added | ||
// twice when application gets bootstrapped the second time. | ||
module.onDestroy(function () { return singleSpaPlatformLocation.destroy(); }); | ||
module.onDestroy(function () { return singleSpaPlatformLocation.destroyApplication(zoneIdentifier_1); }); | ||
} | ||
bootstrappedOpts.bootstrappedNgZone = ngZone; | ||
bootstrappedOpts.bootstrappedNgZone['_inner']._properties[bootstrappedOpts.zoneIdentifier] = true; | ||
window.addEventListener('single-spa:routing-event', bootstrappedOpts.routingEventListener); | ||
bootstrappedOptions.bootstrappedNgZone = ngZone; | ||
bootstrappedOptions.bootstrappedNgZone['_inner']._properties[zoneIdentifier_1] = true; | ||
window.addEventListener('single-spa:routing-event', bootstrappedOptions.routingEventListener); | ||
} | ||
bootstrappedOpts.bootstrappedModule = module; | ||
bootstrappedOptions.bootstrappedModule = module; | ||
return [2 /*return*/, module]; | ||
@@ -452,22 +563,22 @@ } | ||
// eslint-disable-next-line @typescript-eslint/no-unused-vars | ||
function unmount(opts, props) { | ||
function unmount(options, props) { | ||
return __awaiter(this, void 0, void 0, function () { | ||
var router, animationEngine; | ||
return __generator(this, function (_a) { | ||
if (opts.Router) { | ||
router = opts.bootstrappedModule.injector.get(opts.Router); | ||
if (options.Router) { | ||
router = options.bootstrappedModule.injector.get(options.Router); | ||
router.dispose(); | ||
} | ||
if (opts.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', opts.routingEventListener); | ||
if (options.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', options.routingEventListener); | ||
} | ||
if (opts.AnimationEngine) { | ||
animationEngine = opts.bootstrappedModule.injector.get(opts.AnimationEngine); | ||
if (options.AnimationEngine) { | ||
animationEngine = options.bootstrappedModule.injector.get(options.AnimationEngine); | ||
animationEngine._transitionEngine.flush(); | ||
} | ||
opts.bootstrappedModule.destroy(); | ||
delete opts.bootstrappedModule; | ||
options.bootstrappedModule.destroy(); | ||
delete options.bootstrappedModule; | ||
// This is an issue. Issue has been created and Angular team is working on the fix: | ||
// https://github.com/angular/angular/issues/36449 | ||
internals.removeApplicationFromDOMIfIvyEnabled(opts, props); | ||
internals.removeApplicationFromDOMIfIvyEnabled(options, props); | ||
return [2 /*return*/]; | ||
@@ -512,3 +623,3 @@ }); | ||
} | ||
_this.parcel = mountParcel(_this.config, __assign({ domElement: domElement }, _this.customProps)); | ||
_this.parcel = mountParcel(_this.config, Object.assign({ domElement: domElement }, _this.customProps)); | ||
if (_this.onParcelMount) { | ||
@@ -577,34 +688,22 @@ _this.parcel.mountPromise.then(_this.onParcelMount); | ||
}; | ||
ParcelComponent.ctorParameters = function () { return [ | ||
{ type: core.ElementRef } | ||
]; }; | ||
__decorate([ | ||
core.Input() | ||
], ParcelComponent.prototype, "config", void 0); | ||
__decorate([ | ||
core.Input() | ||
], ParcelComponent.prototype, "mountParcel", void 0); | ||
__decorate([ | ||
core.Input() | ||
], ParcelComponent.prototype, "onParcelMount", void 0); | ||
__decorate([ | ||
core.Input() | ||
], ParcelComponent.prototype, "wrapWith", void 0); | ||
__decorate([ | ||
core.Input() | ||
], ParcelComponent.prototype, "customProps", void 0); | ||
__decorate([ | ||
core.Input() | ||
], ParcelComponent.prototype, "appendTo", void 0); | ||
__decorate([ | ||
core.Input() | ||
], ParcelComponent.prototype, "handleError", void 0); | ||
ParcelComponent = __decorate([ | ||
core.Component({ | ||
selector: 'parcel', | ||
template: '<div></div>' | ||
}) | ||
], ParcelComponent); | ||
return ParcelComponent; | ||
}()); | ||
ParcelComponent.decorators = [ | ||
{ type: core.Component, args: [{ | ||
selector: 'parcel', | ||
template: '<div></div>' | ||
},] } | ||
]; | ||
ParcelComponent.ctorParameters = function () { return [ | ||
{ type: core.ElementRef } | ||
]; }; | ||
ParcelComponent.propDecorators = { | ||
config: [{ type: core.Input }], | ||
mountParcel: [{ type: core.Input }], | ||
onParcelMount: [{ type: core.Input }], | ||
wrapWith: [{ type: core.Input }], | ||
customProps: [{ type: core.Input }], | ||
appendTo: [{ type: core.Input }], | ||
handleError: [{ type: core.Input }] | ||
}; | ||
@@ -614,12 +713,20 @@ var ParcelModule = /** @class */ (function () { | ||
} | ||
ParcelModule = __decorate([ | ||
core.NgModule({ | ||
declarations: [ParcelComponent], | ||
exports: [ParcelComponent], | ||
entryComponents: [ParcelComponent], | ||
}) | ||
], ParcelModule); | ||
return ParcelModule; | ||
}()); | ||
ParcelModule.decorators = [ | ||
{ type: core.NgModule, args: [{ | ||
declarations: [ParcelComponent], | ||
exports: [ParcelComponent], | ||
entryComponents: [ParcelComponent], | ||
},] } | ||
]; | ||
/** | ||
* The public api for consumers of single-spa-angular | ||
*/ | ||
/** | ||
* Generated bundle index. Do not edit. | ||
*/ | ||
exports.ParcelComponent = ParcelComponent; | ||
@@ -626,0 +733,0 @@ exports.ParcelModule = ParcelModule; |
@@ -1,16 +0,16 @@ | ||
!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports,require("single-spa-angular/internals"),require("@angular/core"),require("@angular/common")):"function"==typeof define&&define.amd?define("single-spa-angular",["exports","single-spa-angular/internals","@angular/core","@angular/common"],e):e((t=t||self)["single-spa-angular"]={},t["single-spa-angular"].internals,t.ng.core,t.ng.common)}(this,(function(t,e,n,o){"use strict"; | ||
!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports,require("single-spa-angular/internals"),require("@angular/core"),require("@angular/common")):"function"==typeof define&&define.amd?define("single-spa-angular",["exports","single-spa-angular/internals","@angular/core","@angular/common"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self)["single-spa-angular"]={},t["single-spa-angular"].internals,t.ng.core,t.ng.common)}(this,(function(t,e,n,o){"use strict"; | ||
/*! ***************************************************************************** | ||
Copyright (c) Microsoft Corporation. All rights reserved. | ||
Licensed under the Apache License, Version 2.0 (the "License"); you may not use | ||
this file except in compliance with the License. You may obtain a copy of the | ||
License at http://www.apache.org/licenses/LICENSE-2.0 | ||
Copyright (c) Microsoft Corporation. | ||
THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | ||
KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED | ||
WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE, | ||
MERCHANTABLITY OR NON-INFRINGEMENT. | ||
Permission to use, copy, modify, and/or distribute this software for any | ||
purpose with or without fee is hereby granted. | ||
See the Apache Version 2.0 License for specific language governing permissions | ||
and limitations under the License. | ||
***************************************************************************** */var r=function(t,e){return(r=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(t,e){t.__proto__=e}||function(t,e){for(var n in e)e.hasOwnProperty(n)&&(t[n]=e[n])})(t,e)};var i=function(){return(i=Object.assign||function(t){for(var e,n=1,o=arguments.length;n<o;n++)for(var r in e=arguments[n])Object.prototype.hasOwnProperty.call(e,r)&&(t[r]=e[r]);return t}).apply(this,arguments)};function a(t,e,n,o){var r,i=arguments.length,a=i<3?e:null===o?o=Object.getOwnPropertyDescriptor(e,n):o;if("object"==typeof Reflect&&"function"==typeof Reflect.decorate)a=Reflect.decorate(t,e,n,o);else for(var u=t.length-1;u>=0;u--)(r=t[u])&&(a=(i<3?r(a):i>3?r(e,n,a):r(e,n))||a);return i>3&&a&&Object.defineProperty(e,n,a),a}function u(t,e,n,o){return new(n||(n=Promise))((function(r,i){function a(t){try{s(o.next(t))}catch(t){i(t)}}function u(t){try{s(o.throw(t))}catch(t){i(t)}}function s(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(a,u)}s((o=o.apply(t,e||[])).next())}))}function s(t,e){var n,o,r,i,a={label:0,sent:function(){if(1&r[0])throw r[1];return r[1]},trys:[],ops:[]};return i={next:u(0),throw:u(1),return:u(2)},"function"==typeof Symbol&&(i[Symbol.iterator]=function(){return this}),i;function u(i){return function(u){return function(i){if(n)throw new TypeError("Generator is already executing.");for(;a;)try{if(n=1,o&&(r=2&i[0]?o.return:i[0]?o.throw||((r=o.return)&&r.call(o),0):o.next)&&!(r=r.call(o,i[1])).done)return r;switch(o=0,r&&(i=[2&i[0],r.value]),i[0]){case 0:case 1:r=i;break;case 4:return a.label++,{value:i[1],done:!1};case 5:a.label++,o=i[1],i=[0];continue;case 7:i=a.ops.pop(),a.trys.pop();continue;default:if(!(r=a.trys,(r=r.length>0&&r[r.length-1])||6!==i[0]&&2!==i[0])){a=0;continue}if(3===i[0]&&(!r||i[1]>r[0]&&i[1]<r[3])){a.label=i[1];break}if(6===i[0]&&a.label<r[1]){a.label=r[1],r=i;break}if(r&&a.label<r[2]){a.label=r[2],a.ops.push(i);break}r[2]&&a.ops.pop(),a.trys.pop();continue}i=e.call(t,a)}catch(t){i=[6,t],o=0}finally{n=r=0}if(5&i[0])throw i[1];return{value:i[0]?i[1]:void 0,done:!0}}([i,u])}}}function p(t){var e="function"==typeof Symbol&&Symbol.iterator,n=e&&t[e],o=0;if(n)return n.call(t);if(t&&"number"==typeof t.length)return{next:function(){return t&&o>=t.length&&(t=void 0),{value:t&&t[o++],done:!t}}};throw new TypeError(e?"Object is not iterable.":"Symbol.iterator is not defined.")}function l(t,e){var n="function"==typeof Symbol&&t[Symbol.iterator];if(!n)return t;var o,r,i=n.call(t),a=[];try{for(;(void 0===e||e-- >0)&&!(o=i.next()).done;)a.push(o.value)}catch(t){r={error:t}}finally{try{o&&!o.done&&(n=i.return)&&n.call(i)}finally{if(r)throw r.error}}return a}function c(){for(var t=[],e=0;e<arguments.length;e++)t=t.concat(l(arguments[e]));return t}var f=function(t){function e(){var e=null!==t&&t.apply(this,arguments)||this;return e.skipNextPopState=!1,e.onPopStateListeners=[],e}return function(t,e){function n(){this.constructor=t}r(t,e),t.prototype=null===e?Object.create(e):(n.prototype=e.prototype,new n)}(e,t),e.prototype.destroy=function(){var t,e;try{for(var n=p(this.onPopStateListeners),o=n.next();!o.done;o=n.next()){var r=o.value;window.removeEventListener("popstate",r)}}catch(e){t={error:e}}finally{try{o&&!o.done&&(e=n.return)&&e.call(n)}finally{if(t)throw t.error}}this.onPopStateListeners=[]},e.prototype.pushState=function(e,n,o){this.skipNextPopState=!0,t.prototype.pushState.call(this,e,n,o)},e.prototype.replaceState=function(e,n,o){this.skipNextPopState=!0,t.prototype.replaceState.call(this,e,n,o)},e.prototype.onPopState=function(e){var n=this,o=function(t){var o=!!t.singleSpa;n.skipNextPopState&&o?n.skipNextPopState=!1:n.ngZone.run((function(){return e(t)}))};this.onPopStateListeners.push(o),t.prototype.onPopState.call(this,o)},e.prototype.setNgZone=function(t){this.ngZone=t},e=a([n.Injectable()],e)}(o["ɵBrowserPlatformLocation"]);var d={NgZone:null,bootstrapFunction:null,template:null,Router:void 0,domElementGetter:void 0,AnimationEngine:void 0,updateFunction:function(){return Promise.resolve()}};function h(t,e){return u(this,void 0,void 0,(function(){return s(this,(function(n){return"noop"===t.NgZone||(t.zoneIdentifier="single-spa-angular:"+(e.name||e.appName),t.NgZone.isInAngularZone=function(){return!0===window.Zone.current._properties[t.zoneIdentifier]},t.routingEventListener=function(){t.bootstrappedNgZone.run((function(){}))}),[2]}))}))}function g(t,n){return u(this,void 0,void 0,(function(){var o,r,i,a,u,p,l;return s(this,(function(s){switch(s.label){case 0:if(!(o=e.chooseDomElementGetter(t,n)))throw Error("cannot mount angular application '"+(n.name||n.appName)+"' without a domElementGetter provided either as an opt or a prop");if(e.getContainerEl(o).innerHTML=t.template,!((r=t.bootstrapFunction(n))instanceof Promise))throw Error("single-spa-angular: the opts.bootstrapFunction must return a promise, but instead returned a '"+typeof r+"' that is not a Promise");return[4,r];case 1:if(!(i=s.sent())||"function"!=typeof i.destroy)throw Error("single-spa-angular: the opts.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?");if(a=i.injector.get(f,null),(u="noop"!==t.NgZone)&&t.Router&&null===a)throw new Error("\t\n single-spa-angular: could not retrieve extra providers from the platform injector. Did you call platformBrowserDynamic(getSingleSpaExtraProviders()).bootstrapModule()?\n ");return p=t,u&&(l=i.injector.get(t.NgZone),null!==a&&(a.setNgZone(l),i.onDestroy((function(){return a.destroy()}))),p.bootstrappedNgZone=l,p.bootstrappedNgZone._inner._properties[p.zoneIdentifier]=!0,window.addEventListener("single-spa:routing-event",p.routingEventListener)),p.bootstrappedModule=i,[2,i]}}))}))}function m(t,n){return u(this,void 0,void 0,(function(){return s(this,(function(o){return t.Router&&t.bootstrappedModule.injector.get(t.Router).dispose(),t.routingEventListener&&window.removeEventListener("single-spa:routing-event",t.routingEventListener),t.AnimationEngine&&t.bootstrappedModule.injector.get(t.AnimationEngine)._transitionEngine.flush(),t.bootstrappedModule.destroy(),delete t.bootstrappedModule,e.removeApplicationFromDOMIfIvyEnabled(t,n),[2]}))}))}var y=function(){function t(t){this.host=t,this.onParcelMount=null,this.wrapWith="div",this.customProps={},this.appendTo=null,this.handleError=function(t){return console.error(t)},this.createdDomElement=null,this.hasError=!1,this.unmounted=!0}return t.prototype.ngOnInit=function(){var t=this;if(!this.config)throw new Error("single-spa-angular's Parcel component requires the [config] binding to either be a parcel config or a loading function that returns a promise. See https://github.com/CanopyTax/single-spa-angular");this.addThingToDo("mount",(function(){var e,n=t.mountParcel;if(!n)throw new Error("\n\t\t\t\t <parcel> was not passed a [mountParcel] binding.\n\t\t\t\t If you are using <parcel> within a module that is not a single-spa application, you will need to import mountRootParcel from single-spa and pass it into <parcel> as a [mountParcel] binding\n\t\t\t\t");t.appendTo?(t.createdDomElement=e=document.createElement(t.wrapWith),t.appendTo.appendChild(e)):(t.createdDomElement=e=document.createElement(t.wrapWith),t.host.nativeElement.children[0].appendChild(e));return t.parcel=n(t.config,i({domElement:e},t.customProps)),t.onParcelMount&&t.parcel.mountPromise.then(t.onParcelMount),t.unmounted=!1,t.parcel.mountPromise}))},t.prototype.ngOnChanges=function(){var t=this;this.addThingToDo("update",(function(){if(t.parcel&&t.parcel.update)return t.parcel.update(t.customProps)}))},t.prototype.ngOnDestroy=function(){var t=this;this.addThingToDo("unmount",(function(){if(t.parcel&&"MOUNTED"===t.parcel.getStatus())return t.parcel.unmount()})),this.createdDomElement&&this.createdDomElement.parentNode.removeChild(this.createdDomElement),this.unmounted=!0},t.prototype.addThingToDo=function(t,e){var n=this;this.hasError&&"unmount"!==t||(this.nextThingToDo=(this.nextThingToDo||Promise.resolve()).then((function(){for(var o=[],r=0;r<arguments.length;r++)o[r]=arguments[r];if(!n.unmounted||"unmount"===t)return e.apply(void 0,c(o))})).catch((function(e){throw n.nextThingToDo=Promise.resolve(),n.hasError=!0,e&&e.message&&(e.message="During '"+t+"', parcel threw an error: "+e.message),"function"==typeof n.handleError?n.handleError(e):setTimeout((function(){throw e})),e})))},t.ctorParameters=function(){return[{type:n.ElementRef}]},a([n.Input()],t.prototype,"config",void 0),a([n.Input()],t.prototype,"mountParcel",void 0),a([n.Input()],t.prototype,"onParcelMount",void 0),a([n.Input()],t.prototype,"wrapWith",void 0),a([n.Input()],t.prototype,"customProps",void 0),a([n.Input()],t.prototype,"appendTo",void 0),a([n.Input()],t.prototype,"handleError",void 0),t=a([n.Component({selector:"parcel",template:"<div></div>"})],t)}(),v=function(){function t(){}return t=a([n.NgModule({declarations:[y],exports:[y],entryComponents:[y]})],t)}();t.ParcelComponent=y,t.ParcelModule=v,t.getSingleSpaExtraProviders=function(){return[{provide:f,useClass:f,deps:[[new n.Inject(o.DOCUMENT)]]},{provide:o.PlatformLocation,useExisting:f}]},t.singleSpaAngular=function(t){if("object"!=typeof t)throw Error("single-spa-angular requires a configuration object");var e=i(i({},d),t);if("function"!=typeof e.bootstrapFunction)throw Error("single-spa-angular must be passed an opts.bootstrapFunction");if("string"!=typeof e.template)throw Error("single-spa-angular must be passed opts.template string");if(!e.NgZone)throw Error("single-spa-angular must be passed the NgZone opt");return{bootstrap:h.bind(null,e),mount:g.bind(null,e),unmount:m.bind(null,e),update:e.updateFunction}},t.ɵa=f,Object.defineProperty(t,"__esModule",{value:!0})})); | ||
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH | ||
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY | ||
AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, | ||
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM | ||
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR | ||
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR | ||
PERFORMANCE OF THIS SOFTWARE. | ||
***************************************************************************** */var r=function(t,e){return(r=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(t,e){t.__proto__=e}||function(t,e){for(var n in e)Object.prototype.hasOwnProperty.call(e,n)&&(t[n]=e[n])})(t,e)};function i(t,e,n,o){return new(n||(n=Promise))((function(r,i){function a(t){try{u(o.next(t))}catch(t){i(t)}}function s(t){try{u(o.throw(t))}catch(t){i(t)}}function u(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(a,s)}u((o=o.apply(t,e||[])).next())}))}function a(t,e){var n,o,r,i,a={label:0,sent:function(){if(1&r[0])throw r[1];return r[1]},trys:[],ops:[]};return i={next:s(0),throw:s(1),return:s(2)},"function"==typeof Symbol&&(i[Symbol.iterator]=function(){return this}),i;function s(i){return function(s){return function(i){if(n)throw new TypeError("Generator is already executing.");for(;a;)try{if(n=1,o&&(r=2&i[0]?o.return:i[0]?o.throw||((r=o.return)&&r.call(o),0):o.next)&&!(r=r.call(o,i[1])).done)return r;switch(o=0,r&&(i=[2&i[0],r.value]),i[0]){case 0:case 1:r=i;break;case 4:return a.label++,{value:i[1],done:!1};case 5:a.label++,o=i[1],i=[0];continue;case 7:i=a.ops.pop(),a.trys.pop();continue;default:if(!(r=a.trys,(r=r.length>0&&r[r.length-1])||6!==i[0]&&2!==i[0])){a=0;continue}if(3===i[0]&&(!r||i[1]>r[0]&&i[1]<r[3])){a.label=i[1];break}if(6===i[0]&&a.label<r[1]){a.label=r[1],r=i;break}if(r&&a.label<r[2]){a.label=r[2],a.ops.push(i);break}r[2]&&a.ops.pop(),a.trys.pop();continue}i=e.call(t,a)}catch(t){i=[6,t],o=0}finally{n=r=0}if(5&i[0])throw i[1];return{value:i[0]?i[1]:void 0,done:!0}}([i,s])}}}Object.create;function s(t){var e="function"==typeof Symbol&&Symbol.iterator,n=e&&t[e],o=0;if(n)return n.call(t);if(t&&"number"==typeof t.length)return{next:function(){return t&&o>=t.length&&(t=void 0),{value:t&&t[o++],done:!t}}};throw new TypeError(e?"Object is not iterable.":"Symbol.iterator is not defined.")}function u(t,e){var n="function"==typeof Symbol&&t[Symbol.iterator];if(!n)return t;var o,r,i=n.call(t),a=[];try{for(;(void 0===e||e-- >0)&&!(o=i.next()).done;)a.push(o.value)}catch(t){r={error:t}}finally{try{o&&!o.done&&(n=i.return)&&n.call(i)}finally{if(r)throw r.error}}return a}function p(){for(var t=[],e=0;e<arguments.length;e++)t=t.concat(u(arguments[e]));return t}Object.create;var l=function(t){function e(){var e=t.apply(this,p(arguments))||this;return e.skipNextPopState=!1,e.zoneToOnPopStateListenersMap=new Map,e.source=0,e}return function(t,e){function n(){this.constructor=t}r(t,e),t.prototype=null===e?Object.create(e):(n.prototype=e.prototype,new n)}(e,t),e.prototype.destroyApplication=function(t){var e,n,o=p(this.zoneToOnPopStateListenersMap.keys()).find((function(e){return null!==e.getZoneWith(t)})),r=this.zoneToOnPopStateListenersMap.get(o);if(Array.isArray(r))try{for(var i=s(r),a=i.next();!a.done;a=i.next()){var u=a.value;window.removeEventListener("popstate",u)}}catch(t){e={error:t}}finally{try{a&&!a.done&&(n=i.return)&&n.call(i)}finally{if(e)throw e.error}}this.zoneToOnPopStateListenersMap.delete(o)},e.prototype.pushState=function(e,n,o){this.skipNextPopState=!0,t.prototype.pushState.call(this,e,n,o)},e.prototype.replaceState=function(e,n,o){this.skipNextPopState=!0,t.prototype.replaceState.call(this,e,n,o)},e.prototype.onPopState=function(e){var n=this,o=Zone.current;e=o.wrap(e,""+this.source++);var r=function(t){var o=!!t.singleSpa;n.skipNextPopState&&o?n.skipNextPopState=!1:e(t)};this.storeOnPopStateListener(o,r),t.prototype.onPopState.call(this,r)},e.prototype.storeOnPopStateListener=function(t,e){var n=this.zoneToOnPopStateListenersMap.get(t)||[];n.push(e),this.zoneToOnPopStateListenersMap.has(t)||this.zoneToOnPopStateListenersMap.set(t,n)},e}(o.ɵBrowserPlatformLocation);l.decorators=[{type:n.Injectable}];var c={NgZone:null,bootstrapFunction:null,template:null,Router:void 0,domElementGetter:void 0,AnimationEngine:void 0,updateFunction:function(){return Promise.resolve()}};function f(t,e){return i(this,void 0,void 0,(function(){return a(this,(function(n){return"noop"===t.NgZone||(t.zoneIdentifier="single-spa-angular:"+(e.name||e.appName),t.NgZone.isInAngularZone=function(){return!0===window.Zone.current._properties[t.zoneIdentifier]},t.routingEventListener=function(){t.bootstrappedNgZone.run((function(){}))}),[2]}))}))}function d(t,n){return i(this,void 0,void 0,(function(){var o,r,i,s,u,p,c;return a(this,(function(a){switch(a.label){case 0:if(e.getContainerElementAndSetTemplate(t,n),!((o=t.bootstrapFunction(n))instanceof Promise))throw Error("single-spa-angular: the options.bootstrapFunction must return a promise, but instead returned a '"+typeof o+"' that is not a Promise");return[4,o];case 1:if(!(r=a.sent())||"function"!=typeof r.destroy)throw Error("single-spa-angular: the options.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?");if(i=r.injector.get(l,null),(s="noop"!==t.NgZone)&&t.Router&&null===i)throw new Error("\t\n single-spa-angular: could not retrieve extra providers from the platform injector. Did you call platformBrowserDynamic(getSingleSpaExtraProviders()).bootstrapModule()?\n ");return u=t,s&&(p=r.injector.get(t.NgZone),c=u.zoneIdentifier,null!==i&&r.onDestroy((function(){return i.destroyApplication(c)})),u.bootstrappedNgZone=p,u.bootstrappedNgZone._inner._properties[c]=!0,window.addEventListener("single-spa:routing-event",u.routingEventListener)),u.bootstrappedModule=r,[2,r]}}))}))}function h(t,n){return i(this,void 0,void 0,(function(){return a(this,(function(o){return t.Router&&t.bootstrappedModule.injector.get(t.Router).dispose(),t.routingEventListener&&window.removeEventListener("single-spa:routing-event",t.routingEventListener),t.AnimationEngine&&t.bootstrappedModule.injector.get(t.AnimationEngine)._transitionEngine.flush(),t.bootstrappedModule.destroy(),delete t.bootstrappedModule,e.removeApplicationFromDOMIfIvyEnabled(t,n),[2]}))}))}var g=function(){function t(t){this.host=t,this.onParcelMount=null,this.wrapWith="div",this.customProps={},this.appendTo=null,this.handleError=function(t){return console.error(t)},this.createdDomElement=null,this.hasError=!1,this.unmounted=!0}return t.prototype.ngOnInit=function(){var t=this;if(!this.config)throw new Error("single-spa-angular's Parcel component requires the [config] binding to either be a parcel config or a loading function that returns a promise. See https://github.com/CanopyTax/single-spa-angular");this.addThingToDo("mount",(function(){var e,n=t.mountParcel;if(!n)throw new Error("\n\t\t\t\t <parcel> was not passed a [mountParcel] binding.\n\t\t\t\t If you are using <parcel> within a module that is not a single-spa application, you will need to import mountRootParcel from single-spa and pass it into <parcel> as a [mountParcel] binding\n\t\t\t\t");t.appendTo?(t.createdDomElement=e=document.createElement(t.wrapWith),t.appendTo.appendChild(e)):(t.createdDomElement=e=document.createElement(t.wrapWith),t.host.nativeElement.children[0].appendChild(e));return t.parcel=n(t.config,Object.assign({domElement:e},t.customProps)),t.onParcelMount&&t.parcel.mountPromise.then(t.onParcelMount),t.unmounted=!1,t.parcel.mountPromise}))},t.prototype.ngOnChanges=function(){var t=this;this.addThingToDo("update",(function(){if(t.parcel&&t.parcel.update)return t.parcel.update(t.customProps)}))},t.prototype.ngOnDestroy=function(){var t=this;this.addThingToDo("unmount",(function(){if(t.parcel&&"MOUNTED"===t.parcel.getStatus())return t.parcel.unmount()})),this.createdDomElement&&this.createdDomElement.parentNode.removeChild(this.createdDomElement),this.unmounted=!0},t.prototype.addThingToDo=function(t,e){var n=this;this.hasError&&"unmount"!==t||(this.nextThingToDo=(this.nextThingToDo||Promise.resolve()).then((function(){for(var o=[],r=0;r<arguments.length;r++)o[r]=arguments[r];if(!n.unmounted||"unmount"===t)return e.apply(void 0,p(o))})).catch((function(e){throw n.nextThingToDo=Promise.resolve(),n.hasError=!0,e&&e.message&&(e.message="During '"+t+"', parcel threw an error: "+e.message),"function"==typeof n.handleError?n.handleError(e):setTimeout((function(){throw e})),e})))},t}();g.decorators=[{type:n.Component,args:[{selector:"parcel",template:"<div></div>"}]}],g.ctorParameters=function(){return[{type:n.ElementRef}]},g.propDecorators={config:[{type:n.Input}],mountParcel:[{type:n.Input}],onParcelMount:[{type:n.Input}],wrapWith:[{type:n.Input}],customProps:[{type:n.Input}],appendTo:[{type:n.Input}],handleError:[{type:n.Input}]};var m=function(){};m.decorators=[{type:n.NgModule,args:[{declarations:[g],exports:[g],entryComponents:[g]}]}],t.ParcelComponent=g,t.ParcelModule=m,t.getSingleSpaExtraProviders=function(){return[{provide:l,useClass:l,deps:[[new n.Inject(o.DOCUMENT)]]},{provide:o.PlatformLocation,useExisting:l}]},t.singleSpaAngular=function(t){if("object"!=typeof t)throw Error("single-spa-angular requires a configuration object");var e=Object.assign(Object.assign({},c),t);if("function"!=typeof e.bootstrapFunction)throw Error("single-spa-angular must be passed an options.bootstrapFunction");if("string"!=typeof e.template)throw Error("single-spa-angular must be passed options.template string");if(!e.NgZone)throw Error("single-spa-angular must be passed the NgZone option");return{bootstrap:f.bind(null,e),mount:d.bind(null,e),unmount:h.bind(null,e),update:e.updateFunction}},t.ɵa=l,Object.defineProperty(t,"__esModule",{value:!0})})); | ||
//# sourceMappingURL=single-spa-angular.umd.min.js.map |
@@ -5,2 +5,2 @@ /** | ||
export * from './src/public_api'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290Ijoibmc6Ly9zaW5nbGUtc3BhLWFuZ3VsYXIvIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUNILGNBQWMsa0JBQWtCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFRoZSBwdWJsaWMgYXBpIGZvciBjb25zdW1lcnMgb2Ygc2luZ2xlLXNwYS1hbmd1bGFyXG4gKi9cbmV4cG9ydCAqIGZyb20gJy4vc3JjL3B1YmxpY19hcGknO1xuIl19 | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFDSCxjQUFjLGtCQUFrQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBUaGUgcHVibGljIGFwaSBmb3IgY29uc3VtZXJzIG9mIHNpbmdsZS1zcGEtYW5ndWxhclxuICovXG5leHBvcnQgKiBmcm9tICcuL3NyYy9wdWJsaWNfYXBpJztcbiJdfQ== |
@@ -1,5 +0,5 @@ | ||
export function removeApplicationFromDOMIfIvyEnabled(opts, props) { | ||
export function removeApplicationFromDOMIfIvyEnabled(options, props) { | ||
if (ivyEnabled()) { | ||
const domElementGetter = chooseDomElementGetter(opts, props); | ||
const domElement = getContainerEl(domElementGetter); | ||
const domElementGetter = chooseDomElementGetter(options, props); | ||
const domElement = getContainerElement(domElementGetter); | ||
// View Engine removes all nodes automatically when calling `NgModuleRef.destroy()`, | ||
@@ -12,3 +12,12 @@ // which calls `ComponentRef.destroy()`. | ||
} | ||
export function getContainerEl(domElementGetter) { | ||
export function getContainerElementAndSetTemplate(options, props) { | ||
const domElementGetter = chooseDomElementGetter(options, props); | ||
if (!domElementGetter) { | ||
throw Error(`Cannot mount angular application '${props.name || props.appName}' without a domElementGetter provided either as an opt or a prop`); | ||
} | ||
const containerElement = getContainerElement(domElementGetter); | ||
containerElement.innerHTML = options.template; | ||
return containerElement; | ||
} | ||
function getContainerElement(domElementGetter) { | ||
const element = domElementGetter(); | ||
@@ -20,3 +29,3 @@ if (!element) { | ||
} | ||
export function chooseDomElementGetter(opts, props) { | ||
function chooseDomElementGetter(opts, props) { | ||
var _a; | ||
@@ -63,2 +72,2 @@ props = (_a = props === null || props === void 0 ? void 0 : props.customProps) !== null && _a !== void 0 ? _a : props; | ||
} | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9tLmpzIiwic291cmNlUm9vdCI6Im5nOi8vc2luZ2xlLXNwYS1hbmd1bGFyL2ludGVybmFscy8iLCJzb3VyY2VzIjpbImRvbS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxNQUFNLFVBQVUsb0NBQW9DLENBQUMsSUFBMEIsRUFBRSxLQUFVO0lBQ3pGLElBQUksVUFBVSxFQUFFLEVBQUU7UUFDaEIsTUFBTSxnQkFBZ0IsR0FBRyxzQkFBc0IsQ0FBQyxJQUFJLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDN0QsTUFBTSxVQUFVLEdBQUcsY0FBYyxDQUFDLGdCQUFnQixDQUFDLENBQUM7UUFDcEQsb0ZBQW9GO1FBQ3BGLHdDQUF3QztRQUN4QywwRkFBMEY7UUFDMUYsT0FBTyxVQUFVLENBQUMsVUFBVTtZQUFFLFVBQVUsQ0FBQyxXQUFXLENBQUMsVUFBVSxDQUFDLFVBQVUsQ0FBQyxDQUFDO0tBQzdFO0FBQ0gsQ0FBQztBQUVELE1BQU0sVUFBVSxjQUFjLENBQUMsZ0JBQWtDO0lBQy9ELE1BQU0sT0FBTyxHQUFHLGdCQUFnQixFQUFFLENBQUM7SUFFbkMsSUFBSSxDQUFDLE9BQU8sRUFBRTtRQUNaLE1BQU0sS0FBSyxDQUFDLHFEQUFxRCxDQUFDLENBQUM7S0FDcEU7SUFFRCxPQUFPLE9BQU8sQ0FBQztBQUNqQixDQUFDO0FBRUQsTUFBTSxVQUFVLHNCQUFzQixDQUFDLElBQTBCLEVBQUUsS0FBVTs7SUFDM0UsS0FBSyxTQUFHLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxXQUFXLG1DQUFJLEtBQUssQ0FBQztJQUVwQyxJQUFJLEtBQUssQ0FBQyxVQUFVLEVBQUU7UUFDcEIsT0FBTyxHQUFHLEVBQUUsQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDO0tBQy9CO1NBQU0sSUFBSSxLQUFLLENBQUMsZ0JBQWdCLEVBQUU7UUFDakMsT0FBTyxLQUFLLENBQUMsZ0JBQWdCLENBQUM7S0FDL0I7U0FBTSxJQUFJLElBQUksQ0FBQyxnQkFBZ0IsRUFBRTtRQUNoQyxPQUFPLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQztLQUM5QjtTQUFNO1FBQ0wsT0FBTyx1QkFBdUIsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7S0FDNUM7QUFDSCxDQUFDO0FBRUQsU0FBUyx1QkFBdUIsQ0FBQyxJQUFZO0lBQzNDLE9BQU8sU0FBUyxvQkFBb0I7UUFDbEMsTUFBTSxFQUFFLEdBQUcsMEJBQTBCLElBQUksRUFBRSxDQUFDO1FBQzVDLElBQUksVUFBVSxHQUF1QixRQUFRLENBQUMsY0FBYyxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBRWpFLElBQUksQ0FBQyxVQUFVLEVBQUU7WUFDZixVQUFVLEdBQUcsUUFBUSxDQUFDLGFBQWEsQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUMzQyxVQUFVLENBQUMsRUFBRSxHQUFHLEVBQUUsQ0FBQztZQUNuQixRQUFRLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxVQUFVLENBQUMsQ0FBQztTQUN2QztRQUVELE9BQU8sVUFBVSxDQUFDO0lBQ3BCLENBQUMsQ0FBQztBQUNKLENBQUM7QUFFRCxTQUFTLFVBQVU7SUFDakIsSUFBSTtRQUNGLDZEQUE2RDtRQUM3RCxnRkFBZ0Y7UUFDaEYsOEVBQThFO1FBQzlFLGlEQUFpRDtRQUNqRCwyQkFBMkI7UUFDM0IsTUFBTSxFQUFFLFdBQVcsRUFBRSxHQUFHLE9BQU8sQ0FBQyxlQUFlLENBQUMsQ0FBQztRQUNqRCxPQUFPLENBQUMsQ0FBQyxXQUFXLENBQUM7S0FDdEI7SUFBQyxXQUFNO1FBQ04sT0FBTyxLQUFLLENBQUM7S0FDZDtBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTaW5nbGVTcGFBbmd1bGFyT3B0cywgRG9tRWxlbWVudEdldHRlciB9IGZyb20gJy4vdHlwZXMnO1xuXG5leHBvcnQgZnVuY3Rpb24gcmVtb3ZlQXBwbGljYXRpb25Gcm9tRE9NSWZJdnlFbmFibGVkKG9wdHM6IFNpbmdsZVNwYUFuZ3VsYXJPcHRzLCBwcm9wczogYW55KTogdm9pZCB7XG4gIGlmIChpdnlFbmFibGVkKCkpIHtcbiAgICBjb25zdCBkb21FbGVtZW50R2V0dGVyID0gY2hvb3NlRG9tRWxlbWVudEdldHRlcihvcHRzLCBwcm9wcyk7XG4gICAgY29uc3QgZG9tRWxlbWVudCA9IGdldENvbnRhaW5lckVsKGRvbUVsZW1lbnRHZXR0ZXIpO1xuICAgIC8vIFZpZXcgRW5naW5lIHJlbW92ZXMgYWxsIG5vZGVzIGF1dG9tYXRpY2FsbHkgd2hlbiBjYWxsaW5nIGBOZ01vZHVsZVJlZi5kZXN0cm95KClgLFxuICAgIC8vIHdoaWNoIGNhbGxzIGBDb21wb25lbnRSZWYuZGVzdHJveSgpYC5cbiAgICAvLyBCYXNpY2FsbHkgdGhpcyB3aWxsIHJlbW92ZSBgYXBwLXJvb3RgIG9yIGFueSBvdGhlciBzZWxlY3RvciBmcm9tIHRoZSBjb250YWluZXIgZWxlbWVudC5cbiAgICB3aGlsZSAoZG9tRWxlbWVudC5maXJzdENoaWxkKSBkb21FbGVtZW50LnJlbW92ZUNoaWxkKGRvbUVsZW1lbnQuZmlyc3RDaGlsZCk7XG4gIH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGdldENvbnRhaW5lckVsKGRvbUVsZW1lbnRHZXR0ZXI6IERvbUVsZW1lbnRHZXR0ZXIpOiBuZXZlciB8IEhUTUxFbGVtZW50IHtcbiAgY29uc3QgZWxlbWVudCA9IGRvbUVsZW1lbnRHZXR0ZXIoKTtcblxuICBpZiAoIWVsZW1lbnQpIHtcbiAgICB0aHJvdyBFcnJvcignZG9tRWxlbWVudEdldHRlciBkaWQgbm90IHJldHVybiBhIHZhbGlkIGRvbSBlbGVtZW50Jyk7XG4gIH1cblxuICByZXR1cm4gZWxlbWVudDtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNob29zZURvbUVsZW1lbnRHZXR0ZXIob3B0czogU2luZ2xlU3BhQW5ndWxhck9wdHMsIHByb3BzOiBhbnkpOiBEb21FbGVtZW50R2V0dGVyIHtcbiAgcHJvcHMgPSBwcm9wcz8uY3VzdG9tUHJvcHMgPz8gcHJvcHM7XG5cbiAgaWYgKHByb3BzLmRvbUVsZW1lbnQpIHtcbiAgICByZXR1cm4gKCkgPT4gcHJvcHMuZG9tRWxlbWVudDtcbiAgfSBlbHNlIGlmIChwcm9wcy5kb21FbGVtZW50R2V0dGVyKSB7XG4gICAgcmV0dXJuIHByb3BzLmRvbUVsZW1lbnRHZXR0ZXI7XG4gIH0gZWxzZSBpZiAob3B0cy5kb21FbGVtZW50R2V0dGVyKSB7XG4gICAgcmV0dXJuIG9wdHMuZG9tRWxlbWVudEdldHRlcjtcbiAgfSBlbHNlIHtcbiAgICByZXR1cm4gZGVmYXVsdERvbUVsZW1lbnRHZXR0ZXIocHJvcHMubmFtZSk7XG4gIH1cbn1cblxuZnVuY3Rpb24gZGVmYXVsdERvbUVsZW1lbnRHZXR0ZXIobmFtZTogc3RyaW5nKTogRG9tRWxlbWVudEdldHRlciB7XG4gIHJldHVybiBmdW5jdGlvbiBnZXREZWZhdWx0RG9tRWxlbWVudCgpIHtcbiAgICBjb25zdCBpZCA9IGBzaW5nbGUtc3BhLWFwcGxpY2F0aW9uOiR7bmFtZX1gO1xuICAgIGxldCBkb21FbGVtZW50OiBIVE1MRWxlbWVudCB8IG51bGwgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChpZCk7XG5cbiAgICBpZiAoIWRvbUVsZW1lbnQpIHtcbiAgICAgIGRvbUVsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdkaXYnKTtcbiAgICAgIGRvbUVsZW1lbnQuaWQgPSBpZDtcbiAgICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kQ2hpbGQoZG9tRWxlbWVudCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGRvbUVsZW1lbnQ7XG4gIH07XG59XG5cbmZ1bmN0aW9uIGl2eUVuYWJsZWQoKTogYm9vbGVhbiB7XG4gIHRyeSB7XG4gICAgLy8gYMm1aXZ5RW5hYmxlZGAgdmFyaWFibGUgaXMgZXhwb3NlZCBzdGFydGluZyBmcm9tIHZlcnNpb24gOC5cbiAgICAvLyBXZSB1c2UgYHJlcXVpcmVgIGhlcmUgZXhjZXB0IG9mIGEgc2luZ2xlIGBpbXBvcnQgeyDJtWl2eUVuYWJsZWQgfWAgYmVjYXVzZSB0aGVcbiAgICAvLyBkZXZlbG9wZXIgY2FuIHVzZSBBbmd1bGFyIHZlcnNpb24gdGhhdCBkb2Vzbid0IGV4cG9zZSBpdCAoYWxsIHZlcnNpb25zIDw4KS5cbiAgICAvLyBUaGUgYGNhdGNoYCBzdGF0ZW1lbnQgd2lsbCBoYW5kbGUgdGhvc2UgY2FzZXMuXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lXG4gICAgY29uc3QgeyDJtWl2eUVuYWJsZWQgfSA9IHJlcXVpcmUoJ0Bhbmd1bGFyL2NvcmUnKTtcbiAgICByZXR1cm4gISHJtWl2eUVuYWJsZWQ7XG4gIH0gY2F0Y2gge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxufVxuIl19 | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9tLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2ludGVybmFscy9zcmMvZG9tLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE1BQU0sVUFBVSxvQ0FBb0MsQ0FDbEQsT0FBVSxFQUNWLEtBQVU7SUFFVixJQUFJLFVBQVUsRUFBRSxFQUFFO1FBQ2hCLE1BQU0sZ0JBQWdCLEdBQUcsc0JBQXNCLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQyxDQUFDO1FBQ2hFLE1BQU0sVUFBVSxHQUFHLG1CQUFtQixDQUFDLGdCQUFnQixDQUFDLENBQUM7UUFDekQsb0ZBQW9GO1FBQ3BGLHdDQUF3QztRQUN4QywwRkFBMEY7UUFDMUYsT0FBTyxVQUFVLENBQUMsVUFBVTtZQUFFLFVBQVUsQ0FBQyxXQUFXLENBQUMsVUFBVSxDQUFDLFVBQVUsQ0FBQyxDQUFDO0tBQzdFO0FBQ0gsQ0FBQztBQUVELE1BQU0sVUFBVSxpQ0FBaUMsQ0FDL0MsT0FBVSxFQUNWLEtBQVU7SUFFVixNQUFNLGdCQUFnQixHQUFHLHNCQUFzQixDQUFDLE9BQU8sRUFBRSxLQUFLLENBQUMsQ0FBQztJQUVoRSxJQUFJLENBQUMsZ0JBQWdCLEVBQUU7UUFDckIsTUFBTSxLQUFLLENBQ1QscUNBQ0UsS0FBSyxDQUFDLElBQUksSUFBSSxLQUFLLENBQUMsT0FDdEIsa0VBQWtFLENBQ25FLENBQUM7S0FDSDtJQUVELE1BQU0sZ0JBQWdCLEdBQUcsbUJBQW1CLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztJQUMvRCxnQkFBZ0IsQ0FBQyxTQUFTLEdBQUcsT0FBTyxDQUFDLFFBQVEsQ0FBQztJQUM5QyxPQUFPLGdCQUFnQixDQUFDO0FBQzFCLENBQUM7QUFFRCxTQUFTLG1CQUFtQixDQUFDLGdCQUFrQztJQUM3RCxNQUFNLE9BQU8sR0FBRyxnQkFBZ0IsRUFBRSxDQUFDO0lBRW5DLElBQUksQ0FBQyxPQUFPLEVBQUU7UUFDWixNQUFNLEtBQUssQ0FBQyxxREFBcUQsQ0FBQyxDQUFDO0tBQ3BFO0lBRUQsT0FBTyxPQUFPLENBQUM7QUFDakIsQ0FBQztBQUVELFNBQVMsc0JBQXNCLENBQzdCLElBQU8sRUFDUCxLQUFVOztJQUVWLEtBQUssU0FBRyxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsV0FBVyxtQ0FBSSxLQUFLLENBQUM7SUFFcEMsSUFBSSxLQUFLLENBQUMsVUFBVSxFQUFFO1FBQ3BCLE9BQU8sR0FBRyxFQUFFLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQztLQUMvQjtTQUFNLElBQUksS0FBSyxDQUFDLGdCQUFnQixFQUFFO1FBQ2pDLE9BQU8sS0FBSyxDQUFDLGdCQUFnQixDQUFDO0tBQy9CO1NBQU0sSUFBSSxJQUFJLENBQUMsZ0JBQWdCLEVBQUU7UUFDaEMsT0FBTyxJQUFJLENBQUMsZ0JBQWdCLENBQUM7S0FDOUI7U0FBTTtRQUNMLE9BQU8sdUJBQXVCLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDO0tBQzVDO0FBQ0gsQ0FBQztBQUVELFNBQVMsdUJBQXVCLENBQUMsSUFBWTtJQUMzQyxPQUFPLFNBQVMsb0JBQW9CO1FBQ2xDLE1BQU0sRUFBRSxHQUFHLDBCQUEwQixJQUFJLEVBQUUsQ0FBQztRQUM1QyxJQUFJLFVBQVUsR0FBdUIsUUFBUSxDQUFDLGNBQWMsQ0FBQyxFQUFFLENBQUMsQ0FBQztRQUVqRSxJQUFJLENBQUMsVUFBVSxFQUFFO1lBQ2YsVUFBVSxHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDM0MsVUFBVSxDQUFDLEVBQUUsR0FBRyxFQUFFLENBQUM7WUFDbkIsUUFBUSxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsVUFBVSxDQUFDLENBQUM7U0FDdkM7UUFFRCxPQUFPLFVBQVUsQ0FBQztJQUNwQixDQUFDLENBQUM7QUFDSixDQUFDO0FBRUQsU0FBUyxVQUFVO0lBQ2pCLElBQUk7UUFDRiw2REFBNkQ7UUFDN0QsZ0ZBQWdGO1FBQ2hGLDhFQUE4RTtRQUM5RSxpREFBaUQ7UUFDakQsMkJBQTJCO1FBQzNCLE1BQU0sRUFBRSxXQUFXLEVBQUUsR0FBRyxPQUFPLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDakQsT0FBTyxDQUFDLENBQUMsV0FBVyxDQUFDO0tBQ3RCO0lBQUMsV0FBTTtRQUNOLE9BQU8sS0FBSyxDQUFDO0tBQ2Q7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRG9tRWxlbWVudEdldHRlciwgQmFzZVNpbmdsZVNwYUFuZ3VsYXJPcHRpb25zIH0gZnJvbSAnLi90eXBlcyc7XG5cbmV4cG9ydCBmdW5jdGlvbiByZW1vdmVBcHBsaWNhdGlvbkZyb21ET01JZkl2eUVuYWJsZWQ8VCBleHRlbmRzIEJhc2VTaW5nbGVTcGFBbmd1bGFyT3B0aW9ucz4oXG4gIG9wdGlvbnM6IFQsXG4gIHByb3BzOiBhbnksXG4pOiB2b2lkIHtcbiAgaWYgKGl2eUVuYWJsZWQoKSkge1xuICAgIGNvbnN0IGRvbUVsZW1lbnRHZXR0ZXIgPSBjaG9vc2VEb21FbGVtZW50R2V0dGVyKG9wdGlvbnMsIHByb3BzKTtcbiAgICBjb25zdCBkb21FbGVtZW50ID0gZ2V0Q29udGFpbmVyRWxlbWVudChkb21FbGVtZW50R2V0dGVyKTtcbiAgICAvLyBWaWV3IEVuZ2luZSByZW1vdmVzIGFsbCBub2RlcyBhdXRvbWF0aWNhbGx5IHdoZW4gY2FsbGluZyBgTmdNb2R1bGVSZWYuZGVzdHJveSgpYCxcbiAgICAvLyB3aGljaCBjYWxscyBgQ29tcG9uZW50UmVmLmRlc3Ryb3koKWAuXG4gICAgLy8gQmFzaWNhbGx5IHRoaXMgd2lsbCByZW1vdmUgYGFwcC1yb290YCBvciBhbnkgb3RoZXIgc2VsZWN0b3IgZnJvbSB0aGUgY29udGFpbmVyIGVsZW1lbnQuXG4gICAgd2hpbGUgKGRvbUVsZW1lbnQuZmlyc3RDaGlsZCkgZG9tRWxlbWVudC5yZW1vdmVDaGlsZChkb21FbGVtZW50LmZpcnN0Q2hpbGQpO1xuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRDb250YWluZXJFbGVtZW50QW5kU2V0VGVtcGxhdGU8VCBleHRlbmRzIEJhc2VTaW5nbGVTcGFBbmd1bGFyT3B0aW9ucz4oXG4gIG9wdGlvbnM6IFQsXG4gIHByb3BzOiBhbnksXG4pOiBIVE1MRWxlbWVudCB7XG4gIGNvbnN0IGRvbUVsZW1lbnRHZXR0ZXIgPSBjaG9vc2VEb21FbGVtZW50R2V0dGVyKG9wdGlvbnMsIHByb3BzKTtcblxuICBpZiAoIWRvbUVsZW1lbnRHZXR0ZXIpIHtcbiAgICB0aHJvdyBFcnJvcihcbiAgICAgIGBDYW5ub3QgbW91bnQgYW5ndWxhciBhcHBsaWNhdGlvbiAnJHtcbiAgICAgICAgcHJvcHMubmFtZSB8fCBwcm9wcy5hcHBOYW1lXG4gICAgICB9JyB3aXRob3V0IGEgZG9tRWxlbWVudEdldHRlciBwcm92aWRlZCBlaXRoZXIgYXMgYW4gb3B0IG9yIGEgcHJvcGAsXG4gICAgKTtcbiAgfVxuXG4gIGNvbnN0IGNvbnRhaW5lckVsZW1lbnQgPSBnZXRDb250YWluZXJFbGVtZW50KGRvbUVsZW1lbnRHZXR0ZXIpO1xuICBjb250YWluZXJFbGVtZW50LmlubmVySFRNTCA9IG9wdGlvbnMudGVtcGxhdGU7XG4gIHJldHVybiBjb250YWluZXJFbGVtZW50O1xufVxuXG5mdW5jdGlvbiBnZXRDb250YWluZXJFbGVtZW50KGRvbUVsZW1lbnRHZXR0ZXI6IERvbUVsZW1lbnRHZXR0ZXIpOiBuZXZlciB8IEhUTUxFbGVtZW50IHtcbiAgY29uc3QgZWxlbWVudCA9IGRvbUVsZW1lbnRHZXR0ZXIoKTtcblxuICBpZiAoIWVsZW1lbnQpIHtcbiAgICB0aHJvdyBFcnJvcignZG9tRWxlbWVudEdldHRlciBkaWQgbm90IHJldHVybiBhIHZhbGlkIGRvbSBlbGVtZW50Jyk7XG4gIH1cblxuICByZXR1cm4gZWxlbWVudDtcbn1cblxuZnVuY3Rpb24gY2hvb3NlRG9tRWxlbWVudEdldHRlcjxUIGV4dGVuZHMgQmFzZVNpbmdsZVNwYUFuZ3VsYXJPcHRpb25zPihcbiAgb3B0czogVCxcbiAgcHJvcHM6IGFueSxcbik6IERvbUVsZW1lbnRHZXR0ZXIge1xuICBwcm9wcyA9IHByb3BzPy5jdXN0b21Qcm9wcyA/PyBwcm9wcztcblxuICBpZiAocHJvcHMuZG9tRWxlbWVudCkge1xuICAgIHJldHVybiAoKSA9PiBwcm9wcy5kb21FbGVtZW50O1xuICB9IGVsc2UgaWYgKHByb3BzLmRvbUVsZW1lbnRHZXR0ZXIpIHtcbiAgICByZXR1cm4gcHJvcHMuZG9tRWxlbWVudEdldHRlcjtcbiAgfSBlbHNlIGlmIChvcHRzLmRvbUVsZW1lbnRHZXR0ZXIpIHtcbiAgICByZXR1cm4gb3B0cy5kb21FbGVtZW50R2V0dGVyO1xuICB9IGVsc2Uge1xuICAgIHJldHVybiBkZWZhdWx0RG9tRWxlbWVudEdldHRlcihwcm9wcy5uYW1lKTtcbiAgfVxufVxuXG5mdW5jdGlvbiBkZWZhdWx0RG9tRWxlbWVudEdldHRlcihuYW1lOiBzdHJpbmcpOiBEb21FbGVtZW50R2V0dGVyIHtcbiAgcmV0dXJuIGZ1bmN0aW9uIGdldERlZmF1bHREb21FbGVtZW50KCkge1xuICAgIGNvbnN0IGlkID0gYHNpbmdsZS1zcGEtYXBwbGljYXRpb246JHtuYW1lfWA7XG4gICAgbGV0IGRvbUVsZW1lbnQ6IEhUTUxFbGVtZW50IHwgbnVsbCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKGlkKTtcblxuICAgIGlmICghZG9tRWxlbWVudCkge1xuICAgICAgZG9tRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ2RpdicpO1xuICAgICAgZG9tRWxlbWVudC5pZCA9IGlkO1xuICAgICAgZG9jdW1lbnQuYm9keS5hcHBlbmRDaGlsZChkb21FbGVtZW50KTtcbiAgICB9XG5cbiAgICByZXR1cm4gZG9tRWxlbWVudDtcbiAgfTtcbn1cblxuZnVuY3Rpb24gaXZ5RW5hYmxlZCgpOiBib29sZWFuIHtcbiAgdHJ5IHtcbiAgICAvLyBgybVpdnlFbmFibGVkYCB2YXJpYWJsZSBpcyBleHBvc2VkIHN0YXJ0aW5nIGZyb20gdmVyc2lvbiA4LlxuICAgIC8vIFdlIHVzZSBgcmVxdWlyZWAgaGVyZSBleGNlcHQgb2YgYSBzaW5nbGUgYGltcG9ydCB7IMm1aXZ5RW5hYmxlZCB9YCBiZWNhdXNlIHRoZVxuICAgIC8vIGRldmVsb3BlciBjYW4gdXNlIEFuZ3VsYXIgdmVyc2lvbiB0aGF0IGRvZXNuJ3QgZXhwb3NlIGl0IChhbGwgdmVyc2lvbnMgPDgpLlxuICAgIC8vIFRoZSBgY2F0Y2hgIHN0YXRlbWVudCB3aWxsIGhhbmRsZSB0aG9zZSBjYXNlcy5cbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmVcbiAgICBjb25zdCB7IMm1aXZ5RW5hYmxlZCB9ID0gcmVxdWlyZSgnQGFuZ3VsYXIvY29yZScpO1xuICAgIHJldHVybiAhIcm1aXZ5RW5hYmxlZDtcbiAgfSBjYXRjaCB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG59XG4iXX0= |
@@ -1,2 +0,3 @@ | ||
export { getContainerEl, chooseDomElementGetter, removeApplicationFromDOMIfIvyEnabled, } from './dom'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290Ijoibmc6Ly9zaW5nbGUtc3BhLWFuZ3VsYXIvaW50ZXJuYWxzLyIsInNvdXJjZXMiOlsiaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUNMLGNBQWMsRUFDZCxzQkFBc0IsRUFDdEIsb0NBQW9DLEdBQ3JDLE1BQU0sT0FBTyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgU2luZ2xlU3BhQW5ndWxhck9wdHMsIEJvb3RzdHJhcHBlZFNpbmdsZVNwYUFuZ3VsYXJPcHRzIH0gZnJvbSAnLi90eXBlcyc7XG5leHBvcnQge1xuICBnZXRDb250YWluZXJFbCxcbiAgY2hvb3NlRG9tRWxlbWVudEdldHRlcixcbiAgcmVtb3ZlQXBwbGljYXRpb25Gcm9tRE9NSWZJdnlFbmFibGVkLFxufSBmcm9tICcuL2RvbSc7XG4iXX0= | ||
export * from './types'; | ||
export { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled } from './dom'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaW50ZXJuYWxzL3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLFNBQVMsQ0FBQztBQUN4QixPQUFPLEVBQUUsaUNBQWlDLEVBQUUsb0NBQW9DLEVBQUUsTUFBTSxPQUFPLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbmV4cG9ydCB7IGdldENvbnRhaW5lckVsZW1lbnRBbmRTZXRUZW1wbGF0ZSwgcmVtb3ZlQXBwbGljYXRpb25Gcm9tRE9NSWZJdnlFbmFibGVkIH0gZnJvbSAnLi9kb20nO1xuIl19 |
@@ -5,2 +5,2 @@ /** | ||
export * from './index'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXNwYS1hbmd1bGFyLWludGVybmFscy5qcyIsInNvdXJjZVJvb3QiOiJuZzovL3NpbmdsZS1zcGEtYW5ndWxhci9pbnRlcm5hbHMvIiwic291cmNlcyI6WyJzaW5nbGUtc3BhLWFuZ3VsYXItaW50ZXJuYWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vaW5kZXgnO1xuIl19 | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXNwYS1hbmd1bGFyLWludGVybmFscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9pbnRlcm5hbHMvc3JjL3NpbmdsZS1zcGEtYW5ndWxhci1pbnRlcm5hbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLFNBQVMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9pbmRleCc7XG4iXX0= |
@@ -1,1 +0,1 @@ | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290Ijoibmc6Ly9zaW5nbGUtc3BhLWFuZ3VsYXIvaW50ZXJuYWxzLyIsInNvdXJjZXMiOlsidHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlUmVmLCBUeXBlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBBcHBQcm9wcyB9IGZyb20gJ3NpbmdsZS1zcGEnO1xuXG5leHBvcnQgdHlwZSBEb21FbGVtZW50R2V0dGVyID0gKCkgPT4gSFRNTEVsZW1lbnQ7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2luZ2xlU3BhQW5ndWxhck9wdHMge1xuICAvLyBUaGlzIG1pZ2h0IGJlIGBub29wYCBpZiB0aGUgcm9vdCBtb2R1bGUgaXMgYm9vdHN0cmFwcGVkXG4gIC8vIHdpdGggYHsgbmdab25lOiAnbm9vcCcgfWAgb3B0aW9ucy5cbiAgTmdab25lOiB0eXBlb2YgaW1wb3J0KCdAYW5ndWxhci9jb3JlJykuTmdab25lIHwgJ25vb3AnO1xuICBib290c3RyYXBGdW5jdGlvbihwcm9wczogQXBwUHJvcHMpOiBQcm9taXNlPE5nTW9kdWxlUmVmPGFueT4+O1xuICB1cGRhdGVGdW5jdGlvbj8ocHJvcHM6IEFwcFByb3BzKTogUHJvbWlzZTxhbnk+O1xuICB0ZW1wbGF0ZTogc3RyaW5nO1xuICBSb3V0ZXI/OiBUeXBlPGFueT47XG4gIGRvbUVsZW1lbnRHZXR0ZXI/KCk6IEhUTUxFbGVtZW50O1xuICBBbmltYXRpb25FbmdpbmU/OiBUeXBlPGFueT47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQm9vdHN0cmFwcGVkU2luZ2xlU3BhQW5ndWxhck9wdHMgZXh0ZW5kcyBTaW5nbGVTcGFBbmd1bGFyT3B0cyB7XG4gIGJvb3RzdHJhcHBlZE1vZHVsZTogTmdNb2R1bGVSZWY8YW55PjtcbiAgLy8gQWxsIGJlbG93IHByb3BlcnRpZXMgY2FuIGJlIG9wdGlvbmFsIGluIGNhc2Ugb2ZcbiAgLy8gYFNpbmdsZVNwYUFuZ3VsYXJPcHRzLk5nWm9uZWAgaXMgYSBgbm9vcGAgc3RyaW5nIGFuZCBub3QgYW4gYE5nWm9uZWAgY2xhc3MuXG4gIGJvb3RzdHJhcHBlZE5nWm9uZT86IGltcG9ydCgnQGFuZ3VsYXIvY29yZScpLk5nWm9uZTtcbiAgcm91dGluZ0V2ZW50TGlzdGVuZXI/OiAoKSA9PiB2b2lkO1xuICB6b25lSWRlbnRpZmllcj86IHN0cmluZztcbn1cbiJdfQ== | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaW50ZXJuYWxzL3NyYy90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQXBwUHJvcHMgfSBmcm9tICdzaW5nbGUtc3BhJztcbmltcG9ydCB7IE5nTW9kdWxlUmVmIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbmV4cG9ydCB0eXBlIERvbUVsZW1lbnRHZXR0ZXIgPSAoKSA9PiBIVE1MRWxlbWVudDtcblxuZXhwb3J0IGludGVyZmFjZSBCYXNlU2luZ2xlU3BhQW5ndWxhck9wdGlvbnMge1xuICB0ZW1wbGF0ZTogc3RyaW5nO1xuICBkb21FbGVtZW50R2V0dGVyPygpOiBIVE1MRWxlbWVudDtcbiAgYm9vdHN0cmFwRnVuY3Rpb24ocHJvcHM6IEFwcFByb3BzKTogUHJvbWlzZTxOZ01vZHVsZVJlZjxhbnk+Pjtcbn1cbiJdfQ== |
@@ -6,2 +6,2 @@ /** | ||
export { SingleSpaPlatformLocation as ɵa } from './src/extra-providers'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXNwYS1hbmd1bGFyLmpzIiwic291cmNlUm9vdCI6Im5nOi8vc2luZ2xlLXNwYS1hbmd1bGFyLyIsInNvdXJjZXMiOlsic2luZ2xlLXNwYS1hbmd1bGFyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxTQUFTLENBQUM7QUFFeEIsT0FBTyxFQUFDLHlCQUF5QixJQUFJLEVBQUUsRUFBQyxNQUFNLHVCQUF1QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2luZGV4JztcblxuZXhwb3J0IHtTaW5nbGVTcGFQbGF0Zm9ybUxvY2F0aW9uIGFzIMm1YX0gZnJvbSAnLi9zcmMvZXh0cmEtcHJvdmlkZXJzJzsiXX0= | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXNwYS1hbmd1bGFyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NpbmdsZS1zcGEtYW5ndWxhci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsU0FBUyxDQUFDO0FBRXhCLE9BQU8sRUFBQyx5QkFBeUIsSUFBSSxFQUFFLEVBQUMsTUFBTSx1QkFBdUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9pbmRleCc7XG5cbmV4cG9ydCB7U2luZ2xlU3BhUGxhdGZvcm1Mb2NhdGlvbiBhcyDJtWF9IGZyb20gJy4vc3JjL2V4dHJhLXByb3ZpZGVycyc7Il19 |
@@ -1,5 +0,4 @@ | ||
import { __decorate } from "tslib"; | ||
import { Injectable, Inject } from '@angular/core'; | ||
import { ɵBrowserPlatformLocation, PlatformLocation, DOCUMENT, } from '@angular/common'; | ||
let SingleSpaPlatformLocation = class SingleSpaPlatformLocation extends ɵBrowserPlatformLocation { | ||
export class SingleSpaPlatformLocation extends ɵBrowserPlatformLocation { | ||
constructor() { | ||
@@ -10,5 +9,21 @@ super(...arguments); | ||
this.skipNextPopState = false; | ||
this.onPopStateListeners = []; | ||
// The key here is an actual forked `Zone` of some specific application. | ||
// We will be able to find the specific zone when application gets destroyed | ||
// by application `name`. | ||
// The reason of that the `onPopState` method is invoked during `bootstrapModule` | ||
// and we can't know what application has invoked it. Why should we know the application | ||
// that has invoked `onPopState`? When application gets destroyed in a `sharing dependencies mode` | ||
// (when there is a single platform per all applications) we want to remove application | ||
// specific `popstate` listeners. E.g. if there are 2 applications: | ||
// * shop application adds `popstate` listener | ||
// * navbar application adds `popstate` listener | ||
// When shop application gets destroyed we want to remove only its `popstate` listener. | ||
this.zoneToOnPopStateListenersMap = new Map(); | ||
// This is used only to make `Zone.wrap` happy, since it requires 2 arguments | ||
// and the second argument is a unique string which `zone.js` uses for debugging purposes. | ||
// We might want to use the application name, but we're not able to get it when `onPopState` | ||
// method is called during module bootstrapping. | ||
this.source = 0; | ||
} | ||
destroy() { | ||
destroyApplication(zoneIdentifier) { | ||
// TLDR: Angular adds `popstate` event listener and then doesn't remove it when application gets destroyed. | ||
@@ -18,10 +33,15 @@ // Basically, Angular has a potentional memory leak. The `ɵBrowserPlatformLocation` | ||
// https://github.com/angular/angular/blob/14be55c9facf3e47b8c97df4502dc3f0f897da03/packages/common/src/location/platform_location.ts#L126 | ||
for (const onPopStateListener of this.onPopStateListeners) { | ||
window.removeEventListener('popstate', onPopStateListener); | ||
const zone = [...this.zoneToOnPopStateListenersMap.keys()].find( | ||
// `getZoneWith` will return a zone which defines a `key` and in our case | ||
// we define a custom key in `single-spa-angular.ts` | ||
// via this line of code: | ||
// `_properties[zoneIdentifier] = true;` | ||
zone => zone.getZoneWith(zoneIdentifier) !== null); | ||
const onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone); | ||
if (Array.isArray(onPopStateListeners)) { | ||
for (const onPopStateListener of onPopStateListeners) { | ||
window.removeEventListener('popstate', onPopStateListener); | ||
} | ||
} | ||
// We do this because the `SingleSpaPlatformLocation` is a part of PLATFORM_INJECTOR, | ||
// which means it's created only once and will not be garbage collected, since the PLATFORM_INJECTOR | ||
// will keep reference to its instance. | ||
// TODO: https://github.com/single-spa/single-spa-angular/issues/170 | ||
this.onPopStateListeners = []; | ||
this.zoneToOnPopStateListenersMap.delete(zone); | ||
} | ||
@@ -37,2 +57,12 @@ pushState(state, title, url) { | ||
onPopState(fn) { | ||
// `Zone.current` will reference the zone that serves as an execution context | ||
// to some specific application, especially when `onPopState` is called. | ||
const zone = Zone.current; | ||
// Wrap any event listener into zone that is specific to some application. | ||
// The main issue is `back/forward` buttons of browsers, because they invoke | ||
// `history.back|forward` which dispatch `popstate` event. Since `single-spa` | ||
// overrides `history.replaceState` Angular's zone cannot intercept this event. | ||
// Only the root zone is able to intercept all events. | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
fn = zone.wrap(fn, `${this.source++}`); | ||
const onPopStateListener = (event) => { | ||
@@ -48,24 +78,21 @@ // The `LocationChangeEvent` doesn't have the `singleSpa` property, since it's added | ||
else { | ||
// Wrap any event listener into zone that is specific to some application. | ||
// The main issue is `back/forward` buttons of browsers, because they invoke | ||
// `history.back|forward` which dispatch `popstate` event. Since `single-spa` | ||
// overrides `history.replaceState` Angular's zone cannot intercept this event. | ||
// Only the root zone is able to intercept all events. | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
this.ngZone.run(() => fn(event)); | ||
fn(event); | ||
} | ||
}; | ||
this.storeOnPopStateListener(zone, onPopStateListener); | ||
super.onPopState(onPopStateListener); | ||
} | ||
storeOnPopStateListener(zone, onPopStateListener) { | ||
// All listeners should be stored inside an array because the `onPopState` can be called | ||
// multiple times thus we wanna reference all listeners to remove them further. | ||
this.onPopStateListeners.push(onPopStateListener); | ||
super.onPopState(onPopStateListener); | ||
const onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone) || []; | ||
onPopStateListeners.push(onPopStateListener); | ||
if (!this.zoneToOnPopStateListenersMap.has(zone)) { | ||
this.zoneToOnPopStateListenersMap.set(zone, onPopStateListeners); | ||
} | ||
} | ||
setNgZone(ngZone) { | ||
this.ngZone = ngZone; | ||
} | ||
}; | ||
SingleSpaPlatformLocation = __decorate([ | ||
Injectable() | ||
], SingleSpaPlatformLocation); | ||
export { SingleSpaPlatformLocation }; | ||
} | ||
SingleSpaPlatformLocation.decorators = [ | ||
{ type: Injectable } | ||
]; | ||
/** | ||
@@ -89,2 +116,2 @@ * The `PlatformLocation` class is an "injectee" of the `PathLocationStrategy`, | ||
} | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXh0cmEtcHJvdmlkZXJzLmpzIiwic291cmNlUm9vdCI6Im5nOi8vc2luZ2xlLXNwYS1hbmd1bGFyLyIsInNvdXJjZXMiOlsic3JjL2V4dHJhLXByb3ZpZGVycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxFQUFFLFVBQVUsRUFBMEIsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNFLE9BQU8sRUFDTCx3QkFBd0IsRUFDeEIsZ0JBQWdCLEVBRWhCLFFBQVEsR0FDVCxNQUFNLGlCQUFpQixDQUFDO0FBR3pCLElBQWEseUJBQXlCLEdBQXRDLE1BQWEseUJBQTBCLFNBQVEsd0JBQXdCO0lBQXZFOztRQUdFLGlFQUFpRTtRQUNqRSwwQ0FBMEM7UUFDbEMscUJBQWdCLEdBQUcsS0FBSyxDQUFDO1FBRXpCLHdCQUFtQixHQUE2QyxFQUFFLENBQUM7SUEyRDdFLENBQUM7SUF6REMsT0FBTztRQUNMLDJHQUEyRztRQUMzRyxtRkFBbUY7UUFDbkYsc0ZBQXNGO1FBQ3RGLDBJQUEwSTtRQUUxSSxLQUFLLE1BQU0sa0JBQWtCLElBQUksSUFBSSxDQUFDLG1CQUFtQixFQUFFO1lBQ3pELE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxVQUFVLEVBQUUsa0JBQWtCLENBQUMsQ0FBQztTQUM1RDtRQUVELHFGQUFxRjtRQUNyRixvR0FBb0c7UUFDcEcsdUNBQXVDO1FBQ3ZDLG9FQUFvRTtRQUNwRSxJQUFJLENBQUMsbUJBQW1CLEdBQUcsRUFBRSxDQUFDO0lBQ2hDLENBQUM7SUFFRCxTQUFTLENBQUMsS0FBVSxFQUFFLEtBQWEsRUFBRSxHQUFXO1FBQzlDLElBQUksQ0FBQyxnQkFBZ0IsR0FBRyxJQUFJLENBQUM7UUFDN0IsS0FBSyxDQUFDLFNBQVMsQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ3JDLENBQUM7SUFFRCxZQUFZLENBQUMsS0FBVSxFQUFFLEtBQWEsRUFBRSxHQUFXO1FBQ2pELElBQUksQ0FBQyxnQkFBZ0IsR0FBRyxJQUFJLENBQUM7UUFDN0IsS0FBSyxDQUFDLFlBQVksQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ3hDLENBQUM7SUFFRCxVQUFVLENBQUMsRUFBd0M7UUFDakQsTUFBTSxrQkFBa0IsR0FBRyxDQUFDLEtBQTBCLEVBQUUsRUFBRTtZQUN4RCxvRkFBb0Y7WUFDcEYsa0ZBQWtGO1lBQ2xGLHNFQUFzRTtZQUN0RSxNQUFNLHFDQUFxQyxHQUFHLENBQUMsQ0FBRyxLQUE0QztpQkFDM0YsU0FBUyxDQUFDO1lBRWIsSUFBSSxJQUFJLENBQUMsZ0JBQWdCLElBQUkscUNBQXFDLEVBQUU7Z0JBQ2xFLElBQUksQ0FBQyxnQkFBZ0IsR0FBRyxLQUFLLENBQUM7YUFDL0I7aUJBQU07Z0JBQ0wsMEVBQTBFO2dCQUMxRSw0RUFBNEU7Z0JBQzVFLDZFQUE2RTtnQkFDN0UsK0VBQStFO2dCQUMvRSxzREFBc0Q7Z0JBQ3RELGtGQUFrRjtnQkFDbEYsSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLENBQUMsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7YUFDbEM7UUFDSCxDQUFDLENBQUM7UUFFRix3RkFBd0Y7UUFDeEYsK0VBQStFO1FBQy9FLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUMsQ0FBQztRQUNsRCxLQUFLLENBQUMsVUFBVSxDQUFDLGtCQUFrQixDQUFDLENBQUM7SUFDdkMsQ0FBQztJQUVELFNBQVMsQ0FBQyxNQUFjO1FBQ3RCLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFDO0lBQ3ZCLENBQUM7Q0FDRixDQUFBO0FBbEVZLHlCQUF5QjtJQURyQyxVQUFVLEVBQUU7R0FDQSx5QkFBeUIsQ0FrRXJDO1NBbEVZLHlCQUF5QjtBQW9FdEM7Ozs7R0FJRztBQUNILE1BQU0sVUFBVSwwQkFBMEI7SUFDeEMsT0FBTztRQUNMO1lBQ0UsT0FBTyxFQUFFLHlCQUF5QjtZQUNsQyxRQUFRLEVBQUUseUJBQXlCO1lBQ25DLElBQUksRUFBRSxDQUFDLENBQUMsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQztTQUMvQjtRQUNEO1lBQ0UsT0FBTyxFQUFFLGdCQUFnQjtZQUN6QixXQUFXLEVBQUUseUJBQXlCO1NBQ3ZDO0tBQ0YsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlLCBOZ1pvbmUsIFN0YXRpY1Byb3ZpZGVyLCBJbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7XG4gIMm1QnJvd3NlclBsYXRmb3JtTG9jYXRpb24sXG4gIFBsYXRmb3JtTG9jYXRpb24sXG4gIExvY2F0aW9uQ2hhbmdlRXZlbnQsXG4gIERPQ1VNRU5ULFxufSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuXG5ASW5qZWN0YWJsZSgpXG5leHBvcnQgY2xhc3MgU2luZ2xlU3BhUGxhdGZvcm1Mb2NhdGlvbiBleHRlbmRzIMm1QnJvd3NlclBsYXRmb3JtTG9jYXRpb24ge1xuICBwcml2YXRlIG5nWm9uZSE6IE5nWm9uZTtcblxuICAvLyBUaGlzIGlzIGEgc2ltcGxlIG1hcmtlciB0aGF0IGhlbHBzIHVzIHRvIGlnbm9yZSBQb3BTdGF0ZUV2ZW50c1xuICAvLyB0aGF0IHdhcyBub3QgZGlzcGF0Y2hlZCBieSB0aGUgYnJvd3Nlci5cbiAgcHJpdmF0ZSBza2lwTmV4dFBvcFN0YXRlID0gZmFsc2U7XG5cbiAgcHJpdmF0ZSBvblBvcFN0YXRlTGlzdGVuZXJzOiAoKGV2ZW50OiBMb2NhdGlvbkNoYW5nZUV2ZW50KSA9PiB2b2lkKVtdID0gW107XG5cbiAgZGVzdHJveSgpOiB2b2lkIHtcbiAgICAvLyBUTERSOiBBbmd1bGFyIGFkZHMgYHBvcHN0YXRlYCBldmVudCBsaXN0ZW5lciBhbmQgdGhlbiBkb2Vzbid0IHJlbW92ZSBpdCB3aGVuIGFwcGxpY2F0aW9uIGdldHMgZGVzdHJveWVkLlxuICAgIC8vIEJhc2ljYWxseSwgQW5ndWxhciBoYXMgYSBwb3RlbnRpb25hbCBtZW1vcnkgbGVhay4gVGhlIGDJtUJyb3dzZXJQbGF0Zm9ybUxvY2F0aW9uYFxuICAgIC8vIGhhcyBgb25Qb3BTdGF0ZWAgbWV0aG9kIHdoaWNoIGFkZHMgYHBvcHN0YXRlYCBldmVudCBsaXN0ZW5lciBhbmQgZm9yZ2V0cywgc2VlIGhlcmU6XG4gICAgLy8gaHR0cHM6Ly9naXRodWIuY29tL2FuZ3VsYXIvYW5ndWxhci9ibG9iLzE0YmU1NWM5ZmFjZjNlNDdiOGM5N2RmNDUwMmRjM2YwZjg5N2RhMDMvcGFja2FnZXMvY29tbW9uL3NyYy9sb2NhdGlvbi9wbGF0Zm9ybV9sb2NhdGlvbi50cyNMMTI2XG5cbiAgICBmb3IgKGNvbnN0IG9uUG9wU3RhdGVMaXN0ZW5lciBvZiB0aGlzLm9uUG9wU3RhdGVMaXN0ZW5lcnMpIHtcbiAgICAgIHdpbmRvdy5yZW1vdmVFdmVudExpc3RlbmVyKCdwb3BzdGF0ZScsIG9uUG9wU3RhdGVMaXN0ZW5lcik7XG4gICAgfVxuXG4gICAgLy8gV2UgZG8gdGhpcyBiZWNhdXNlIHRoZSBgU2luZ2xlU3BhUGxhdGZvcm1Mb2NhdGlvbmAgaXMgYSBwYXJ0IG9mIFBMQVRGT1JNX0lOSkVDVE9SLFxuICAgIC8vIHdoaWNoIG1lYW5zIGl0J3MgY3JlYXRlZCBvbmx5IG9uY2UgYW5kIHdpbGwgbm90IGJlIGdhcmJhZ2UgY29sbGVjdGVkLCBzaW5jZSB0aGUgUExBVEZPUk1fSU5KRUNUT1JcbiAgICAvLyB3aWxsIGtlZXAgcmVmZXJlbmNlIHRvIGl0cyBpbnN0YW5jZS5cbiAgICAvLyBUT0RPOiBodHRwczovL2dpdGh1Yi5jb20vc2luZ2xlLXNwYS9zaW5nbGUtc3BhLWFuZ3VsYXIvaXNzdWVzLzE3MFxuICAgIHRoaXMub25Qb3BTdGF0ZUxpc3RlbmVycyA9IFtdO1xuICB9XG5cbiAgcHVzaFN0YXRlKHN0YXRlOiBhbnksIHRpdGxlOiBzdHJpbmcsIHVybDogc3RyaW5nKTogdm9pZCB7XG4gICAgdGhpcy5za2lwTmV4dFBvcFN0YXRlID0gdHJ1ZTtcbiAgICBzdXBlci5wdXNoU3RhdGUoc3RhdGUsIHRpdGxlLCB1cmwpO1xuICB9XG5cbiAgcmVwbGFjZVN0YXRlKHN0YXRlOiBhbnksIHRpdGxlOiBzdHJpbmcsIHVybDogc3RyaW5nKTogdm9pZCB7XG4gICAgdGhpcy5za2lwTmV4dFBvcFN0YXRlID0gdHJ1ZTtcbiAgICBzdXBlci5yZXBsYWNlU3RhdGUoc3RhdGUsIHRpdGxlLCB1cmwpO1xuICB9XG5cbiAgb25Qb3BTdGF0ZShmbjogKGV2ZW50OiBMb2NhdGlvbkNoYW5nZUV2ZW50KSA9PiB2b2lkKTogdm9pZCB7XG4gICAgY29uc3Qgb25Qb3BTdGF0ZUxpc3RlbmVyID0gKGV2ZW50OiBMb2NhdGlvbkNoYW5nZUV2ZW50KSA9PiB7XG4gICAgICAvLyBUaGUgYExvY2F0aW9uQ2hhbmdlRXZlbnRgIGRvZXNuJ3QgaGF2ZSB0aGUgYHNpbmdsZVNwYWAgcHJvcGVydHksIHNpbmNlIGl0J3MgYWRkZWRcbiAgICAgIC8vIGJ5IGBzaW5nbGUtc3BhYCBzdGFydGluZyBmcm9tIGA1LjRgIHZlcnNpb24uIFdlIG5lZWQgdGhpcyBjaGVjayBiZWNhdXNlIHdlIHdhbnRcbiAgICAgIC8vIHRvIHNraXAgXCJ1bm5hdHVyYWxcIiBQb3BTdGF0ZUV2ZW50cywgdGhlIG9uZSBjYXVzZWQgYnkgYHNpbmdsZS1zcGFgLlxuICAgICAgY29uc3QgcG9wU3RhdGVFdmVudFdhc0Rpc3BhdGNoZWRCeVNpbmdsZVNwYSA9ICEhKChldmVudCBhcyB1bmtub3duKSBhcyB7IHNpbmdsZVNwYTogYm9vbGVhbiB9KVxuICAgICAgICAuc2luZ2xlU3BhO1xuXG4gICAgICBpZiAodGhpcy5za2lwTmV4dFBvcFN0YXRlICYmIHBvcFN0YXRlRXZlbnRXYXNEaXNwYXRjaGVkQnlTaW5nbGVTcGEpIHtcbiAgICAgICAgdGhpcy5za2lwTmV4dFBvcFN0YXRlID0gZmFsc2U7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICAvLyBXcmFwIGFueSBldmVudCBsaXN0ZW5lciBpbnRvIHpvbmUgdGhhdCBpcyBzcGVjaWZpYyB0byBzb21lIGFwcGxpY2F0aW9uLlxuICAgICAgICAvLyBUaGUgbWFpbiBpc3N1ZSBpcyBgYmFjay9mb3J3YXJkYCBidXR0b25zIG9mIGJyb3dzZXJzLCBiZWNhdXNlIHRoZXkgaW52b2tlXG4gICAgICAgIC8vIGBoaXN0b3J5LmJhY2t8Zm9yd2FyZGAgd2hpY2ggZGlzcGF0Y2ggYHBvcHN0YXRlYCBldmVudC4gU2luY2UgYHNpbmdsZS1zcGFgXG4gICAgICAgIC8vIG92ZXJyaWRlcyBgaGlzdG9yeS5yZXBsYWNlU3RhdGVgIEFuZ3VsYXIncyB6b25lIGNhbm5vdCBpbnRlcmNlcHQgdGhpcyBldmVudC5cbiAgICAgICAgLy8gT25seSB0aGUgcm9vdCB6b25lIGlzIGFibGUgdG8gaW50ZXJjZXB0IGFsbCBldmVudHMuXG4gICAgICAgIC8vIFNlZSBodHRwczovL2dpdGh1Yi5jb20vc2luZ2xlLXNwYS9zaW5nbGUtc3BhLWFuZ3VsYXIvaXNzdWVzLzk0IGZvciBtb3JlIGRldGFpbHNcbiAgICAgICAgdGhpcy5uZ1pvbmUucnVuKCgpID0+IGZuKGV2ZW50KSk7XG4gICAgICB9XG4gICAgfTtcblxuICAgIC8vIEFsbCBsaXN0ZW5lcnMgc2hvdWxkIGJlIHN0b3JlZCBpbnNpZGUgYW4gYXJyYXkgYmVjYXVzZSB0aGUgYG9uUG9wU3RhdGVgIGNhbiBiZSBjYWxsZWRcbiAgICAvLyBtdWx0aXBsZSB0aW1lcyB0aHVzIHdlIHdhbm5hIHJlZmVyZW5jZSBhbGwgbGlzdGVuZXJzIHRvIHJlbW92ZSB0aGVtIGZ1cnRoZXIuXG4gICAgdGhpcy5vblBvcFN0YXRlTGlzdGVuZXJzLnB1c2gob25Qb3BTdGF0ZUxpc3RlbmVyKTtcbiAgICBzdXBlci5vblBvcFN0YXRlKG9uUG9wU3RhdGVMaXN0ZW5lcik7XG4gIH1cblxuICBzZXROZ1pvbmUobmdab25lOiBOZ1pvbmUpOiB2b2lkIHtcbiAgICB0aGlzLm5nWm9uZSA9IG5nWm9uZTtcbiAgfVxufVxuXG4vKipcbiAqIFRoZSBgUGxhdGZvcm1Mb2NhdGlvbmAgY2xhc3MgaXMgYW4gXCJpbmplY3RlZVwiIG9mIHRoZSBgUGF0aExvY2F0aW9uU3RyYXRlZ3lgLFxuICogd2hpY2ggY3JlYXRlcyBgU3ViamVjdGAgaW50ZXJuYWxseSBmb3IgbGlzdGVuaW5nIG9uIGBwb3BzdGF0ZWAgZXZlbnRzLiBXZSB3YW50XG4gKiB0byBwcm92aWRlIHRoaXMgY2xhc3MgaW4gdGhlIG1vc3QgdG9wIGluamVjdG9yIHRoYXQncyB1c2VkIGR1cmluZyBib290c3RyYXBwaW5nLlxuICovXG5leHBvcnQgZnVuY3Rpb24gZ2V0U2luZ2xlU3BhRXh0cmFQcm92aWRlcnMoKTogU3RhdGljUHJvdmlkZXJbXSB7XG4gIHJldHVybiBbXG4gICAge1xuICAgICAgcHJvdmlkZTogU2luZ2xlU3BhUGxhdGZvcm1Mb2NhdGlvbixcbiAgICAgIHVzZUNsYXNzOiBTaW5nbGVTcGFQbGF0Zm9ybUxvY2F0aW9uLFxuICAgICAgZGVwczogW1tuZXcgSW5qZWN0KERPQ1VNRU5UKV1dLFxuICAgIH0sXG4gICAge1xuICAgICAgcHJvdmlkZTogUGxhdGZvcm1Mb2NhdGlvbixcbiAgICAgIHVzZUV4aXN0aW5nOiBTaW5nbGVTcGFQbGF0Zm9ybUxvY2F0aW9uLFxuICAgIH0sXG4gIF07XG59XG4iXX0= | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extra-providers.js","sourceRoot":"","sources":["../../../src/src/extra-providers.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, StaticProvider, Inject } from '@angular/core';\nimport {\n  ɵBrowserPlatformLocation,\n  PlatformLocation,\n  LocationChangeEvent,\n  DOCUMENT,\n} from '@angular/common';\n\ntype OnPopStateListener = (event: LocationChangeEvent) => void;\n\ndeclare const Zone: any;\n\n@Injectable()\nexport class SingleSpaPlatformLocation extends ɵBrowserPlatformLocation {\n  // This is a simple marker that helps us to ignore PopStateEvents\n  // that was not dispatched by the browser.\n  private skipNextPopState = false;\n\n  // The key here is an actual forked `Zone` of some specific application.\n  // We will be able to find the specific zone when application gets destroyed\n  // by application `name`.\n  // The reason of that the `onPopState` method is invoked during `bootstrapModule`\n  // and we can't know what application has invoked it. Why should we know the application\n  // that has invoked `onPopState`? When application gets destroyed in a `sharing dependencies mode`\n  // (when there is a single platform per all applications) we want to remove application\n  // specific `popstate` listeners. E.g. if there are 2 applications:\n  // * shop application adds `popstate` listener\n  // * navbar application adds `popstate` listener\n  // When shop application gets destroyed we want to remove only its `popstate` listener.\n  private zoneToOnPopStateListenersMap = new Map<any, OnPopStateListener[]>();\n\n  // This is used only to make `Zone.wrap` happy, since it requires 2 arguments\n  // and the second argument is a unique string which `zone.js` uses for debugging purposes.\n  // We might want to use the application name, but we're not able to get it when `onPopState`\n  // method is called during module bootstrapping.\n  private source = 0;\n\n  destroyApplication(zoneIdentifier: string): void {\n    // TLDR: Angular adds `popstate` event listener and then doesn't remove it when application gets destroyed.\n    // Basically, Angular has a potentional memory leak. The `ɵBrowserPlatformLocation`\n    // has `onPopState` method which adds `popstate` event listener and forgets, see here:\n    // https://github.com/angular/angular/blob/14be55c9facf3e47b8c97df4502dc3f0f897da03/packages/common/src/location/platform_location.ts#L126\n    const zone = [...this.zoneToOnPopStateListenersMap.keys()].find(\n      // `getZoneWith` will return a zone which defines a `key` and in our case\n      // we define a custom key in `single-spa-angular.ts`\n      // via this line of code:\n      // `_properties[zoneIdentifier] = true;`\n      zone => zone.getZoneWith(zoneIdentifier) !== null,\n    );\n\n    const onPopStateListeners:\n      | OnPopStateListener[]\n      | undefined = this.zoneToOnPopStateListenersMap.get(zone);\n\n    if (Array.isArray(onPopStateListeners)) {\n      for (const onPopStateListener of onPopStateListeners) {\n        window.removeEventListener('popstate', onPopStateListener);\n      }\n    }\n\n    this.zoneToOnPopStateListenersMap.delete(zone);\n  }\n\n  pushState(state: any, title: string, url: string): void {\n    this.skipNextPopState = true;\n    super.pushState(state, title, url);\n  }\n\n  replaceState(state: any, title: string, url: string): void {\n    this.skipNextPopState = true;\n    super.replaceState(state, title, url);\n  }\n\n  onPopState(fn: OnPopStateListener): void {\n    // `Zone.current` will reference the zone that serves as an execution context\n    // to some specific application, especially when `onPopState` is called.\n    const zone = Zone.current;\n\n    // Wrap any event listener into zone that is specific to some application.\n    // The main issue is `back/forward` buttons of browsers, because they invoke\n    // `history.back|forward` which dispatch `popstate` event. Since `single-spa`\n    // overrides `history.replaceState` Angular's zone cannot intercept this event.\n    // Only the root zone is able to intercept all events.\n    // See https://github.com/single-spa/single-spa-angular/issues/94 for more details\n    fn = zone.wrap(fn, `${this.source++}`);\n\n    const onPopStateListener = (event: LocationChangeEvent) => {\n      // The `LocationChangeEvent` doesn't have the `singleSpa` property, since it's added\n      // by `single-spa` starting from `5.4` version. We need this check because we want\n      // to skip \"unnatural\" PopStateEvents, the one caused by `single-spa`.\n      const popStateEventWasDispatchedBySingleSpa = !!((event as unknown) as { singleSpa: boolean })\n        .singleSpa;\n\n      if (this.skipNextPopState && popStateEventWasDispatchedBySingleSpa) {\n        this.skipNextPopState = false;\n      } else {\n        fn(event);\n      }\n    };\n\n    this.storeOnPopStateListener(zone, onPopStateListener);\n    super.onPopState(onPopStateListener);\n  }\n\n  private storeOnPopStateListener(zone: any, onPopStateListener: OnPopStateListener): void {\n    // All listeners should be stored inside an array because the `onPopState` can be called\n    // multiple times thus we wanna reference all listeners to remove them further.\n    const onPopStateListeners: OnPopStateListener[] =\n      this.zoneToOnPopStateListenersMap.get(zone) || [];\n\n    onPopStateListeners.push(onPopStateListener);\n\n    if (!this.zoneToOnPopStateListenersMap.has(zone)) {\n      this.zoneToOnPopStateListenersMap.set(zone, onPopStateListeners);\n    }\n  }\n}\n\n/**\n * The `PlatformLocation` class is an \"injectee\" of the `PathLocationStrategy`,\n * which creates `Subject` internally for listening on `popstate` events. We want\n * to provide this class in the most top injector that's used during bootstrapping.\n */\nexport function getSingleSpaExtraProviders(): StaticProvider[] {\n  return [\n    {\n      provide: SingleSpaPlatformLocation,\n      useClass: SingleSpaPlatformLocation,\n      deps: [[new Inject(DOCUMENT)]],\n    },\n    {\n      provide: PlatformLocation,\n      useExisting: SingleSpaPlatformLocation,\n    },\n  ];\n}\n"]} |
@@ -1,14 +0,12 @@ | ||
import { __decorate } from "tslib"; | ||
import { NgModule } from '@angular/core'; | ||
import { ParcelComponent } from './parcel.component'; | ||
let ParcelModule = class ParcelModule { | ||
}; | ||
ParcelModule = __decorate([ | ||
NgModule({ | ||
declarations: [ParcelComponent], | ||
exports: [ParcelComponent], | ||
entryComponents: [ParcelComponent], | ||
}) | ||
], ParcelModule); | ||
export { ParcelModule }; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290Ijoibmc6Ly9zaW5nbGUtc3BhLWFuZ3VsYXIvIiwic291cmNlcyI6WyJzcmMvcGFyY2VsLWxpYi9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFPckQsSUFBYSxZQUFZLEdBQXpCLE1BQWEsWUFBWTtDQUFHLENBQUE7QUFBZixZQUFZO0lBTHhCLFFBQVEsQ0FBQztRQUNSLFlBQVksRUFBRSxDQUFDLGVBQWUsQ0FBQztRQUMvQixPQUFPLEVBQUUsQ0FBQyxlQUFlLENBQUM7UUFDMUIsZUFBZSxFQUFFLENBQUMsZUFBZSxDQUFDO0tBQ25DLENBQUM7R0FDVyxZQUFZLENBQUc7U0FBZixZQUFZIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFBhcmNlbENvbXBvbmVudCB9IGZyb20gJy4vcGFyY2VsLmNvbXBvbmVudCc7XG5cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1BhcmNlbENvbXBvbmVudF0sXG4gIGV4cG9ydHM6IFtQYXJjZWxDb21wb25lbnRdLFxuICBlbnRyeUNvbXBvbmVudHM6IFtQYXJjZWxDb21wb25lbnRdLFxufSlcbmV4cG9ydCBjbGFzcyBQYXJjZWxNb2R1bGUge31cbiJdfQ== | ||
export class ParcelModule { | ||
} | ||
ParcelModule.decorators = [ | ||
{ type: NgModule, args: [{ | ||
declarations: [ParcelComponent], | ||
exports: [ParcelComponent], | ||
entryComponents: [ParcelComponent], | ||
},] } | ||
]; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvc3JjL3BhcmNlbC1saWIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFPckQsTUFBTSxPQUFPLFlBQVk7OztZQUx4QixRQUFRLFNBQUM7Z0JBQ1IsWUFBWSxFQUFFLENBQUMsZUFBZSxDQUFDO2dCQUMvQixPQUFPLEVBQUUsQ0FBQyxlQUFlLENBQUM7Z0JBQzFCLGVBQWUsRUFBRSxDQUFDLGVBQWUsQ0FBQzthQUNuQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBQYXJjZWxDb21wb25lbnQgfSBmcm9tICcuL3BhcmNlbC5jb21wb25lbnQnO1xuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtQYXJjZWxDb21wb25lbnRdLFxuICBleHBvcnRzOiBbUGFyY2VsQ29tcG9uZW50XSxcbiAgZW50cnlDb21wb25lbnRzOiBbUGFyY2VsQ29tcG9uZW50XSxcbn0pXG5leHBvcnQgY2xhc3MgUGFyY2VsTW9kdWxlIHt9XG4iXX0= |
@@ -1,4 +0,3 @@ | ||
import { __decorate } from "tslib"; | ||
import { Component, OnInit, OnDestroy, Input, OnChanges, ElementRef } from '@angular/core'; | ||
let ParcelComponent = class ParcelComponent { | ||
import { Component, Input, ElementRef } from '@angular/core'; | ||
export class ParcelComponent { | ||
// eslint-disable-next-line @typescript-eslint/no-parameter-properties | ||
@@ -96,34 +95,21 @@ constructor(host) { | ||
} | ||
}; | ||
} | ||
ParcelComponent.decorators = [ | ||
{ type: Component, args: [{ | ||
selector: 'parcel', | ||
template: '<div></div>' | ||
},] } | ||
]; | ||
ParcelComponent.ctorParameters = () => [ | ||
{ type: ElementRef } | ||
]; | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "config", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "mountParcel", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "onParcelMount", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "wrapWith", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "customProps", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "appendTo", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "handleError", void 0); | ||
ParcelComponent = __decorate([ | ||
Component({ | ||
selector: 'parcel', | ||
template: '<div></div>' | ||
}) | ||
], ParcelComponent); | ||
export { ParcelComponent }; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parcel.component.js","sourceRoot":"ng://single-spa-angular/","sources":["src/parcel-lib/parcel.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy, Input, OnChanges, ElementRef } from '@angular/core';\nimport { Parcel, ParcelConfig, AppProps } from 'single-spa';\n\n@Component({\n  selector: 'parcel',\n  template: '<div></div>',\n})\nexport class ParcelComponent implements OnInit, OnDestroy, OnChanges {\n  @Input() config!: ParcelConfig;\n  @Input() mountParcel!: AppProps['mountParcel'];\n  @Input() onParcelMount: (() => void) | null = null;\n  @Input() wrapWith = 'div';\n  @Input() customProps: object = {};\n  @Input() appendTo: Node | null = null;\n  @Input() handleError = (error: Error) => console.error(error);\n\n  createdDomElement: HTMLElement | null = null;\n  hasError = false;\n  unmounted = true;\n  nextThingToDo!: Promise<any>;\n  parcel!: Parcel;\n\n  // eslint-disable-next-line @typescript-eslint/no-parameter-properties\n  constructor(private host: ElementRef<HTMLElement>) {}\n\n  ngOnInit() {\n    if (!this.config) {\n      throw new Error(\n        `single-spa-angular's Parcel component requires the [config] binding to either be a parcel config or a loading function that returns a promise. See https://github.com/CanopyTax/single-spa-angular`,\n      );\n    }\n\n    this.addThingToDo('mount', () => {\n      const mountParcel = this.mountParcel;\n      if (!mountParcel) {\n        throw new Error(`\n\t\t\t\t  <parcel> was not passed a [mountParcel] binding.\n\t\t\t\t  If you are using <parcel> within a module that is not a single-spa application, you will need to import mountRootParcel from single-spa and pass it into <parcel> as a [mountParcel] binding\n\t\t\t\t`);\n      }\n      let domElement: HTMLElement;\n\n      if (this.appendTo) {\n        this.createdDomElement = domElement = document.createElement(this.wrapWith);\n        this.appendTo.appendChild(domElement);\n      } else {\n        this.createdDomElement = domElement = document.createElement(this.wrapWith);\n        // Except of having `@ViewChild` we can simply get the first child element.\n        const parcelDiv = this.host.nativeElement.children[0];\n        parcelDiv.appendChild(domElement);\n      }\n\n      this.parcel = mountParcel(this.config, { domElement, ...this.customProps });\n\n      if (this.onParcelMount) {\n        this.parcel.mountPromise.then(this.onParcelMount);\n      }\n      this.unmounted = false;\n      return this.parcel.mountPromise;\n    });\n  }\n\n  ngOnChanges() {\n    this.addThingToDo('update', () => {\n      if (this.parcel && this.parcel.update) {\n        return this.parcel.update(this.customProps);\n      }\n    });\n  }\n\n  ngOnDestroy() {\n    this.addThingToDo('unmount', () => {\n      if (this.parcel && this.parcel.getStatus() === 'MOUNTED') {\n        return this.parcel.unmount();\n      }\n    });\n\n    if (this.createdDomElement) {\n      this.createdDomElement.parentNode!.removeChild(this.createdDomElement);\n    }\n\n    this.unmounted = true;\n  }\n\n  addThingToDo(action: string, thing: Function) {\n    if (this.hasError && action !== 'unmount') {\n      // In an error state, we don't do anything anymore except for unmounting\n      return;\n    }\n\n    this.nextThingToDo = (this.nextThingToDo || Promise.resolve())\n      .then((...args) => {\n        if (this.unmounted && action !== 'unmount') {\n          // Never do anything once the angular component unmounts\n          return;\n        }\n\n        return thing(...args);\n      })\n      .catch(error => {\n        this.nextThingToDo = Promise.resolve(); // reset so we don't .then() the bad promise again\n        this.hasError = true;\n\n        if (error && error.message) {\n          error.message = `During '${action}', parcel threw an error: ${error.message}`;\n        }\n\n        if (typeof this.handleError === 'function') {\n          this.handleError(error);\n        } else {\n          setTimeout(() => {\n            throw error;\n          });\n        }\n\n        // No more things to do should be done -- the parcel is in an error state\n        throw error;\n      });\n  }\n}\n"]} | ||
ParcelComponent.propDecorators = { | ||
config: [{ type: Input }], | ||
mountParcel: [{ type: Input }], | ||
onParcelMount: [{ type: Input }], | ||
wrapWith: [{ type: Input }], | ||
customProps: [{ type: Input }], | ||
appendTo: [{ type: Input }], | ||
handleError: [{ type: Input }] | ||
}; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parcel.component.js","sourceRoot":"","sources":["../../../../src/src/parcel-lib/parcel.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy, Input, OnChanges, ElementRef } from '@angular/core';\nimport { Parcel, ParcelConfig, AppProps } from 'single-spa';\n\n@Component({\n  selector: 'parcel',\n  template: '<div></div>',\n})\nexport class ParcelComponent implements OnInit, OnDestroy, OnChanges {\n  @Input() config!: ParcelConfig;\n  @Input() mountParcel!: AppProps['mountParcel'];\n  @Input() onParcelMount: (() => void) | null = null;\n  @Input() wrapWith = 'div';\n  @Input() customProps: object = {};\n  @Input() appendTo: Node | null = null;\n  @Input() handleError = (error: Error) => console.error(error);\n\n  createdDomElement: HTMLElement | null = null;\n  hasError = false;\n  unmounted = true;\n  nextThingToDo!: Promise<any>;\n  parcel!: Parcel;\n\n  // eslint-disable-next-line @typescript-eslint/no-parameter-properties\n  constructor(private host: ElementRef<HTMLElement>) {}\n\n  ngOnInit() {\n    if (!this.config) {\n      throw new Error(\n        `single-spa-angular's Parcel component requires the [config] binding to either be a parcel config or a loading function that returns a promise. See https://github.com/CanopyTax/single-spa-angular`,\n      );\n    }\n\n    this.addThingToDo('mount', () => {\n      const mountParcel = this.mountParcel;\n      if (!mountParcel) {\n        throw new Error(`\n\t\t\t\t  <parcel> was not passed a [mountParcel] binding.\n\t\t\t\t  If you are using <parcel> within a module that is not a single-spa application, you will need to import mountRootParcel from single-spa and pass it into <parcel> as a [mountParcel] binding\n\t\t\t\t`);\n      }\n      let domElement: HTMLElement;\n\n      if (this.appendTo) {\n        this.createdDomElement = domElement = document.createElement(this.wrapWith);\n        this.appendTo.appendChild(domElement);\n      } else {\n        this.createdDomElement = domElement = document.createElement(this.wrapWith);\n        // Except of having `@ViewChild` we can simply get the first child element.\n        const parcelDiv = this.host.nativeElement.children[0];\n        parcelDiv.appendChild(domElement);\n      }\n\n      this.parcel = mountParcel(this.config, { domElement, ...this.customProps });\n\n      if (this.onParcelMount) {\n        this.parcel.mountPromise.then(this.onParcelMount);\n      }\n      this.unmounted = false;\n      return this.parcel.mountPromise;\n    });\n  }\n\n  ngOnChanges() {\n    this.addThingToDo('update', () => {\n      if (this.parcel && this.parcel.update) {\n        return this.parcel.update(this.customProps);\n      }\n    });\n  }\n\n  ngOnDestroy() {\n    this.addThingToDo('unmount', () => {\n      if (this.parcel && this.parcel.getStatus() === 'MOUNTED') {\n        return this.parcel.unmount();\n      }\n    });\n\n    if (this.createdDomElement) {\n      this.createdDomElement.parentNode!.removeChild(this.createdDomElement);\n    }\n\n    this.unmounted = true;\n  }\n\n  addThingToDo(action: string, thing: Function) {\n    if (this.hasError && action !== 'unmount') {\n      // In an error state, we don't do anything anymore except for unmounting\n      return;\n    }\n\n    this.nextThingToDo = (this.nextThingToDo || Promise.resolve())\n      .then((...args) => {\n        if (this.unmounted && action !== 'unmount') {\n          // Never do anything once the angular component unmounts\n          return;\n        }\n\n        return thing(...args);\n      })\n      .catch(error => {\n        this.nextThingToDo = Promise.resolve(); // reset so we don't .then() the bad promise again\n        this.hasError = true;\n\n        if (error && error.message) {\n          error.message = `During '${action}', parcel threw an error: ${error.message}`;\n        }\n\n        if (typeof this.handleError === 'function') {\n          this.handleError(error);\n        } else {\n          setTimeout(() => {\n            throw error;\n          });\n        }\n\n        // No more things to do should be done -- the parcel is in an error state\n        throw error;\n      });\n  }\n}\n"]} |
@@ -5,2 +5,2 @@ export { singleSpaAngular } from './single-spa-angular'; | ||
export { ParcelComponent } from './parcel-lib/parcel.component'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiJuZzovL3NpbmdsZS1zcGEtYW5ndWxhci8iLCJzb3VyY2VzIjpbInNyYy9wdWJsaWNfYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBQ3hELE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBQy9ELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUNsRCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sK0JBQStCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBzaW5nbGVTcGFBbmd1bGFyIH0gZnJvbSAnLi9zaW5nbGUtc3BhLWFuZ3VsYXInO1xuZXhwb3J0IHsgZ2V0U2luZ2xlU3BhRXh0cmFQcm92aWRlcnMgfSBmcm9tICcuL2V4dHJhLXByb3ZpZGVycyc7XG5leHBvcnQgeyBQYXJjZWxNb2R1bGUgfSBmcm9tICcuL3BhcmNlbC1saWIvaW5kZXgnO1xuZXhwb3J0IHsgUGFyY2VsQ29tcG9uZW50IH0gZnJvbSAnLi9wYXJjZWwtbGliL3BhcmNlbC5jb21wb25lbnQnO1xuIl19 | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zcmMvcHVibGljX2FwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQztBQUN4RCxPQUFPLEVBQUUsMEJBQTBCLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUMvRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFDbEQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLCtCQUErQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgc2luZ2xlU3BhQW5ndWxhciB9IGZyb20gJy4vc2luZ2xlLXNwYS1hbmd1bGFyJztcbmV4cG9ydCB7IGdldFNpbmdsZVNwYUV4dHJhUHJvdmlkZXJzIH0gZnJvbSAnLi9leHRyYS1wcm92aWRlcnMnO1xuZXhwb3J0IHsgUGFyY2VsTW9kdWxlIH0gZnJvbSAnLi9wYXJjZWwtbGliL2luZGV4JztcbmV4cG9ydCB7IFBhcmNlbENvbXBvbmVudCB9IGZyb20gJy4vcGFyY2VsLWxpYi9wYXJjZWwuY29tcG9uZW50JztcbiJdfQ== |
import { __awaiter } from "tslib"; | ||
import { getContainerEl, chooseDomElementGetter, removeApplicationFromDOMIfIvyEnabled, } from 'single-spa-angular/internals'; | ||
import { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled, } from 'single-spa-angular/internals'; | ||
import { SingleSpaPlatformLocation } from './extra-providers'; | ||
const defaultOpts = { | ||
// Required opts that will be set by the library consumer. | ||
const defaultOptions = { | ||
// Required options that will be set by the library consumer. | ||
NgZone: null, | ||
bootstrapFunction: null, | ||
template: null, | ||
// optional opts | ||
// Optional options | ||
Router: undefined, | ||
@@ -15,24 +15,24 @@ domElementGetter: undefined, | ||
}; | ||
export function singleSpaAngular(userOpts) { | ||
if (typeof userOpts !== 'object') { | ||
export function singleSpaAngular(userOptions) { | ||
if (typeof userOptions !== 'object') { | ||
throw Error('single-spa-angular requires a configuration object'); | ||
} | ||
const opts = Object.assign(Object.assign({}, defaultOpts), userOpts); | ||
if (typeof opts.bootstrapFunction !== 'function') { | ||
throw Error('single-spa-angular must be passed an opts.bootstrapFunction'); | ||
const options = Object.assign(Object.assign({}, defaultOptions), userOptions); | ||
if (typeof options.bootstrapFunction !== 'function') { | ||
throw Error('single-spa-angular must be passed an options.bootstrapFunction'); | ||
} | ||
if (typeof opts.template !== 'string') { | ||
throw Error('single-spa-angular must be passed opts.template string'); | ||
if (typeof options.template !== 'string') { | ||
throw Error('single-spa-angular must be passed options.template string'); | ||
} | ||
if (!opts.NgZone) { | ||
throw Error(`single-spa-angular must be passed the NgZone opt`); | ||
if (!options.NgZone) { | ||
throw Error(`single-spa-angular must be passed the NgZone option`); | ||
} | ||
return { | ||
bootstrap: bootstrap.bind(null, opts), | ||
mount: mount.bind(null, opts), | ||
unmount: unmount.bind(null, opts), | ||
update: opts.updateFunction, | ||
bootstrap: bootstrap.bind(null, options), | ||
mount: mount.bind(null, options), | ||
unmount: unmount.bind(null, options), | ||
update: options.updateFunction, | ||
}; | ||
} | ||
function bootstrap(opts, props) { | ||
function bootstrap(options, props) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
@@ -42,7 +42,7 @@ // Angular provides an opportunity to develop `zone-less` application, where developers | ||
// See https://angular.io/guide/zone#noopzone | ||
if (opts.NgZone === 'noop') { | ||
if (options.NgZone === 'noop') { | ||
return; | ||
} | ||
// In order for multiple Angular apps to work concurrently on a page, they each need a unique identifier. | ||
opts.zoneIdentifier = `single-spa-angular:${props.name || props.appName}`; | ||
options.zoneIdentifier = `single-spa-angular:${props.name || props.appName}`; | ||
// This is a hack, since NgZone doesn't allow you to configure the property that identifies your zone. | ||
@@ -53,8 +53,8 @@ // See https://github.com/PlaceMe-SAS/single-spa-angular-cli/issues/33, | ||
// and https://github.com/angular/angular/blob/a14dc2d7a4821a19f20a9547053a5734798f541e/packages/core/src/zone/ng_zone.ts#L257 | ||
opts.NgZone.isInAngularZone = function () { | ||
options.NgZone.isInAngularZone = () => { | ||
// @ts-ignore | ||
return window.Zone.current._properties[opts.zoneIdentifier] === true; | ||
return window.Zone.current._properties[options.zoneIdentifier] === true; | ||
}; | ||
opts.routingEventListener = () => { | ||
opts.bootstrappedNgZone.run(() => { | ||
options.routingEventListener = () => { | ||
options.bootstrappedNgZone.run(() => { | ||
// See https://github.com/single-spa/single-spa-angular/issues/86 | ||
@@ -67,20 +67,15 @@ // Zone is unaware of the single-spa navigation change and so Angular change detection doesn't work | ||
} | ||
function mount(opts, props) { | ||
function mount(options, props) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
const domElementGetter = chooseDomElementGetter(opts, props); | ||
if (!domElementGetter) { | ||
throw Error(`cannot mount angular application '${props.name || props.appName}' without a domElementGetter provided either as an opt or a prop`); | ||
} | ||
const containerEl = getContainerEl(domElementGetter); | ||
containerEl.innerHTML = opts.template; | ||
const bootstrapPromise = opts.bootstrapFunction(props); | ||
getContainerElementAndSetTemplate(options, props); | ||
const bootstrapPromise = options.bootstrapFunction(props); | ||
if (!(bootstrapPromise instanceof Promise)) { | ||
throw Error(`single-spa-angular: the opts.bootstrapFunction must return a promise, but instead returned a '${typeof bootstrapPromise}' that is not a Promise`); | ||
throw Error(`single-spa-angular: the options.bootstrapFunction must return a promise, but instead returned a '${typeof bootstrapPromise}' that is not a Promise`); | ||
} | ||
const module = yield bootstrapPromise; | ||
if (!module || typeof module.destroy !== 'function') { | ||
throw Error(`single-spa-angular: the opts.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?`); | ||
throw Error(`single-spa-angular: the options.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?`); | ||
} | ||
const singleSpaPlatformLocation = module.injector.get(SingleSpaPlatformLocation, null); | ||
const ngZoneEnabled = opts.NgZone !== 'noop'; | ||
const ngZoneEnabled = options.NgZone !== 'noop'; | ||
// The user has to provide `BrowserPlatformLocation` only if his application uses routing. | ||
@@ -90,3 +85,3 @@ // So if he provided `Router` but didn't provide `BrowserPlatformLocation` then we have to inform him. | ||
// `zone-less` change detection, if `NgZone` is `noop` then we can skip it. | ||
if (ngZoneEnabled && opts.Router && singleSpaPlatformLocation === null) { | ||
if (ngZoneEnabled && options.Router && singleSpaPlatformLocation === null) { | ||
throw new Error(` | ||
@@ -96,18 +91,18 @@ single-spa-angular: could not retrieve extra providers from the platform injector. Did you call platformBrowserDynamic(getSingleSpaExtraProviders()).bootstrapModule()? | ||
} | ||
const bootstrappedOpts = opts; | ||
const bootstrappedOptions = options; | ||
if (ngZoneEnabled) { | ||
const ngZone = module.injector.get(opts.NgZone); | ||
const ngZone = module.injector.get(options.NgZone); | ||
const zoneIdentifier = bootstrappedOptions.zoneIdentifier; | ||
// `NgZone` can be enabled but routing may not be used thus `getSingleSpaExtraProviders()` | ||
// function was not called. | ||
if (singleSpaPlatformLocation !== null) { | ||
singleSpaPlatformLocation.setNgZone(ngZone); | ||
// Cleanup resources, especially remove event listeners thus they will not be added | ||
// twice when application gets bootstrapped the second time. | ||
module.onDestroy(() => singleSpaPlatformLocation.destroy()); | ||
module.onDestroy(() => singleSpaPlatformLocation.destroyApplication(zoneIdentifier)); | ||
} | ||
bootstrappedOpts.bootstrappedNgZone = ngZone; | ||
bootstrappedOpts.bootstrappedNgZone['_inner']._properties[bootstrappedOpts.zoneIdentifier] = true; | ||
window.addEventListener('single-spa:routing-event', bootstrappedOpts.routingEventListener); | ||
bootstrappedOptions.bootstrappedNgZone = ngZone; | ||
bootstrappedOptions.bootstrappedNgZone['_inner']._properties[zoneIdentifier] = true; | ||
window.addEventListener('single-spa:routing-event', bootstrappedOptions.routingEventListener); | ||
} | ||
bootstrappedOpts.bootstrappedModule = module; | ||
bootstrappedOptions.bootstrappedModule = module; | ||
return module; | ||
@@ -117,13 +112,13 @@ }); | ||
// eslint-disable-next-line @typescript-eslint/no-unused-vars | ||
function unmount(opts, props) { | ||
function unmount(options, props) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
if (opts.Router) { | ||
if (options.Router) { | ||
// Workaround for https://github.com/angular/angular/issues/19079 | ||
const router = opts.bootstrappedModule.injector.get(opts.Router); | ||
const router = options.bootstrappedModule.injector.get(options.Router); | ||
router.dispose(); | ||
} | ||
if (opts.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', opts.routingEventListener); | ||
if (options.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', options.routingEventListener); | ||
} | ||
if (opts.AnimationEngine) { | ||
if (options.AnimationEngine) { | ||
/* | ||
@@ -153,12 +148,12 @@ The BrowserAnimationsModule does not clean up after itself :'(. When you unmount/destroy the main module, the | ||
*/ | ||
const animationEngine = opts.bootstrappedModule.injector.get(opts.AnimationEngine); | ||
const animationEngine = options.bootstrappedModule.injector.get(options.AnimationEngine); | ||
animationEngine._transitionEngine.flush(); | ||
} | ||
opts.bootstrappedModule.destroy(); | ||
delete opts.bootstrappedModule; | ||
options.bootstrappedModule.destroy(); | ||
delete options.bootstrappedModule; | ||
// This is an issue. Issue has been created and Angular team is working on the fix: | ||
// https://github.com/angular/angular/issues/36449 | ||
removeApplicationFromDOMIfIvyEnabled(opts, props); | ||
removeApplicationFromDOMIfIvyEnabled(options, props); | ||
}); | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-spa-angular.js","sourceRoot":"ng://single-spa-angular/","sources":["src/single-spa-angular.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgModuleRef } from '@angular/core';\nimport { LifeCycles } from 'single-spa';\nimport {\n  getContainerEl,\n  chooseDomElementGetter,\n  removeApplicationFromDOMIfIvyEnabled,\n  SingleSpaAngularOpts,\n  BootstrappedSingleSpaAngularOpts,\n} from 'single-spa-angular/internals';\n\nimport { SingleSpaPlatformLocation } from './extra-providers';\n\nconst defaultOpts = {\n  // Required opts that will be set by the library consumer.\n  NgZone: null!,\n  bootstrapFunction: null!,\n  template: null!,\n  // optional opts\n  Router: undefined,\n  domElementGetter: undefined, // only optional if you provide a domElementGetter as a custom prop\n  AnimationEngine: undefined,\n  updateFunction: () => Promise.resolve(),\n};\n\nexport function singleSpaAngular(userOpts: SingleSpaAngularOpts): LifeCycles {\n  if (typeof userOpts !== 'object') {\n    throw Error('single-spa-angular requires a configuration object');\n  }\n\n  const opts: SingleSpaAngularOpts = {\n    ...defaultOpts,\n    ...userOpts,\n  };\n\n  if (typeof opts.bootstrapFunction !== 'function') {\n    throw Error('single-spa-angular must be passed an opts.bootstrapFunction');\n  }\n\n  if (typeof opts.template !== 'string') {\n    throw Error('single-spa-angular must be passed opts.template string');\n  }\n\n  if (!opts.NgZone) {\n    throw Error(`single-spa-angular must be passed the NgZone opt`);\n  }\n\n  return {\n    bootstrap: bootstrap.bind(null, opts as BootstrappedSingleSpaAngularOpts),\n    mount: mount.bind(null, opts),\n    unmount: unmount.bind(null, opts as BootstrappedSingleSpaAngularOpts),\n    update: opts.updateFunction,\n  };\n}\n\nasync function bootstrap(opts: BootstrappedSingleSpaAngularOpts, props: any): Promise<void> {\n  // Angular provides an opportunity to develop `zone-less` application, where developers\n  // have to trigger change detection manually.\n  // See https://angular.io/guide/zone#noopzone\n  if (opts.NgZone === 'noop') {\n    return;\n  }\n\n  // In order for multiple Angular apps to work concurrently on a page, they each need a unique identifier.\n  opts.zoneIdentifier = `single-spa-angular:${props.name || props.appName}`;\n\n  // This is a hack, since NgZone doesn't allow you to configure the property that identifies your zone.\n  // See https://github.com/PlaceMe-SAS/single-spa-angular-cli/issues/33,\n  // https://github.com/single-spa/single-spa-angular/issues/47,\n  // https://github.com/angular/angular/blob/a14dc2d7a4821a19f20a9547053a5734798f541e/packages/core/src/zone/ng_zone.ts#L144,\n  // and https://github.com/angular/angular/blob/a14dc2d7a4821a19f20a9547053a5734798f541e/packages/core/src/zone/ng_zone.ts#L257\n  opts.NgZone.isInAngularZone = function () {\n    // @ts-ignore\n    return window.Zone.current._properties[opts.zoneIdentifier] === true;\n  };\n\n  opts.routingEventListener = () => {\n    opts.bootstrappedNgZone!.run(() => {\n      // See https://github.com/single-spa/single-spa-angular/issues/86\n      // Zone is unaware of the single-spa navigation change and so Angular change detection doesn't work\n      // unless we tell Zone that something happened\n    });\n  };\n}\n\nasync function mount(opts: SingleSpaAngularOpts, props: any): Promise<NgModuleRef<any>> {\n  const domElementGetter = chooseDomElementGetter(opts, props);\n\n  if (!domElementGetter) {\n    throw Error(\n      `cannot mount angular application '${\n        props.name || props.appName\n      }' without a domElementGetter provided either as an opt or a prop`,\n    );\n  }\n\n  const containerEl = getContainerEl(domElementGetter);\n  containerEl.innerHTML = opts.template;\n  const bootstrapPromise = opts.bootstrapFunction(props);\n\n  if (!(bootstrapPromise instanceof Promise)) {\n    throw Error(\n      `single-spa-angular: the opts.bootstrapFunction must return a promise, but instead returned a '${typeof bootstrapPromise}' that is not a Promise`,\n    );\n  }\n\n  const module: NgModuleRef<any> = await bootstrapPromise;\n\n  if (!module || typeof module.destroy !== 'function') {\n    throw Error(\n      `single-spa-angular: the opts.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?`,\n    );\n  }\n\n  const singleSpaPlatformLocation: SingleSpaPlatformLocation | null = module.injector.get(\n    SingleSpaPlatformLocation,\n    null,\n  );\n\n  const ngZoneEnabled = opts.NgZone !== 'noop';\n\n  // The user has to provide `BrowserPlatformLocation` only if his application uses routing.\n  // So if he provided `Router` but didn't provide `BrowserPlatformLocation` then we have to inform him.\n  // Also `getSingleSpaExtraProviders()` function should be called only if the user doesn't use\n  // `zone-less` change detection, if `NgZone` is `noop` then we can skip it.\n  if (ngZoneEnabled && opts.Router && singleSpaPlatformLocation === null) {\n    throw new Error(`\t\n      single-spa-angular: could not retrieve extra providers from the platform injector. Did you call platformBrowserDynamic(getSingleSpaExtraProviders()).bootstrapModule()?\n    `);\n  }\n\n  const bootstrappedOpts = opts as BootstrappedSingleSpaAngularOpts;\n\n  if (ngZoneEnabled) {\n    const ngZone: import('@angular/core').NgZone = module.injector.get(opts.NgZone);\n\n    // `NgZone` can be enabled but routing may not be used thus `getSingleSpaExtraProviders()`\n    // function was not called.\n    if (singleSpaPlatformLocation !== null) {\n      singleSpaPlatformLocation.setNgZone(ngZone);\n      // Cleanup resources, especially remove event listeners thus they will not be added\n      // twice when application gets bootstrapped the second time.\n      module.onDestroy(() => singleSpaPlatformLocation.destroy());\n    }\n\n    bootstrappedOpts.bootstrappedNgZone = ngZone;\n    bootstrappedOpts.bootstrappedNgZone['_inner']._properties[\n      bootstrappedOpts.zoneIdentifier\n    ] = true;\n    window.addEventListener('single-spa:routing-event', bootstrappedOpts.routingEventListener!);\n  }\n\n  bootstrappedOpts.bootstrappedModule = module;\n  return module;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nasync function unmount(opts: BootstrappedSingleSpaAngularOpts, props: any): Promise<void> {\n  if (opts.Router) {\n    // Workaround for https://github.com/angular/angular/issues/19079\n    const router = opts.bootstrappedModule.injector.get(opts.Router);\n    router.dispose();\n  }\n\n  if (opts.routingEventListener) {\n    window.removeEventListener('single-spa:routing-event', opts.routingEventListener);\n  }\n\n  if (opts.AnimationEngine) {\n    /*\n    The BrowserAnimationsModule does not clean up after itself :'(. When you unmount/destroy the main module, the\n    BrowserAnimationsModule uses an AnimationRenderer thing to remove dom elements from the page. But the AnimationRenderer\n    defers the actual work to the TransitionAnimationEngine to do this, and the TransitionAnimationEngine doesn't actually\n    remove the dom node, but just calls \"markElementAsRemoved()\".\n\n    See https://github.com/angular/angular/blob/db62ccf9eb46ee89366ade586365ea027bb93eb1/packages/animations/browser/src/render/transition_animation_engine.ts#L717\n\n    What markAsRemovedDoes is put it into an array called \"collectedLeaveElements\", which is all the elements that should be removed\n    after the DOM has had a chance to do any animations.\n\n    See https://github.com/angular/angular/blob/master/packages/animations/browser/src/render/transition_animation_engine.ts#L525\n\n    The actual dom nodes aren't removed until the TransitionAnimationEngine \"flushes\".\n\n    See https://github.com/angular/angular/blob/db62ccf9eb46ee89366ade586365ea027bb93eb1/packages/animations/browser/src/render/transition_animation_engine.ts#L851\n\n    Unfortunately, though, that \"flush\" will never happen, since the entire module is being destroyed and there will be no more flushes.\n    So what we do in this code is force one more flush of the animations after the module is destroyed.\n\n    Ideally, we would do this by getting the TransitionAnimationEngine directly and flushing it. Unfortunately, though, it's private class\n    that cannot be imported and is not provided to the dependency injector. So, instead, we get its wrapper class, AnimationEngine, and then\n    access its private variable reference to the TransitionAnimationEngine so that we can call flush.\n    */\n    const animationEngine = opts.bootstrappedModule.injector.get(opts.AnimationEngine);\n    animationEngine._transitionEngine.flush();\n  }\n\n  opts.bootstrappedModule.destroy();\n  delete opts.bootstrappedModule;\n\n  // This is an issue. Issue has been created and Angular team is working on the fix:\n  // https://github.com/angular/angular/issues/36449\n  removeApplicationFromDOMIfIvyEnabled(opts, props);\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-spa-angular.js","sourceRoot":"","sources":["../../../src/src/single-spa-angular.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgModuleRef, NgZone } from '@angular/core';\nimport { LifeCycles } from 'single-spa';\nimport {\n  getContainerElementAndSetTemplate,\n  removeApplicationFromDOMIfIvyEnabled,\n} from 'single-spa-angular/internals';\n\nimport { SingleSpaPlatformLocation } from './extra-providers';\nimport { SingleSpaAngularOptions, BootstrappedSingleSpaAngularOptions } from './types';\n\nconst defaultOptions = {\n  // Required options that will be set by the library consumer.\n  NgZone: null!,\n  bootstrapFunction: null!,\n  template: null!,\n  // Optional options\n  Router: undefined,\n  domElementGetter: undefined, // only optional if you provide a domElementGetter as a custom prop\n  AnimationEngine: undefined,\n  updateFunction: () => Promise.resolve(),\n};\n\nexport function singleSpaAngular(userOptions: SingleSpaAngularOptions): LifeCycles {\n  if (typeof userOptions !== 'object') {\n    throw Error('single-spa-angular requires a configuration object');\n  }\n\n  const options: SingleSpaAngularOptions = {\n    ...defaultOptions,\n    ...userOptions,\n  };\n\n  if (typeof options.bootstrapFunction !== 'function') {\n    throw Error('single-spa-angular must be passed an options.bootstrapFunction');\n  }\n\n  if (typeof options.template !== 'string') {\n    throw Error('single-spa-angular must be passed options.template string');\n  }\n\n  if (!options.NgZone) {\n    throw Error(`single-spa-angular must be passed the NgZone option`);\n  }\n\n  return {\n    bootstrap: bootstrap.bind(null, options as BootstrappedSingleSpaAngularOptions),\n    mount: mount.bind(null, options),\n    unmount: unmount.bind(null, options as BootstrappedSingleSpaAngularOptions),\n    update: options.updateFunction,\n  };\n}\n\nasync function bootstrap(options: BootstrappedSingleSpaAngularOptions, props: any): Promise<void> {\n  // Angular provides an opportunity to develop `zone-less` application, where developers\n  // have to trigger change detection manually.\n  // See https://angular.io/guide/zone#noopzone\n  if (options.NgZone === 'noop') {\n    return;\n  }\n\n  // In order for multiple Angular apps to work concurrently on a page, they each need a unique identifier.\n  options.zoneIdentifier = `single-spa-angular:${props.name || props.appName}`;\n\n  // This is a hack, since NgZone doesn't allow you to configure the property that identifies your zone.\n  // See https://github.com/PlaceMe-SAS/single-spa-angular-cli/issues/33,\n  // https://github.com/single-spa/single-spa-angular/issues/47,\n  // https://github.com/angular/angular/blob/a14dc2d7a4821a19f20a9547053a5734798f541e/packages/core/src/zone/ng_zone.ts#L144,\n  // and https://github.com/angular/angular/blob/a14dc2d7a4821a19f20a9547053a5734798f541e/packages/core/src/zone/ng_zone.ts#L257\n  options.NgZone.isInAngularZone = () => {\n    // @ts-ignore\n    return window.Zone.current._properties[options.zoneIdentifier] === true;\n  };\n\n  options.routingEventListener = () => {\n    options.bootstrappedNgZone!.run(() => {\n      // See https://github.com/single-spa/single-spa-angular/issues/86\n      // Zone is unaware of the single-spa navigation change and so Angular change detection doesn't work\n      // unless we tell Zone that something happened\n    });\n  };\n}\n\nasync function mount(options: SingleSpaAngularOptions, props: any): Promise<NgModuleRef<any>> {\n  getContainerElementAndSetTemplate(options, props);\n\n  const bootstrapPromise = options.bootstrapFunction(props);\n\n  if (!(bootstrapPromise instanceof Promise)) {\n    throw Error(\n      `single-spa-angular: the options.bootstrapFunction must return a promise, but instead returned a '${typeof bootstrapPromise}' that is not a Promise`,\n    );\n  }\n\n  const module: NgModuleRef<any> = await bootstrapPromise;\n\n  if (!module || typeof module.destroy !== 'function') {\n    throw Error(\n      `single-spa-angular: the options.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?`,\n    );\n  }\n\n  const singleSpaPlatformLocation: SingleSpaPlatformLocation | null = module.injector.get(\n    SingleSpaPlatformLocation,\n    null,\n  );\n\n  const ngZoneEnabled = options.NgZone !== 'noop';\n\n  // The user has to provide `BrowserPlatformLocation` only if his application uses routing.\n  // So if he provided `Router` but didn't provide `BrowserPlatformLocation` then we have to inform him.\n  // Also `getSingleSpaExtraProviders()` function should be called only if the user doesn't use\n  // `zone-less` change detection, if `NgZone` is `noop` then we can skip it.\n  if (ngZoneEnabled && options.Router && singleSpaPlatformLocation === null) {\n    throw new Error(`\t\n      single-spa-angular: could not retrieve extra providers from the platform injector. Did you call platformBrowserDynamic(getSingleSpaExtraProviders()).bootstrapModule()?\n    `);\n  }\n\n  const bootstrappedOptions = options as BootstrappedSingleSpaAngularOptions;\n\n  if (ngZoneEnabled) {\n    const ngZone: NgZone = module.injector.get(options.NgZone);\n    const zoneIdentifier: string = bootstrappedOptions.zoneIdentifier!;\n\n    // `NgZone` can be enabled but routing may not be used thus `getSingleSpaExtraProviders()`\n    // function was not called.\n    if (singleSpaPlatformLocation !== null) {\n      // Cleanup resources, especially remove event listeners thus they will not be added\n      // twice when application gets bootstrapped the second time.\n      module.onDestroy(() => singleSpaPlatformLocation.destroyApplication(zoneIdentifier));\n    }\n\n    bootstrappedOptions.bootstrappedNgZone = ngZone;\n    bootstrappedOptions.bootstrappedNgZone['_inner']._properties[zoneIdentifier] = true;\n    window.addEventListener('single-spa:routing-event', bootstrappedOptions.routingEventListener!);\n  }\n\n  bootstrappedOptions.bootstrappedModule = module;\n  return module;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nasync function unmount(options: BootstrappedSingleSpaAngularOptions, props: any): Promise<void> {\n  if (options.Router) {\n    // Workaround for https://github.com/angular/angular/issues/19079\n    const router = options.bootstrappedModule.injector.get(options.Router);\n    router.dispose();\n  }\n\n  if (options.routingEventListener) {\n    window.removeEventListener('single-spa:routing-event', options.routingEventListener);\n  }\n\n  if (options.AnimationEngine) {\n    /*\n    The BrowserAnimationsModule does not clean up after itself :'(. When you unmount/destroy the main module, the\n    BrowserAnimationsModule uses an AnimationRenderer thing to remove dom elements from the page. But the AnimationRenderer\n    defers the actual work to the TransitionAnimationEngine to do this, and the TransitionAnimationEngine doesn't actually\n    remove the dom node, but just calls \"markElementAsRemoved()\".\n\n    See https://github.com/angular/angular/blob/db62ccf9eb46ee89366ade586365ea027bb93eb1/packages/animations/browser/src/render/transition_animation_engine.ts#L717\n\n    What markAsRemovedDoes is put it into an array called \"collectedLeaveElements\", which is all the elements that should be removed\n    after the DOM has had a chance to do any animations.\n\n    See https://github.com/angular/angular/blob/master/packages/animations/browser/src/render/transition_animation_engine.ts#L525\n\n    The actual dom nodes aren't removed until the TransitionAnimationEngine \"flushes\".\n\n    See https://github.com/angular/angular/blob/db62ccf9eb46ee89366ade586365ea027bb93eb1/packages/animations/browser/src/render/transition_animation_engine.ts#L851\n\n    Unfortunately, though, that \"flush\" will never happen, since the entire module is being destroyed and there will be no more flushes.\n    So what we do in this code is force one more flush of the animations after the module is destroyed.\n\n    Ideally, we would do this by getting the TransitionAnimationEngine directly and flushing it. Unfortunately, though, it's private class\n    that cannot be imported and is not provided to the dependency injector. So, instead, we get its wrapper class, AnimationEngine, and then\n    access its private variable reference to the TransitionAnimationEngine so that we can call flush.\n    */\n    const animationEngine = options.bootstrappedModule.injector.get(options.AnimationEngine);\n    animationEngine._transitionEngine.flush();\n  }\n\n  options.bootstrappedModule.destroy();\n  delete options.bootstrappedModule;\n\n  // This is an issue. Issue has been created and Angular team is working on the fix:\n  // https://github.com/angular/angular/issues/36449\n  removeApplicationFromDOMIfIvyEnabled(options, props);\n}\n"]} |
@@ -1,5 +0,5 @@ | ||
function removeApplicationFromDOMIfIvyEnabled(opts, props) { | ||
function removeApplicationFromDOMIfIvyEnabled(options, props) { | ||
if (ivyEnabled()) { | ||
const domElementGetter = chooseDomElementGetter(opts, props); | ||
const domElement = getContainerEl(domElementGetter); | ||
const domElementGetter = chooseDomElementGetter(options, props); | ||
const domElement = getContainerElement(domElementGetter); | ||
// View Engine removes all nodes automatically when calling `NgModuleRef.destroy()`, | ||
@@ -12,3 +12,12 @@ // which calls `ComponentRef.destroy()`. | ||
} | ||
function getContainerEl(domElementGetter) { | ||
function getContainerElementAndSetTemplate(options, props) { | ||
const domElementGetter = chooseDomElementGetter(options, props); | ||
if (!domElementGetter) { | ||
throw Error(`Cannot mount angular application '${props.name || props.appName}' without a domElementGetter provided either as an opt or a prop`); | ||
} | ||
const containerElement = getContainerElement(domElementGetter); | ||
containerElement.innerHTML = options.template; | ||
return containerElement; | ||
} | ||
function getContainerElement(domElementGetter) { | ||
const element = domElementGetter(); | ||
@@ -67,3 +76,3 @@ if (!element) { | ||
export { chooseDomElementGetter, getContainerEl, removeApplicationFromDOMIfIvyEnabled }; | ||
export { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled }; | ||
//# sourceMappingURL=single-spa-angular-internals.js.map |
@@ -1,7 +0,7 @@ | ||
import { __decorate, __awaiter } from 'tslib'; | ||
import { chooseDomElementGetter, getContainerEl, removeApplicationFromDOMIfIvyEnabled } from 'single-spa-angular/internals'; | ||
import { Injectable, Inject, ElementRef, Input, Component, NgModule } from '@angular/core'; | ||
import { __awaiter } from 'tslib'; | ||
import { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled } from 'single-spa-angular/internals'; | ||
import { Injectable, Inject, Component, ElementRef, Input, NgModule } from '@angular/core'; | ||
import { ɵBrowserPlatformLocation, DOCUMENT, PlatformLocation } from '@angular/common'; | ||
let SingleSpaPlatformLocation = class SingleSpaPlatformLocation extends ɵBrowserPlatformLocation { | ||
class SingleSpaPlatformLocation extends ɵBrowserPlatformLocation { | ||
constructor() { | ||
@@ -12,5 +12,21 @@ super(...arguments); | ||
this.skipNextPopState = false; | ||
this.onPopStateListeners = []; | ||
// The key here is an actual forked `Zone` of some specific application. | ||
// We will be able to find the specific zone when application gets destroyed | ||
// by application `name`. | ||
// The reason of that the `onPopState` method is invoked during `bootstrapModule` | ||
// and we can't know what application has invoked it. Why should we know the application | ||
// that has invoked `onPopState`? When application gets destroyed in a `sharing dependencies mode` | ||
// (when there is a single platform per all applications) we want to remove application | ||
// specific `popstate` listeners. E.g. if there are 2 applications: | ||
// * shop application adds `popstate` listener | ||
// * navbar application adds `popstate` listener | ||
// When shop application gets destroyed we want to remove only its `popstate` listener. | ||
this.zoneToOnPopStateListenersMap = new Map(); | ||
// This is used only to make `Zone.wrap` happy, since it requires 2 arguments | ||
// and the second argument is a unique string which `zone.js` uses for debugging purposes. | ||
// We might want to use the application name, but we're not able to get it when `onPopState` | ||
// method is called during module bootstrapping. | ||
this.source = 0; | ||
} | ||
destroy() { | ||
destroyApplication(zoneIdentifier) { | ||
// TLDR: Angular adds `popstate` event listener and then doesn't remove it when application gets destroyed. | ||
@@ -20,10 +36,15 @@ // Basically, Angular has a potentional memory leak. The `ɵBrowserPlatformLocation` | ||
// https://github.com/angular/angular/blob/14be55c9facf3e47b8c97df4502dc3f0f897da03/packages/common/src/location/platform_location.ts#L126 | ||
for (const onPopStateListener of this.onPopStateListeners) { | ||
window.removeEventListener('popstate', onPopStateListener); | ||
const zone = [...this.zoneToOnPopStateListenersMap.keys()].find( | ||
// `getZoneWith` will return a zone which defines a `key` and in our case | ||
// we define a custom key in `single-spa-angular.ts` | ||
// via this line of code: | ||
// `_properties[zoneIdentifier] = true;` | ||
zone => zone.getZoneWith(zoneIdentifier) !== null); | ||
const onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone); | ||
if (Array.isArray(onPopStateListeners)) { | ||
for (const onPopStateListener of onPopStateListeners) { | ||
window.removeEventListener('popstate', onPopStateListener); | ||
} | ||
} | ||
// We do this because the `SingleSpaPlatformLocation` is a part of PLATFORM_INJECTOR, | ||
// which means it's created only once and will not be garbage collected, since the PLATFORM_INJECTOR | ||
// will keep reference to its instance. | ||
// TODO: https://github.com/single-spa/single-spa-angular/issues/170 | ||
this.onPopStateListeners = []; | ||
this.zoneToOnPopStateListenersMap.delete(zone); | ||
} | ||
@@ -39,2 +60,12 @@ pushState(state, title, url) { | ||
onPopState(fn) { | ||
// `Zone.current` will reference the zone that serves as an execution context | ||
// to some specific application, especially when `onPopState` is called. | ||
const zone = Zone.current; | ||
// Wrap any event listener into zone that is specific to some application. | ||
// The main issue is `back/forward` buttons of browsers, because they invoke | ||
// `history.back|forward` which dispatch `popstate` event. Since `single-spa` | ||
// overrides `history.replaceState` Angular's zone cannot intercept this event. | ||
// Only the root zone is able to intercept all events. | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
fn = zone.wrap(fn, `${this.source++}`); | ||
const onPopStateListener = (event) => { | ||
@@ -50,23 +81,21 @@ // The `LocationChangeEvent` doesn't have the `singleSpa` property, since it's added | ||
else { | ||
// Wrap any event listener into zone that is specific to some application. | ||
// The main issue is `back/forward` buttons of browsers, because they invoke | ||
// `history.back|forward` which dispatch `popstate` event. Since `single-spa` | ||
// overrides `history.replaceState` Angular's zone cannot intercept this event. | ||
// Only the root zone is able to intercept all events. | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
this.ngZone.run(() => fn(event)); | ||
fn(event); | ||
} | ||
}; | ||
this.storeOnPopStateListener(zone, onPopStateListener); | ||
super.onPopState(onPopStateListener); | ||
} | ||
storeOnPopStateListener(zone, onPopStateListener) { | ||
// All listeners should be stored inside an array because the `onPopState` can be called | ||
// multiple times thus we wanna reference all listeners to remove them further. | ||
this.onPopStateListeners.push(onPopStateListener); | ||
super.onPopState(onPopStateListener); | ||
const onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone) || []; | ||
onPopStateListeners.push(onPopStateListener); | ||
if (!this.zoneToOnPopStateListenersMap.has(zone)) { | ||
this.zoneToOnPopStateListenersMap.set(zone, onPopStateListeners); | ||
} | ||
} | ||
setNgZone(ngZone) { | ||
this.ngZone = ngZone; | ||
} | ||
}; | ||
SingleSpaPlatformLocation = __decorate([ | ||
Injectable() | ||
], SingleSpaPlatformLocation); | ||
} | ||
SingleSpaPlatformLocation.decorators = [ | ||
{ type: Injectable } | ||
]; | ||
/** | ||
@@ -91,8 +120,8 @@ * The `PlatformLocation` class is an "injectee" of the `PathLocationStrategy`, | ||
const defaultOpts = { | ||
// Required opts that will be set by the library consumer. | ||
const defaultOptions = { | ||
// Required options that will be set by the library consumer. | ||
NgZone: null, | ||
bootstrapFunction: null, | ||
template: null, | ||
// optional opts | ||
// Optional options | ||
Router: undefined, | ||
@@ -103,24 +132,24 @@ domElementGetter: undefined, | ||
}; | ||
function singleSpaAngular(userOpts) { | ||
if (typeof userOpts !== 'object') { | ||
function singleSpaAngular(userOptions) { | ||
if (typeof userOptions !== 'object') { | ||
throw Error('single-spa-angular requires a configuration object'); | ||
} | ||
const opts = Object.assign(Object.assign({}, defaultOpts), userOpts); | ||
if (typeof opts.bootstrapFunction !== 'function') { | ||
throw Error('single-spa-angular must be passed an opts.bootstrapFunction'); | ||
const options = Object.assign(Object.assign({}, defaultOptions), userOptions); | ||
if (typeof options.bootstrapFunction !== 'function') { | ||
throw Error('single-spa-angular must be passed an options.bootstrapFunction'); | ||
} | ||
if (typeof opts.template !== 'string') { | ||
throw Error('single-spa-angular must be passed opts.template string'); | ||
if (typeof options.template !== 'string') { | ||
throw Error('single-spa-angular must be passed options.template string'); | ||
} | ||
if (!opts.NgZone) { | ||
throw Error(`single-spa-angular must be passed the NgZone opt`); | ||
if (!options.NgZone) { | ||
throw Error(`single-spa-angular must be passed the NgZone option`); | ||
} | ||
return { | ||
bootstrap: bootstrap.bind(null, opts), | ||
mount: mount.bind(null, opts), | ||
unmount: unmount.bind(null, opts), | ||
update: opts.updateFunction, | ||
bootstrap: bootstrap.bind(null, options), | ||
mount: mount.bind(null, options), | ||
unmount: unmount.bind(null, options), | ||
update: options.updateFunction, | ||
}; | ||
} | ||
function bootstrap(opts, props) { | ||
function bootstrap(options, props) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
@@ -130,7 +159,7 @@ // Angular provides an opportunity to develop `zone-less` application, where developers | ||
// See https://angular.io/guide/zone#noopzone | ||
if (opts.NgZone === 'noop') { | ||
if (options.NgZone === 'noop') { | ||
return; | ||
} | ||
// In order for multiple Angular apps to work concurrently on a page, they each need a unique identifier. | ||
opts.zoneIdentifier = `single-spa-angular:${props.name || props.appName}`; | ||
options.zoneIdentifier = `single-spa-angular:${props.name || props.appName}`; | ||
// This is a hack, since NgZone doesn't allow you to configure the property that identifies your zone. | ||
@@ -141,8 +170,8 @@ // See https://github.com/PlaceMe-SAS/single-spa-angular-cli/issues/33, | ||
// and https://github.com/angular/angular/blob/a14dc2d7a4821a19f20a9547053a5734798f541e/packages/core/src/zone/ng_zone.ts#L257 | ||
opts.NgZone.isInAngularZone = function () { | ||
options.NgZone.isInAngularZone = () => { | ||
// @ts-ignore | ||
return window.Zone.current._properties[opts.zoneIdentifier] === true; | ||
return window.Zone.current._properties[options.zoneIdentifier] === true; | ||
}; | ||
opts.routingEventListener = () => { | ||
opts.bootstrappedNgZone.run(() => { | ||
options.routingEventListener = () => { | ||
options.bootstrappedNgZone.run(() => { | ||
// See https://github.com/single-spa/single-spa-angular/issues/86 | ||
@@ -155,20 +184,15 @@ // Zone is unaware of the single-spa navigation change and so Angular change detection doesn't work | ||
} | ||
function mount(opts, props) { | ||
function mount(options, props) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
const domElementGetter = chooseDomElementGetter(opts, props); | ||
if (!domElementGetter) { | ||
throw Error(`cannot mount angular application '${props.name || props.appName}' without a domElementGetter provided either as an opt or a prop`); | ||
} | ||
const containerEl = getContainerEl(domElementGetter); | ||
containerEl.innerHTML = opts.template; | ||
const bootstrapPromise = opts.bootstrapFunction(props); | ||
getContainerElementAndSetTemplate(options, props); | ||
const bootstrapPromise = options.bootstrapFunction(props); | ||
if (!(bootstrapPromise instanceof Promise)) { | ||
throw Error(`single-spa-angular: the opts.bootstrapFunction must return a promise, but instead returned a '${typeof bootstrapPromise}' that is not a Promise`); | ||
throw Error(`single-spa-angular: the options.bootstrapFunction must return a promise, but instead returned a '${typeof bootstrapPromise}' that is not a Promise`); | ||
} | ||
const module = yield bootstrapPromise; | ||
if (!module || typeof module.destroy !== 'function') { | ||
throw Error(`single-spa-angular: the opts.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?`); | ||
throw Error(`single-spa-angular: the options.bootstrapFunction returned a promise that did not resolve with a valid Angular module. Did you call platformBrowserDynamic().bootstrapModule() correctly?`); | ||
} | ||
const singleSpaPlatformLocation = module.injector.get(SingleSpaPlatformLocation, null); | ||
const ngZoneEnabled = opts.NgZone !== 'noop'; | ||
const ngZoneEnabled = options.NgZone !== 'noop'; | ||
// The user has to provide `BrowserPlatformLocation` only if his application uses routing. | ||
@@ -178,3 +202,3 @@ // So if he provided `Router` but didn't provide `BrowserPlatformLocation` then we have to inform him. | ||
// `zone-less` change detection, if `NgZone` is `noop` then we can skip it. | ||
if (ngZoneEnabled && opts.Router && singleSpaPlatformLocation === null) { | ||
if (ngZoneEnabled && options.Router && singleSpaPlatformLocation === null) { | ||
throw new Error(` | ||
@@ -184,18 +208,18 @@ single-spa-angular: could not retrieve extra providers from the platform injector. Did you call platformBrowserDynamic(getSingleSpaExtraProviders()).bootstrapModule()? | ||
} | ||
const bootstrappedOpts = opts; | ||
const bootstrappedOptions = options; | ||
if (ngZoneEnabled) { | ||
const ngZone = module.injector.get(opts.NgZone); | ||
const ngZone = module.injector.get(options.NgZone); | ||
const zoneIdentifier = bootstrappedOptions.zoneIdentifier; | ||
// `NgZone` can be enabled but routing may not be used thus `getSingleSpaExtraProviders()` | ||
// function was not called. | ||
if (singleSpaPlatformLocation !== null) { | ||
singleSpaPlatformLocation.setNgZone(ngZone); | ||
// Cleanup resources, especially remove event listeners thus they will not be added | ||
// twice when application gets bootstrapped the second time. | ||
module.onDestroy(() => singleSpaPlatformLocation.destroy()); | ||
module.onDestroy(() => singleSpaPlatformLocation.destroyApplication(zoneIdentifier)); | ||
} | ||
bootstrappedOpts.bootstrappedNgZone = ngZone; | ||
bootstrappedOpts.bootstrappedNgZone['_inner']._properties[bootstrappedOpts.zoneIdentifier] = true; | ||
window.addEventListener('single-spa:routing-event', bootstrappedOpts.routingEventListener); | ||
bootstrappedOptions.bootstrappedNgZone = ngZone; | ||
bootstrappedOptions.bootstrappedNgZone['_inner']._properties[zoneIdentifier] = true; | ||
window.addEventListener('single-spa:routing-event', bootstrappedOptions.routingEventListener); | ||
} | ||
bootstrappedOpts.bootstrappedModule = module; | ||
bootstrappedOptions.bootstrappedModule = module; | ||
return module; | ||
@@ -205,13 +229,13 @@ }); | ||
// eslint-disable-next-line @typescript-eslint/no-unused-vars | ||
function unmount(opts, props) { | ||
function unmount(options, props) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
if (opts.Router) { | ||
if (options.Router) { | ||
// Workaround for https://github.com/angular/angular/issues/19079 | ||
const router = opts.bootstrappedModule.injector.get(opts.Router); | ||
const router = options.bootstrappedModule.injector.get(options.Router); | ||
router.dispose(); | ||
} | ||
if (opts.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', opts.routingEventListener); | ||
if (options.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', options.routingEventListener); | ||
} | ||
if (opts.AnimationEngine) { | ||
if (options.AnimationEngine) { | ||
/* | ||
@@ -241,14 +265,14 @@ The BrowserAnimationsModule does not clean up after itself :'(. When you unmount/destroy the main module, the | ||
*/ | ||
const animationEngine = opts.bootstrappedModule.injector.get(opts.AnimationEngine); | ||
const animationEngine = options.bootstrappedModule.injector.get(options.AnimationEngine); | ||
animationEngine._transitionEngine.flush(); | ||
} | ||
opts.bootstrappedModule.destroy(); | ||
delete opts.bootstrappedModule; | ||
options.bootstrappedModule.destroy(); | ||
delete options.bootstrappedModule; | ||
// This is an issue. Issue has been created and Angular team is working on the fix: | ||
// https://github.com/angular/angular/issues/36449 | ||
removeApplicationFromDOMIfIvyEnabled(opts, props); | ||
removeApplicationFromDOMIfIvyEnabled(options, props); | ||
}); | ||
} | ||
let ParcelComponent = class ParcelComponent { | ||
class ParcelComponent { | ||
// eslint-disable-next-line @typescript-eslint/no-parameter-properties | ||
@@ -346,44 +370,32 @@ constructor(host) { | ||
} | ||
}; | ||
} | ||
ParcelComponent.decorators = [ | ||
{ type: Component, args: [{ | ||
selector: 'parcel', | ||
template: '<div></div>' | ||
},] } | ||
]; | ||
ParcelComponent.ctorParameters = () => [ | ||
{ type: ElementRef } | ||
]; | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "config", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "mountParcel", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "onParcelMount", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "wrapWith", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "customProps", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "appendTo", void 0); | ||
__decorate([ | ||
Input() | ||
], ParcelComponent.prototype, "handleError", void 0); | ||
ParcelComponent = __decorate([ | ||
Component({ | ||
selector: 'parcel', | ||
template: '<div></div>' | ||
}) | ||
], ParcelComponent); | ||
let ParcelModule = class ParcelModule { | ||
ParcelComponent.propDecorators = { | ||
config: [{ type: Input }], | ||
mountParcel: [{ type: Input }], | ||
onParcelMount: [{ type: Input }], | ||
wrapWith: [{ type: Input }], | ||
customProps: [{ type: Input }], | ||
appendTo: [{ type: Input }], | ||
handleError: [{ type: Input }] | ||
}; | ||
ParcelModule = __decorate([ | ||
NgModule({ | ||
declarations: [ParcelComponent], | ||
exports: [ParcelComponent], | ||
entryComponents: [ParcelComponent], | ||
}) | ||
], ParcelModule); | ||
class ParcelModule { | ||
} | ||
ParcelModule.decorators = [ | ||
{ type: NgModule, args: [{ | ||
declarations: [ParcelComponent], | ||
exports: [ParcelComponent], | ||
entryComponents: [ParcelComponent], | ||
},] } | ||
]; | ||
/** | ||
@@ -390,0 +402,0 @@ * The public api for consumers of single-spa-angular |
@@ -1,4 +0,3 @@ | ||
import { SingleSpaAngularOpts, DomElementGetter } from './types'; | ||
export declare function removeApplicationFromDOMIfIvyEnabled(opts: SingleSpaAngularOpts, props: any): void; | ||
export declare function getContainerEl(domElementGetter: DomElementGetter): never | HTMLElement; | ||
export declare function chooseDomElementGetter(opts: SingleSpaAngularOpts, props: any): DomElementGetter; | ||
import { BaseSingleSpaAngularOptions } from './types'; | ||
export declare function removeApplicationFromDOMIfIvyEnabled<T extends BaseSingleSpaAngularOptions>(options: T, props: any): void; | ||
export declare function getContainerElementAndSetTemplate<T extends BaseSingleSpaAngularOptions>(options: T, props: any): HTMLElement; |
@@ -1,2 +0,2 @@ | ||
export { SingleSpaAngularOpts, BootstrappedSingleSpaAngularOpts } from './types'; | ||
export { getContainerEl, chooseDomElementGetter, removeApplicationFromDOMIfIvyEnabled, } from './dom'; | ||
export * from './types'; | ||
export { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled } from './dom'; |
{ | ||
"main": "../bundles/single-spa-angular-internals.umd.js", | ||
"module": "../fesm5/single-spa-angular-internals.js", | ||
"module": "../fesm2015/single-spa-angular-internals.js", | ||
"es2015": "../fesm2015/single-spa-angular-internals.js", | ||
"esm5": "../esm5/internals/single-spa-angular-internals.js", | ||
"esm2015": "../esm2015/internals/single-spa-angular-internals.js", | ||
"fesm5": "../fesm5/single-spa-angular-internals.js", | ||
"fesm2015": "../fesm2015/single-spa-angular-internals.js", | ||
@@ -9,0 +7,0 @@ "typings": "single-spa-angular-internals.d.ts", |
@@ -1,1 +0,1 @@ | ||
{"__symbolic":"module","version":4,"metadata":{"SingleSpaAngularOpts":{"__symbolic":"interface"},"BootstrappedSingleSpaAngularOpts":{"__symbolic":"interface"},"getContainerEl":{"__symbolic":"function"},"chooseDomElementGetter":{"__symbolic":"function"},"removeApplicationFromDOMIfIvyEnabled":{"__symbolic":"function"}},"origins":{"SingleSpaAngularOpts":"./types","BootstrappedSingleSpaAngularOpts":"./types","getContainerEl":"./dom","chooseDomElementGetter":"./dom","removeApplicationFromDOMIfIvyEnabled":"./dom"},"importAs":"single-spa-angular/internals"} | ||
{"__symbolic":"module","version":4,"metadata":{"DomElementGetter":{"__symbolic":"interface"},"BaseSingleSpaAngularOptions":{"__symbolic":"interface"},"getContainerElementAndSetTemplate":{"__symbolic":"function"},"removeApplicationFromDOMIfIvyEnabled":{"__symbolic":"function"}},"origins":{"DomElementGetter":"./types","BaseSingleSpaAngularOptions":"./types","getContainerElementAndSetTemplate":"./dom","removeApplicationFromDOMIfIvyEnabled":"./dom"},"importAs":"single-spa-angular/internals"} |
@@ -1,18 +0,8 @@ | ||
import { NgModuleRef, Type } from '@angular/core'; | ||
import { AppProps } from 'single-spa'; | ||
import { NgModuleRef } from '@angular/core'; | ||
export declare type DomElementGetter = () => HTMLElement; | ||
export interface SingleSpaAngularOpts { | ||
NgZone: typeof import('@angular/core').NgZone | 'noop'; | ||
bootstrapFunction(props: AppProps): Promise<NgModuleRef<any>>; | ||
updateFunction?(props: AppProps): Promise<any>; | ||
export interface BaseSingleSpaAngularOptions { | ||
template: string; | ||
Router?: Type<any>; | ||
domElementGetter?(): HTMLElement; | ||
AnimationEngine?: Type<any>; | ||
bootstrapFunction(props: AppProps): Promise<NgModuleRef<any>>; | ||
} | ||
export interface BootstrappedSingleSpaAngularOpts extends SingleSpaAngularOpts { | ||
bootstrappedModule: NgModuleRef<any>; | ||
bootstrappedNgZone?: import('@angular/core').NgZone; | ||
routingEventListener?: () => void; | ||
zoneIdentifier?: string; | ||
} |
@@ -70,3 +70,4 @@ "use strict"; | ||
if (rule.use) { | ||
const cssMiniExtractIndex = rule.use.findIndex((use) => typeof use === 'string' && use.includes('mini-css-extract-plugin')); | ||
const cssMiniExtractIndex = rule.use.findIndex((use) => (typeof use === 'string' && use.includes('mini-css-extract-plugin')) || | ||
(typeof use === 'object' && use.loader && use.loader.includes('mini-css-extract-plugin'))); | ||
if (cssMiniExtractIndex >= 0) { | ||
@@ -73,0 +74,0 @@ rule.use[cssMiniExtractIndex] = { loader: 'style-loader' }; |
{ | ||
"$schema": "../node_modules/ng-packagr/ng-package.schema.json", | ||
"name": "single-spa-angular", | ||
"version": "4.3.2", | ||
"version": "4.4.1", | ||
"description": "Helpers for building single-spa applications which use Angular 2", | ||
@@ -28,14 +28,14 @@ "schematics": "./schematics/schematics.json", | ||
"@angular/core": ">=9", | ||
"single-spa": ">=4.0.0", | ||
"tslib": "^1.10.0" | ||
"single-spa": ">=4.0.0" | ||
}, | ||
"main": "bundles/single-spa-angular.umd.js", | ||
"module": "fesm5/single-spa-angular.js", | ||
"module": "fesm2015/single-spa-angular.js", | ||
"es2015": "fesm2015/single-spa-angular.js", | ||
"esm5": "esm5/single-spa-angular.js", | ||
"esm2015": "esm2015/single-spa-angular.js", | ||
"fesm5": "fesm5/single-spa-angular.js", | ||
"fesm2015": "fesm2015/single-spa-angular.js", | ||
"typings": "single-spa-angular.d.ts", | ||
"metadata": "single-spa-angular.metadata.json" | ||
"metadata": "single-spa-angular.metadata.json", | ||
"dependencies": { | ||
"tslib": "^2.0.0" | ||
} | ||
} |
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.addScripts = void 0; | ||
const DEFAULT_PORT = 4200; | ||
@@ -4,0 +5,0 @@ /** |
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.getAngularBuildersCustomWebpackDependency = exports.getSingleSpaAngularDependency = exports.getSingleSpaDependency = void 0; | ||
const dependencies_1 = require("@schematics/angular/utility/dependencies"); | ||
@@ -4,0 +5,0 @@ const { version, peerDependencies, dependencies } = require('../../package.json'); |
@@ -1,2 +0,2 @@ | ||
import { Rule, SchematicContext } from '@angular-devkit/schematics'; | ||
import { Rule, SchematicContext, Tree } from '@angular-devkit/schematics'; | ||
import { Schema as NgAddOptions } from './schema'; | ||
@@ -6,3 +6,3 @@ export default function (options: NgAddOptions): Rule; | ||
export declare function createMainEntry(options: NgAddOptions): Rule; | ||
export declare function updateConfiguration(options: NgAddOptions): (host: import("@angular-devkit/schematics/src/tree/interface").Tree, context: SchematicContext) => import("@angular-devkit/schematics/src/tree/interface").Tree; | ||
export declare function updateConfiguration(options: NgAddOptions): (host: Tree, context: SchematicContext) => import("@angular-devkit/schematics/src/tree/interface").Tree; | ||
export declare function addNPMScripts(options: NgAddOptions): Rule; |
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.addNPMScripts = exports.updateConfiguration = exports.createMainEntry = exports.addDependencies = void 0; | ||
const schematics_1 = require("@angular-devkit/schematics"); | ||
@@ -4,0 +5,0 @@ const config_1 = require("@schematics/angular/utility/config"); |
@@ -1,1 +0,1 @@ | ||
{"__symbolic":"module","version":4,"metadata":{"ɵa":{"__symbolic":"class","extends":{"__symbolic":"reference","module":"@angular/common","name":"ɵBrowserPlatformLocation","line":9,"character":47},"decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Injectable","line":8,"character":1}}],"members":{"destroy":[{"__symbolic":"method"}],"pushState":[{"__symbolic":"method"}],"replaceState":[{"__symbolic":"method"}],"onPopState":[{"__symbolic":"method"}],"setNgZone":[{"__symbolic":"method"}]}},"singleSpaAngular":{"__symbolic":"function"},"getSingleSpaExtraProviders":{"__symbolic":"function","parameters":[],"value":[{"provide":{"__symbolic":"reference","name":"ɵa"},"useClass":{"__symbolic":"reference","name":"ɵa"},"deps":[[{"__symbolic":"new","expression":{"__symbolic":"reference","module":"@angular/core","name":"Inject","line":87,"character":18},"arguments":[{"__symbolic":"reference","module":"@angular/common","name":"DOCUMENT","line":87,"character":25}]}]]},{"provide":{"__symbolic":"reference","module":"@angular/common","name":"PlatformLocation","line":90,"character":15},"useExisting":{"__symbolic":"reference","name":"ɵa"}}]},"ParcelModule":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"NgModule","line":3,"character":1},"arguments":[{"declarations":[{"__symbolic":"reference","name":"ParcelComponent"}],"exports":[{"__symbolic":"reference","name":"ParcelComponent"}],"entryComponents":[{"__symbolic":"reference","name":"ParcelComponent"}]}]}],"members":{}},"ParcelComponent":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Component","line":3,"character":1},"arguments":[{"selector":"parcel","template":"<div></div>"}]}],"members":{"config":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":8,"character":3}}]}],"mountParcel":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":9,"character":3}}]}],"onParcelMount":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":10,"character":3}}]}],"wrapWith":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":11,"character":3}}]}],"customProps":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":12,"character":3}}]}],"appendTo":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":13,"character":3}}]}],"handleError":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":14,"character":3}}]}],"__ctor__":[{"__symbolic":"constructor","parameters":[{"__symbolic":"reference","name":"ElementRef","module":"@angular/core","arguments":[{"__symbolic":"error","message":"Could not resolve type","line":23,"character":39,"context":{"typeName":"HTMLElement"},"module":"./src/parcel-lib/parcel.component"}]}]}],"ngOnInit":[{"__symbolic":"method"}],"ngOnChanges":[{"__symbolic":"method"}],"ngOnDestroy":[{"__symbolic":"method"}],"addThingToDo":[{"__symbolic":"method"}]}}},"origins":{"ɵa":"./src/extra-providers","singleSpaAngular":"./src/single-spa-angular","getSingleSpaExtraProviders":"./src/extra-providers","ParcelModule":"./src/parcel-lib/index","ParcelComponent":"./src/parcel-lib/parcel.component"},"importAs":"single-spa-angular"} | ||
{"__symbolic":"module","version":4,"metadata":{"ɵa":{"__symbolic":"class","extends":{"__symbolic":"reference","module":"@angular/common","name":"ɵBrowserPlatformLocation","line":13,"character":47},"decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Injectable","line":12,"character":1}}],"members":{"destroyApplication":[{"__symbolic":"method"}],"pushState":[{"__symbolic":"method"}],"replaceState":[{"__symbolic":"method"}],"onPopState":[{"__symbolic":"method"}],"storeOnPopStateListener":[{"__symbolic":"method"}]}},"singleSpaAngular":{"__symbolic":"function"},"getSingleSpaExtraProviders":{"__symbolic":"function","parameters":[],"value":[{"provide":{"__symbolic":"reference","name":"ɵa"},"useClass":{"__symbolic":"reference","name":"ɵa"},"deps":[[{"__symbolic":"new","expression":{"__symbolic":"reference","module":"@angular/core","name":"Inject","line":128,"character":18},"arguments":[{"__symbolic":"reference","module":"@angular/common","name":"DOCUMENT","line":128,"character":25}]}]]},{"provide":{"__symbolic":"reference","module":"@angular/common","name":"PlatformLocation","line":131,"character":15},"useExisting":{"__symbolic":"reference","name":"ɵa"}}]},"ParcelModule":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"NgModule","line":3,"character":1},"arguments":[{"declarations":[{"__symbolic":"reference","name":"ParcelComponent"}],"exports":[{"__symbolic":"reference","name":"ParcelComponent"}],"entryComponents":[{"__symbolic":"reference","name":"ParcelComponent"}]}]}],"members":{}},"ParcelComponent":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Component","line":3,"character":1},"arguments":[{"selector":"parcel","template":"<div></div>"}]}],"members":{"config":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":8,"character":3}}]}],"mountParcel":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":9,"character":3}}]}],"onParcelMount":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":10,"character":3}}]}],"wrapWith":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":11,"character":3}}]}],"customProps":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":12,"character":3}}]}],"appendTo":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":13,"character":3}}]}],"handleError":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":14,"character":3}}]}],"__ctor__":[{"__symbolic":"constructor","parameters":[{"__symbolic":"reference","name":"ElementRef","module":"@angular/core","arguments":[{"__symbolic":"error","message":"Could not resolve type","line":23,"character":39,"context":{"typeName":"HTMLElement"},"module":"./src/parcel-lib/parcel.component"}]}]}],"ngOnInit":[{"__symbolic":"method"}],"ngOnChanges":[{"__symbolic":"method"}],"ngOnDestroy":[{"__symbolic":"method"}],"addThingToDo":[{"__symbolic":"method"}]}}},"origins":{"ɵa":"./src/extra-providers","singleSpaAngular":"./src/single-spa-angular","getSingleSpaExtraProviders":"./src/extra-providers","ParcelModule":"./src/parcel-lib/index","ParcelComponent":"./src/parcel-lib/parcel.component"},"importAs":"single-spa-angular"} |
@@ -1,12 +0,13 @@ | ||
import { NgZone, StaticProvider } from '@angular/core'; | ||
import { StaticProvider } from '@angular/core'; | ||
import { ɵBrowserPlatformLocation, LocationChangeEvent } from '@angular/common'; | ||
declare type OnPopStateListener = (event: LocationChangeEvent) => void; | ||
export declare class SingleSpaPlatformLocation extends ɵBrowserPlatformLocation { | ||
private ngZone; | ||
private skipNextPopState; | ||
private onPopStateListeners; | ||
destroy(): void; | ||
private zoneToOnPopStateListenersMap; | ||
private source; | ||
destroyApplication(zoneIdentifier: string): void; | ||
pushState(state: any, title: string, url: string): void; | ||
replaceState(state: any, title: string, url: string): void; | ||
onPopState(fn: (event: LocationChangeEvent) => void): void; | ||
setNgZone(ngZone: NgZone): void; | ||
onPopState(fn: OnPopStateListener): void; | ||
private storeOnPopStateListener; | ||
} | ||
@@ -19,1 +20,2 @@ /** | ||
export declare function getSingleSpaExtraProviders(): StaticProvider[]; | ||
export {}; |
import { LifeCycles } from 'single-spa'; | ||
import { SingleSpaAngularOpts } from 'single-spa-angular/internals'; | ||
export declare function singleSpaAngular(userOpts: SingleSpaAngularOpts): LifeCycles; | ||
import { SingleSpaAngularOptions } from './types'; | ||
export declare function singleSpaAngular(userOptions: SingleSpaAngularOptions): LifeCycles; |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
Minified code
QualityThis package contains minified code. This may be harmless in some cases where minified code is included in packaged libraries, however packages on npm should not minify code.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
78
412208
2869
3
1
+ Addedtslib@^2.0.0
- Removedtslib@1.14.1(transitive)