single-spa-angular
Advanced tools
Comparing version 4.9.2 to 5.0.0
@@ -31,2 +31,4 @@ (function (global, factory) { | ||
function __extends(d, b) { | ||
if (typeof b !== "function" && b !== null) | ||
throw new TypeError("Class extends value " + String(b) + " is not a constructor or null"); | ||
extendStatics(d, b); | ||
@@ -217,2 +219,3 @@ function __() { this.constructor = d; } | ||
} | ||
/** @deprecated */ | ||
function __spread() { | ||
@@ -223,2 +226,3 @@ for (var ar = [], i = 0; i < arguments.length; i++) | ||
} | ||
/** @deprecated */ | ||
function __spreadArrays() { | ||
@@ -232,3 +236,7 @@ for (var s = 0, i = 0, il = arguments.length; i < il; i++) | ||
} | ||
; | ||
function __spreadArray(to, from) { | ||
for (var i = 0, il = from.length, j = to.length; i < il; i++, j++) | ||
to[j] = from[i]; | ||
return to; | ||
} | ||
function __await(v) { | ||
@@ -298,14 +306,17 @@ return this instanceof __await ? (this.v = v, this) : new __await(v); | ||
} | ||
function __classPrivateFieldGet(receiver, privateMap) { | ||
if (!privateMap.has(receiver)) { | ||
throw new TypeError("attempted to get private field on non-instance"); | ||
} | ||
return privateMap.get(receiver); | ||
function __classPrivateFieldGet(receiver, state, kind, f) { | ||
if (kind === "a" && !f) | ||
throw new TypeError("Private accessor was defined without a getter"); | ||
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) | ||
throw new TypeError("Cannot read private member from an object whose class did not declare it"); | ||
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver); | ||
} | ||
function __classPrivateFieldSet(receiver, privateMap, value) { | ||
if (!privateMap.has(receiver)) { | ||
throw new TypeError("attempted to set private field on non-instance"); | ||
} | ||
privateMap.set(receiver, value); | ||
return value; | ||
function __classPrivateFieldSet(receiver, state, value, kind, f) { | ||
if (kind === "m") | ||
throw new TypeError("Private method is not writable"); | ||
if (kind === "a" && !f) | ||
throw new TypeError("Private accessor was defined without a setter"); | ||
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) | ||
throw new TypeError("Cannot write private member to an object whose class did not declare it"); | ||
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value; | ||
} | ||
@@ -312,0 +323,0 @@ |
@@ -7,13 +7,2 @@ (function (global, factory) { | ||
function removeApplicationFromDOMIfIvyEnabled(options, props) { | ||
if (ivyEnabled()) { | ||
var domElementGetter = chooseDomElementGetter(options, props); | ||
var domElement = getContainerElement(domElementGetter); | ||
// View Engine removes all nodes automatically when calling `NgModuleRef.destroy()`, | ||
// which calls `ComponentRef.destroy()`. | ||
// Basically this will remove `app-root` or any other selector from the container element. | ||
while (domElement.firstChild) | ||
domElement.removeChild(domElement.firstChild); | ||
} | ||
} | ||
function getContainerElementAndSetTemplate(options, props) { | ||
@@ -63,16 +52,2 @@ var domElementGetter = chooseDomElementGetter(options, props); | ||
} | ||
function ivyEnabled() { | ||
try { | ||
// `ɵivyEnabled` variable is exposed starting from version 8. | ||
// We use `require` here except of a single `import { ɵivyEnabled }` because the | ||
// developer can use Angular version that doesn't expose it (all versions <8). | ||
// The `catch` statement will handle those cases. | ||
// eslint-disable-next-line | ||
var ɵivyEnabled = require('@angular/core').ɵivyEnabled; | ||
return !!ɵivyEnabled; | ||
} | ||
catch (_a) { | ||
return false; | ||
} | ||
} | ||
@@ -84,3 +59,2 @@ /** | ||
exports.getContainerElementAndSetTemplate = getContainerElementAndSetTemplate; | ||
exports.removeApplicationFromDOMIfIvyEnabled = removeApplicationFromDOMIfIvyEnabled; | ||
@@ -87,0 +61,0 @@ Object.defineProperty(exports, '__esModule', { value: true }); |
@@ -104,2 +104,3 @@ (function (global, factory) { | ||
]; | ||
/** @nocollapse */ | ||
ParcelComponent.ctorParameters = function () { return [ | ||
@@ -106,0 +107,0 @@ { type: core.ElementRef } |
@@ -31,2 +31,4 @@ (function (global, factory) { | ||
function __extends(d, b) { | ||
if (typeof b !== "function" && b !== null) | ||
throw new TypeError("Class extends value " + String(b) + " is not a constructor or null"); | ||
extendStatics(d, b); | ||
@@ -217,2 +219,3 @@ function __() { this.constructor = d; } | ||
} | ||
/** @deprecated */ | ||
function __spread() { | ||
@@ -223,2 +226,3 @@ for (var ar = [], i = 0; i < arguments.length; i++) | ||
} | ||
/** @deprecated */ | ||
function __spreadArrays() { | ||
@@ -232,3 +236,7 @@ for (var s = 0, i = 0, il = arguments.length; i < il; i++) | ||
} | ||
; | ||
function __spreadArray(to, from) { | ||
for (var i = 0, il = from.length, j = to.length; i < il; i++, j++) | ||
to[j] = from[i]; | ||
return to; | ||
} | ||
function __await(v) { | ||
@@ -298,14 +306,17 @@ return this instanceof __await ? (this.v = v, this) : new __await(v); | ||
} | ||
function __classPrivateFieldGet(receiver, privateMap) { | ||
if (!privateMap.has(receiver)) { | ||
throw new TypeError("attempted to get private field on non-instance"); | ||
} | ||
return privateMap.get(receiver); | ||
function __classPrivateFieldGet(receiver, state, kind, f) { | ||
if (kind === "a" && !f) | ||
throw new TypeError("Private accessor was defined without a getter"); | ||
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) | ||
throw new TypeError("Cannot read private member from an object whose class did not declare it"); | ||
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver); | ||
} | ||
function __classPrivateFieldSet(receiver, privateMap, value) { | ||
if (!privateMap.has(receiver)) { | ||
throw new TypeError("attempted to set private field on non-instance"); | ||
} | ||
privateMap.set(receiver, value); | ||
return value; | ||
function __classPrivateFieldSet(receiver, state, value, kind, f) { | ||
if (kind === "m") | ||
throw new TypeError("Private method is not writable"); | ||
if (kind === "a" && !f) | ||
throw new TypeError("Private accessor was defined without a setter"); | ||
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) | ||
throw new TypeError("Cannot write private member to an object whose class did not declare it"); | ||
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value; | ||
} | ||
@@ -316,55 +327,9 @@ | ||
function SingleSpaPlatformLocation() { | ||
var _this = _super.apply(this, __spread(arguments)) || this; | ||
var _this = _super.apply(this, __spreadArray([], __read(arguments))) || this; | ||
// This is a simple marker that helps us to ignore PopStateEvents | ||
// that was not dispatched by the browser. | ||
_this.skipNextPopState = false; | ||
// 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; | ||
_this.source = 'Window.addEventListener:popstate'; | ||
return _this; | ||
} | ||
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. | ||
// Basically, Angular has a potentional memory leak. The `ɵBrowserPlatformLocation` | ||
// has `onPopState` method which adds `popstate` event listener and forgets, see here: | ||
// https://github.com/angular/angular/blob/14be55c9facf3e47b8c97df4502dc3f0f897da03/packages/common/src/location/platform_location.ts#L126 | ||
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 { | ||
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); | ||
} | ||
} | ||
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; } | ||
} | ||
} | ||
this.zoneToOnPopStateListenersMap.delete(zone); | ||
}; | ||
SingleSpaPlatformLocation.prototype.pushState = function (state, title, url) { | ||
@@ -389,3 +354,3 @@ this.skipNextPopState = true; | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
fn = zone.wrap(fn, "" + this.source++); | ||
fn = zone.wrap(fn, this.source); | ||
var onPopStateListener = function (event) { | ||
@@ -404,14 +369,4 @@ // The `LocationChangeEvent` doesn't have the `singleSpa` property, since it's added | ||
}; | ||
this.storeOnPopStateListener(zone, onPopStateListener); | ||
_super.prototype.onPopState.call(this, onPopStateListener); | ||
return _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. | ||
var onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone) || []; | ||
onPopStateListeners.push(onPopStateListener); | ||
if (!this.zoneToOnPopStateListenersMap.has(zone)) { | ||
this.zoneToOnPopStateListenersMap.set(zone, onPopStateListeners); | ||
} | ||
}; | ||
return SingleSpaPlatformLocation; | ||
@@ -431,3 +386,2 @@ }(common.ɵBrowserPlatformLocation)); | ||
provide: SingleSpaPlatformLocation, | ||
useClass: SingleSpaPlatformLocation, | ||
deps: [[new core.Inject(common.DOCUMENT)]], | ||
@@ -450,3 +404,2 @@ }, | ||
domElementGetter: undefined, | ||
AnimationEngine: undefined, | ||
updateFunction: function () { return Promise.resolve(); }, | ||
@@ -514,3 +467,3 @@ bootstrappedModule: null, | ||
return __awaiter(this, void 0, void 0, function () { | ||
var bootstrapPromise, module, singleSpaPlatformLocation, ngZoneEnabled, bootstrappedOptions, ngZone, zoneIdentifier_1; | ||
var bootstrapPromise, module, singleSpaPlatformLocation, ngZoneEnabled, bootstrappedOptions, ngZone, zoneIdentifier; | ||
return __generator(this, function (_a) { | ||
@@ -542,3 +495,3 @@ switch (_a.label) { | ||
ngZone = module.injector.get(options.NgZone); | ||
zoneIdentifier_1 = bootstrappedOptions.zoneIdentifier; | ||
zoneIdentifier = bootstrappedOptions.zoneIdentifier; | ||
// `NgZone` can be enabled but routing may not be used thus `getSingleSpaExtraProviders()` | ||
@@ -548,10 +501,5 @@ // function was not called. | ||
skipLocationChangeOnNonImperativeRoutingTriggers(module, options); | ||
// Cleanup resources, especially remove event listeners thus they will not be added | ||
// twice when application gets bootstrapped the second time. | ||
module.onDestroy(function () { | ||
singleSpaPlatformLocation.destroyApplication(zoneIdentifier_1); | ||
}); | ||
} | ||
bootstrappedOptions.bootstrappedNgZone = ngZone; | ||
bootstrappedOptions.bootstrappedNgZone['_inner']._properties[zoneIdentifier_1] = true; | ||
bootstrappedOptions.bootstrappedNgZone['_inner']._properties[zoneIdentifier] = true; | ||
window.addEventListener('single-spa:routing-event', bootstrappedOptions.routingEventListener); | ||
@@ -565,24 +513,9 @@ } | ||
} | ||
// eslint-disable-next-line @typescript-eslint/no-unused-vars | ||
function unmount(options, props) { | ||
return __awaiter(this, void 0, void 0, function () { | ||
var router, animationEngine; | ||
return __generator(this, function (_a) { | ||
if (options.Router) { | ||
router = options.bootstrappedModule.injector.get(options.Router); | ||
router.dispose(); | ||
} | ||
if (options.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', options.routingEventListener); | ||
} | ||
if (options.AnimationEngine) { | ||
animationEngine = options.bootstrappedModule.injector.get(options.AnimationEngine); | ||
animationEngine._transitionEngine.flush(); | ||
} | ||
options.bootstrappedModule.destroy(); | ||
options.bootstrappedModule = null; | ||
// TODO: this is not an issue anymore and should be removed in the future. | ||
internals.removeApplicationFromDOMIfIvyEnabled(options, props); | ||
return [2 /*return*/]; | ||
}); | ||
function unmount(options) { | ||
return Promise.resolve().then(function () { | ||
if (options.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', options.routingEventListener); | ||
} | ||
options.bootstrappedModule.destroy(); | ||
options.bootstrappedModule = null; | ||
}); | ||
@@ -589,0 +522,0 @@ } |
@@ -11,2 +11,2 @@ { | ||
"name": "single-spa-angular/elements" | ||
} | ||
} |
@@ -46,2 +46,2 @@ import { __awaiter } from "tslib"; | ||
} | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4vLi4vbGlicy9zaW5nbGUtc3BhLWFuZ3VsYXIvZWxlbWVudHMvc3JjLyIsInNvdXJjZXMiOlsiaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUVBLE9BQU8sRUFFTCxpQ0FBaUMsR0FDbEMsTUFBTSw4QkFBOEIsQ0FBQztBQUl0QyxNQUFNLGNBQWMsR0FBZ0Q7SUFDbEUsT0FBTyxFQUFFLElBQUk7SUFDYixRQUFRLEVBQUUsSUFBSztJQUNmLFdBQVcsRUFBRSxJQUFJO0lBQ2pCLGlCQUFpQixFQUFFLElBQUs7SUFDeEIsZ0JBQWdCLEVBQUUsU0FBUztDQUM1QixDQUFDO0FBRUYsU0FBZSxTQUFTLENBQUMsT0FBb0QsRUFBRSxLQUFVOztRQUN2RixJQUFJLE9BQU8sQ0FBQyxXQUFXLEtBQUssSUFBSSxFQUFFO1lBQ2hDLE9BQU87U0FDUjtRQUVELG9FQUFvRTtRQUNwRSxnRUFBZ0U7UUFDaEUsMEJBQTBCO1FBQzFCLE9BQU8sQ0FBQyxXQUFXLEdBQUcsTUFBTSxPQUFPLENBQUMsaUJBQWlCLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDL0QsQ0FBQztDQUFBO0FBRUQsU0FBZSxLQUFLLENBQUMsT0FBb0QsRUFBRSxLQUFVOztRQUNuRixNQUFNLGdCQUFnQixHQUFHLGlDQUFpQyxDQUFDLE9BQU8sRUFBRSxLQUFLLENBQUMsQ0FBQztRQUMzRSw0RUFBNEU7UUFDNUUscUVBQXFFO1FBQ3JFLHFCQUFxQjtRQUNyQixPQUFPLENBQUMsT0FBTyxHQUFHLGdCQUFnQixDQUFDLGlCQUE4QixDQUFDO0lBQ3BFLENBQUM7Q0FBQTtBQUVELFNBQVMsT0FBTyxDQUFDLE9BQW9EO0lBQ25FLE9BQU8sT0FBTyxDQUFDLE9BQU8sRUFBRSxDQUFDLElBQUksQ0FBQyxHQUFHLEVBQUU7UUFDakMsbUVBQW1FO1FBQ25FLG1FQUFtRTtRQUNuRSxPQUFPLENBQUMsT0FBUSxDQUFDLGFBQWMsQ0FBQyxXQUFXLENBQUMsT0FBTyxDQUFDLE9BQVEsQ0FBQyxDQUFDO1FBQzlELE9BQU8sQ0FBQyxPQUFPLEdBQUcsSUFBSSxDQUFDO0lBQ3pCLENBQUMsQ0FBQyxDQUFDO0FBQ0wsQ0FBQztBQUVELE1BQU0sVUFBVSx3QkFBd0IsQ0FBQyxXQUF3QztJQUMvRSxNQUFNLE9BQU8sbUNBQ1IsY0FBYyxHQUNkLFdBQVcsQ0FDZixDQUFDO0lBRUYsT0FBTztRQUNMLFNBQVMsRUFBRSxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxPQUFPLENBQUM7UUFDeEMsS0FBSyxFQUFFLEtBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQztRQUNoQyxPQUFPLEVBQUUsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDO0tBQ3JDLENBQUM7QUFDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTGlmZUN5Y2xlcyB9IGZyb20gJ3NpbmdsZS1zcGEnO1xuaW1wb3J0IHsgTmdFbGVtZW50IH0gZnJvbSAnQGFuZ3VsYXIvZWxlbWVudHMnO1xuaW1wb3J0IHtcbiAgQmFzZVNpbmdsZVNwYUFuZ3VsYXJPcHRpb25zLFxuICBnZXRDb250YWluZXJFbGVtZW50QW5kU2V0VGVtcGxhdGUsXG59IGZyb20gJ3NpbmdsZS1zcGEtYW5ndWxhci9pbnRlcm5hbHMnO1xuXG5pbXBvcnQgeyBCb290c3RyYXBwZWRTaW5nbGVTcGFBbmd1bGFyRWxlbWVudHNPcHRpb25zIH0gZnJvbSAnLi90eXBlcyc7XG5cbmNvbnN0IGRlZmF1bHRPcHRpb25zOiBCb290c3RyYXBwZWRTaW5nbGVTcGFBbmd1bGFyRWxlbWVudHNPcHRpb25zID0ge1xuICBlbGVtZW50OiBudWxsLFxuICB0ZW1wbGF0ZTogbnVsbCEsXG4gIG5nTW9kdWxlUmVmOiBudWxsLFxuICBib290c3RyYXBGdW5jdGlvbjogbnVsbCEsXG4gIGRvbUVsZW1lbnRHZXR0ZXI6IHVuZGVmaW5lZCxcbn07XG5cbmFzeW5jIGZ1bmN0aW9uIGJvb3RzdHJhcChvcHRpb25zOiBCb290c3RyYXBwZWRTaW5nbGVTcGFBbmd1bGFyRWxlbWVudHNPcHRpb25zLCBwcm9wczogYW55KSB7XG4gIGlmIChvcHRpb25zLm5nTW9kdWxlUmVmICE9PSBudWxsKSB7XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgLy8gV2UgY2FsbCBgYm9vdHN0cmFwRnVuY3Rpb24oKWAgaW5zaWRlIHRoZSBib290c3RyYXAgbGlmZWN5Y2xlIGhvb2tcbiAgLy8gYmVjYXVzZSBBbmd1bGFyIG1vZHVsZXMgdGhhdCBleHBvc2UgY3VzdG9tIGVsZW1lbnRzIHNob3VsZCBiZVxuICAvLyBib290c3RyYXBwZWQgb25seSBvbmNlLlxuICBvcHRpb25zLm5nTW9kdWxlUmVmID0gYXdhaXQgb3B0aW9ucy5ib290c3RyYXBGdW5jdGlvbihwcm9wcyk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIG1vdW50KG9wdGlvbnM6IEJvb3RzdHJhcHBlZFNpbmdsZVNwYUFuZ3VsYXJFbGVtZW50c09wdGlvbnMsIHByb3BzOiBhbnkpIHtcbiAgY29uc3QgY29udGFpbmVyRWxlbWVudCA9IGdldENvbnRhaW5lckVsZW1lbnRBbmRTZXRUZW1wbGF0ZShvcHRpb25zLCBwcm9wcyk7XG4gIC8vIGBvcHRpb25zLnRlbXBsYXRlYCB3aGljaCBjYW4gYmUgYDxhcHAtZWxlbWVudCAvPmAgaXMgbm90IGEgdmFsaWQgc2VsZWN0b3JcbiAgLy8gZm9yIGBkb2N1bWVudC5xdWVyeVNlbGVjdG9yYCwgdGh1cyB3ZSByZXRyaWV2ZSB0aGlzIGN1c3RvbSBlbGVtZW50XG4gIC8vIHZpYSB0aGlzIHByb3BlcnR5LlxuICBvcHRpb25zLmVsZW1lbnQgPSBjb250YWluZXJFbGVtZW50LmZpcnN0RWxlbWVudENoaWxkIGFzIE5nRWxlbWVudDtcbn1cblxuZnVuY3Rpb24gdW5tb3VudChvcHRpb25zOiBCb290c3RyYXBwZWRTaW5nbGVTcGFBbmd1bGFyRWxlbWVudHNPcHRpb25zKTogUHJvbWlzZTx2b2lkPiB7XG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUoKS50aGVuKCgpID0+IHtcbiAgICAvLyBSZW1vdmluZyBjdXN0b20gZWxlbWVudCBmcm9tIERPTSBpcyBlbm91Z2ggc2luY2UgaXQgd2lsbCB0cmlnZ2VyXG4gICAgLy8gYGRpc2Nvbm5lY3RlZENhbGxiYWNrKClgIGFuZCBBbmd1bGFyIHdpbGwgZGlzcG9zZSBhbGwgcmVzb3VyY2VzLlxuICAgIG9wdGlvbnMuZWxlbWVudCEucGFyZW50RWxlbWVudCEucmVtb3ZlQ2hpbGQob3B0aW9ucy5lbGVtZW50ISk7XG4gICAgb3B0aW9ucy5lbGVtZW50ID0gbnVsbDtcbiAgfSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBzaW5nbGVTcGFBbmd1bGFyRWxlbWVudHModXNlck9wdGlvbnM6IEJhc2VTaW5nbGVTcGFBbmd1bGFyT3B0aW9ucyk6IExpZmVDeWNsZXMge1xuICBjb25zdCBvcHRpb25zOiBCb290c3RyYXBwZWRTaW5nbGVTcGFBbmd1bGFyRWxlbWVudHNPcHRpb25zID0ge1xuICAgIC4uLmRlZmF1bHRPcHRpb25zLFxuICAgIC4uLnVzZXJPcHRpb25zLFxuICB9O1xuXG4gIHJldHVybiB7XG4gICAgYm9vdHN0cmFwOiBib290c3RyYXAuYmluZChudWxsLCBvcHRpb25zKSxcbiAgICBtb3VudDogbW91bnQuYmluZChudWxsLCBvcHRpb25zKSxcbiAgICB1bm1vdW50OiB1bm1vdW50LmJpbmQobnVsbCwgb3B0aW9ucyksXG4gIH07XG59XG4iXX0= | ||
//# sourceMappingURL=index.js.map |
@@ -5,2 +5,2 @@ /** | ||
export * from './index'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXNwYS1hbmd1bGFyLWVsZW1lbnRzLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uLy4uL2xpYnMvc2luZ2xlLXNwYS1hbmd1bGFyL2VsZW1lbnRzL3NyYy8iLCJzb3VyY2VzIjpbInNpbmdsZS1zcGEtYW5ndWxhci1lbGVtZW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2luZGV4JztcbiJdfQ== | ||
//# sourceMappingURL=single-spa-angular-elements.js.map |
export {}; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4vLi4vbGlicy9zaW5nbGUtc3BhLWFuZ3VsYXIvZWxlbWVudHMvc3JjLyIsInNvdXJjZXMiOlsidHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlUmVmIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBOZ0VsZW1lbnQgfSBmcm9tICdAYW5ndWxhci9lbGVtZW50cyc7XG5pbXBvcnQgeyBCYXNlU2luZ2xlU3BhQW5ndWxhck9wdGlvbnMgfSBmcm9tICdzaW5nbGUtc3BhLWFuZ3VsYXIvaW50ZXJuYWxzJztcblxuZXhwb3J0IGludGVyZmFjZSBCb290c3RyYXBwZWRTaW5nbGVTcGFBbmd1bGFyRWxlbWVudHNPcHRpb25zIGV4dGVuZHMgQmFzZVNpbmdsZVNwYUFuZ3VsYXJPcHRpb25zIHtcbiAgbmdNb2R1bGVSZWY6IE5nTW9kdWxlUmVmPGFueT4gfCBudWxsO1xuICAvLyBUaGlzIHdpbGwgYmUgYW4gYWN0dWFsIGN1c3RvbSBlbGVtZW50LlxuICBlbGVtZW50OiBOZ0VsZW1lbnQgfCBudWxsO1xufVxuIl19 | ||
//# sourceMappingURL=types.js.map |
@@ -5,2 +5,2 @@ /** | ||
export * from './src/public_api'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4vbGlicy9zaW5nbGUtc3BhLWFuZ3VsYXIvIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUNILGNBQWMsa0JBQWtCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFRoZSBwdWJsaWMgYXBpIGZvciBjb25zdW1lcnMgb2Ygc2luZ2xlLXNwYS1hbmd1bGFyXG4gKi9cbmV4cG9ydCAqIGZyb20gJy4vc3JjL3B1YmxpY19hcGknO1xuIl19 | ||
//# sourceMappingURL=index.js.map |
@@ -1,12 +0,1 @@ | ||
export function removeApplicationFromDOMIfIvyEnabled(options, props) { | ||
if (ivyEnabled()) { | ||
const domElementGetter = chooseDomElementGetter(options, props); | ||
const domElement = getContainerElement(domElementGetter); | ||
// View Engine removes all nodes automatically when calling `NgModuleRef.destroy()`, | ||
// which calls `ComponentRef.destroy()`. | ||
// Basically this will remove `app-root` or any other selector from the container element. | ||
while (domElement.firstChild) | ||
domElement.removeChild(domElement.firstChild); | ||
} | ||
} | ||
export function getContainerElementAndSetTemplate(options, props) { | ||
@@ -56,16 +45,2 @@ const domElementGetter = chooseDomElementGetter(options, props); | ||
} | ||
function ivyEnabled() { | ||
try { | ||
// `ɵivyEnabled` variable is exposed starting from version 8. | ||
// We use `require` here except of a single `import { ɵivyEnabled }` because the | ||
// developer can use Angular version that doesn't expose it (all versions <8). | ||
// The `catch` statement will handle those cases. | ||
// eslint-disable-next-line | ||
const { ɵivyEnabled } = require('@angular/core'); | ||
return !!ɵivyEnabled; | ||
} | ||
catch (_a) { | ||
return false; | ||
} | ||
} | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9tLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uLy4uL2xpYnMvc2luZ2xlLXNwYS1hbmd1bGFyL2ludGVybmFscy9zcmMvIiwic291cmNlcyI6WyJkb20udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxVQUFVLG9DQUFvQyxDQUNsRCxPQUFVLEVBQ1YsS0FBVTtJQUVWLElBQUksVUFBVSxFQUFFLEVBQUU7UUFDaEIsTUFBTSxnQkFBZ0IsR0FBRyxzQkFBc0IsQ0FBQyxPQUFPLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDaEUsTUFBTSxVQUFVLEdBQUcsbUJBQW1CLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztRQUN6RCxvRkFBb0Y7UUFDcEYsd0NBQXdDO1FBQ3hDLDBGQUEwRjtRQUMxRixPQUFPLFVBQVUsQ0FBQyxVQUFVO1lBQUUsVUFBVSxDQUFDLFdBQVcsQ0FBQyxVQUFVLENBQUMsVUFBVSxDQUFDLENBQUM7S0FDN0U7QUFDSCxDQUFDO0FBRUQsTUFBTSxVQUFVLGlDQUFpQyxDQUMvQyxPQUFVLEVBQ1YsS0FBVTtJQUVWLE1BQU0sZ0JBQWdCLEdBQUcsc0JBQXNCLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQyxDQUFDO0lBRWhFLElBQUksQ0FBQyxnQkFBZ0IsRUFBRTtRQUNyQixNQUFNLEtBQUssQ0FDVCxxQ0FDRSxLQUFLLENBQUMsSUFBSSxJQUFJLEtBQUssQ0FBQyxPQUN0QixrRUFBa0UsQ0FDbkUsQ0FBQztLQUNIO0lBRUQsTUFBTSxnQkFBZ0IsR0FBRyxtQkFBbUIsQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO0lBQy9ELGdCQUFnQixDQUFDLFNBQVMsR0FBRyxPQUFPLENBQUMsUUFBUSxDQUFDO0lBQzlDLE9BQU8sZ0JBQWdCLENBQUM7QUFDMUIsQ0FBQztBQUVELFNBQVMsbUJBQW1CLENBQUMsZ0JBQWtDO0lBQzdELE1BQU0sT0FBTyxHQUFHLGdCQUFnQixFQUFFLENBQUM7SUFFbkMsSUFBSSxDQUFDLE9BQU8sRUFBRTtRQUNaLE1BQU0sS0FBSyxDQUFDLHFEQUFxRCxDQUFDLENBQUM7S0FDcEU7SUFFRCxPQUFPLE9BQU8sQ0FBQztBQUNqQixDQUFDO0FBRUQsU0FBUyxzQkFBc0IsQ0FDN0IsSUFBTyxFQUNQLEtBQVU7O0lBRVYsS0FBSyxTQUFHLEtBQUssYUFBTCxLQUFLLHVCQUFMLEtBQUssQ0FBRSxXQUFXLG1DQUFJLEtBQUssQ0FBQztJQUVwQyxJQUFJLEtBQUssQ0FBQyxVQUFVLEVBQUU7UUFDcEIsT0FBTyxHQUFHLEVBQUUsQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDO0tBQy9CO1NBQU0sSUFBSSxLQUFLLENBQUMsZ0JBQWdCLEVBQUU7UUFDakMsT0FBTyxLQUFLLENBQUMsZ0JBQWdCLENBQUM7S0FDL0I7U0FBTSxJQUFJLElBQUksQ0FBQyxnQkFBZ0IsRUFBRTtRQUNoQyxPQUFPLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQztLQUM5QjtTQUFNO1FBQ0wsT0FBTyx1QkFBdUIsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7S0FDNUM7QUFDSCxDQUFDO0FBRUQsU0FBUyx1QkFBdUIsQ0FBQyxJQUFZO0lBQzNDLE9BQU8sU0FBUyxvQkFBb0I7UUFDbEMsTUFBTSxFQUFFLEdBQUcsMEJBQTBCLElBQUksRUFBRSxDQUFDO1FBQzVDLElBQUksVUFBVSxHQUF1QixRQUFRLENBQUMsY0FBYyxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBRWpFLElBQUksQ0FBQyxVQUFVLEVBQUU7WUFDZixVQUFVLEdBQUcsUUFBUSxDQUFDLGFBQWEsQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUMzQyxVQUFVLENBQUMsRUFBRSxHQUFHLEVBQUUsQ0FBQztZQUNuQixRQUFRLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxVQUFVLENBQUMsQ0FBQztTQUN2QztRQUVELE9BQU8sVUFBVSxDQUFDO0lBQ3BCLENBQUMsQ0FBQztBQUNKLENBQUM7QUFFRCxTQUFTLFVBQVU7SUFDakIsSUFBSTtRQUNGLDZEQUE2RDtRQUM3RCxnRkFBZ0Y7UUFDaEYsOEVBQThFO1FBQzlFLGlEQUFpRDtRQUNqRCwyQkFBMkI7UUFDM0IsTUFBTSxFQUFFLFdBQVcsRUFBRSxHQUFHLE9BQU8sQ0FBQyxlQUFlLENBQUMsQ0FBQztRQUNqRCxPQUFPLENBQUMsQ0FBQyxXQUFXLENBQUM7S0FDdEI7SUFBQyxXQUFNO1FBQ04sT0FBTyxLQUFLLENBQUM7S0FDZDtBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEb21FbGVtZW50R2V0dGVyLCBCYXNlU2luZ2xlU3BhQW5ndWxhck9wdGlvbnMgfSBmcm9tICcuL3R5cGVzJztcblxuZXhwb3J0IGZ1bmN0aW9uIHJlbW92ZUFwcGxpY2F0aW9uRnJvbURPTUlmSXZ5RW5hYmxlZDxUIGV4dGVuZHMgQmFzZVNpbmdsZVNwYUFuZ3VsYXJPcHRpb25zPihcbiAgb3B0aW9uczogVCxcbiAgcHJvcHM6IGFueSxcbik6IHZvaWQge1xuICBpZiAoaXZ5RW5hYmxlZCgpKSB7XG4gICAgY29uc3QgZG9tRWxlbWVudEdldHRlciA9IGNob29zZURvbUVsZW1lbnRHZXR0ZXIob3B0aW9ucywgcHJvcHMpO1xuICAgIGNvbnN0IGRvbUVsZW1lbnQgPSBnZXRDb250YWluZXJFbGVtZW50KGRvbUVsZW1lbnRHZXR0ZXIpO1xuICAgIC8vIFZpZXcgRW5naW5lIHJlbW92ZXMgYWxsIG5vZGVzIGF1dG9tYXRpY2FsbHkgd2hlbiBjYWxsaW5nIGBOZ01vZHVsZVJlZi5kZXN0cm95KClgLFxuICAgIC8vIHdoaWNoIGNhbGxzIGBDb21wb25lbnRSZWYuZGVzdHJveSgpYC5cbiAgICAvLyBCYXNpY2FsbHkgdGhpcyB3aWxsIHJlbW92ZSBgYXBwLXJvb3RgIG9yIGFueSBvdGhlciBzZWxlY3RvciBmcm9tIHRoZSBjb250YWluZXIgZWxlbWVudC5cbiAgICB3aGlsZSAoZG9tRWxlbWVudC5maXJzdENoaWxkKSBkb21FbGVtZW50LnJlbW92ZUNoaWxkKGRvbUVsZW1lbnQuZmlyc3RDaGlsZCk7XG4gIH1cbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGdldENvbnRhaW5lckVsZW1lbnRBbmRTZXRUZW1wbGF0ZTxUIGV4dGVuZHMgQmFzZVNpbmdsZVNwYUFuZ3VsYXJPcHRpb25zPihcbiAgb3B0aW9uczogVCxcbiAgcHJvcHM6IGFueSxcbik6IEhUTUxFbGVtZW50IHtcbiAgY29uc3QgZG9tRWxlbWVudEdldHRlciA9IGNob29zZURvbUVsZW1lbnRHZXR0ZXIob3B0aW9ucywgcHJvcHMpO1xuXG4gIGlmICghZG9tRWxlbWVudEdldHRlcikge1xuICAgIHRocm93IEVycm9yKFxuICAgICAgYENhbm5vdCBtb3VudCBhbmd1bGFyIGFwcGxpY2F0aW9uICcke1xuICAgICAgICBwcm9wcy5uYW1lIHx8IHByb3BzLmFwcE5hbWVcbiAgICAgIH0nIHdpdGhvdXQgYSBkb21FbGVtZW50R2V0dGVyIHByb3ZpZGVkIGVpdGhlciBhcyBhbiBvcHQgb3IgYSBwcm9wYCxcbiAgICApO1xuICB9XG5cbiAgY29uc3QgY29udGFpbmVyRWxlbWVudCA9IGdldENvbnRhaW5lckVsZW1lbnQoZG9tRWxlbWVudEdldHRlcik7XG4gIGNvbnRhaW5lckVsZW1lbnQuaW5uZXJIVE1MID0gb3B0aW9ucy50ZW1wbGF0ZTtcbiAgcmV0dXJuIGNvbnRhaW5lckVsZW1lbnQ7XG59XG5cbmZ1bmN0aW9uIGdldENvbnRhaW5lckVsZW1lbnQoZG9tRWxlbWVudEdldHRlcjogRG9tRWxlbWVudEdldHRlcik6IG5ldmVyIHwgSFRNTEVsZW1lbnQge1xuICBjb25zdCBlbGVtZW50ID0gZG9tRWxlbWVudEdldHRlcigpO1xuXG4gIGlmICghZWxlbWVudCkge1xuICAgIHRocm93IEVycm9yKCdkb21FbGVtZW50R2V0dGVyIGRpZCBub3QgcmV0dXJuIGEgdmFsaWQgZG9tIGVsZW1lbnQnKTtcbiAgfVxuXG4gIHJldHVybiBlbGVtZW50O1xufVxuXG5mdW5jdGlvbiBjaG9vc2VEb21FbGVtZW50R2V0dGVyPFQgZXh0ZW5kcyBCYXNlU2luZ2xlU3BhQW5ndWxhck9wdGlvbnM+KFxuICBvcHRzOiBULFxuICBwcm9wczogYW55LFxuKTogRG9tRWxlbWVudEdldHRlciB7XG4gIHByb3BzID0gcHJvcHM/LmN1c3RvbVByb3BzID8/IHByb3BzO1xuXG4gIGlmIChwcm9wcy5kb21FbGVtZW50KSB7XG4gICAgcmV0dXJuICgpID0+IHByb3BzLmRvbUVsZW1lbnQ7XG4gIH0gZWxzZSBpZiAocHJvcHMuZG9tRWxlbWVudEdldHRlcikge1xuICAgIHJldHVybiBwcm9wcy5kb21FbGVtZW50R2V0dGVyO1xuICB9IGVsc2UgaWYgKG9wdHMuZG9tRWxlbWVudEdldHRlcikge1xuICAgIHJldHVybiBvcHRzLmRvbUVsZW1lbnRHZXR0ZXI7XG4gIH0gZWxzZSB7XG4gICAgcmV0dXJuIGRlZmF1bHREb21FbGVtZW50R2V0dGVyKHByb3BzLm5hbWUpO1xuICB9XG59XG5cbmZ1bmN0aW9uIGRlZmF1bHREb21FbGVtZW50R2V0dGVyKG5hbWU6IHN0cmluZyk6IERvbUVsZW1lbnRHZXR0ZXIge1xuICByZXR1cm4gZnVuY3Rpb24gZ2V0RGVmYXVsdERvbUVsZW1lbnQoKSB7XG4gICAgY29uc3QgaWQgPSBgc2luZ2xlLXNwYS1hcHBsaWNhdGlvbjoke25hbWV9YDtcbiAgICBsZXQgZG9tRWxlbWVudDogSFRNTEVsZW1lbnQgfCBudWxsID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoaWQpO1xuXG4gICAgaWYgKCFkb21FbGVtZW50KSB7XG4gICAgICBkb21FbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnZGl2Jyk7XG4gICAgICBkb21FbGVtZW50LmlkID0gaWQ7XG4gICAgICBkb2N1bWVudC5ib2R5LmFwcGVuZENoaWxkKGRvbUVsZW1lbnQpO1xuICAgIH1cblxuICAgIHJldHVybiBkb21FbGVtZW50O1xuICB9O1xufVxuXG5mdW5jdGlvbiBpdnlFbmFibGVkKCk6IGJvb2xlYW4ge1xuICB0cnkge1xuICAgIC8vIGDJtWl2eUVuYWJsZWRgIHZhcmlhYmxlIGlzIGV4cG9zZWQgc3RhcnRpbmcgZnJvbSB2ZXJzaW9uIDguXG4gICAgLy8gV2UgdXNlIGByZXF1aXJlYCBoZXJlIGV4Y2VwdCBvZiBhIHNpbmdsZSBgaW1wb3J0IHsgybVpdnlFbmFibGVkIH1gIGJlY2F1c2UgdGhlXG4gICAgLy8gZGV2ZWxvcGVyIGNhbiB1c2UgQW5ndWxhciB2ZXJzaW9uIHRoYXQgZG9lc24ndCBleHBvc2UgaXQgKGFsbCB2ZXJzaW9ucyA8OCkuXG4gICAgLy8gVGhlIGBjYXRjaGAgc3RhdGVtZW50IHdpbGwgaGFuZGxlIHRob3NlIGNhc2VzLlxuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZVxuICAgIGNvbnN0IHsgybVpdnlFbmFibGVkIH0gPSByZXF1aXJlKCdAYW5ndWxhci9jb3JlJyk7XG4gICAgcmV0dXJuICEhybVpdnlFbmFibGVkO1xuICB9IGNhdGNoIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cbn1cbiJdfQ== | ||
//# sourceMappingURL=dom.js.map |
export * from './types'; | ||
export { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled } from './dom'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4vLi4vbGlicy9zaW5nbGUtc3BhLWFuZ3VsYXIvaW50ZXJuYWxzL3NyYy8iLCJzb3VyY2VzIjpbImluZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLE9BQU8sRUFBRSxpQ0FBaUMsRUFBRSxvQ0FBb0MsRUFBRSxNQUFNLE9BQU8sQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuZXhwb3J0IHsgZ2V0Q29udGFpbmVyRWxlbWVudEFuZFNldFRlbXBsYXRlLCByZW1vdmVBcHBsaWNhdGlvbkZyb21ET01JZkl2eUVuYWJsZWQgfSBmcm9tICcuL2RvbSc7XG4iXX0= | ||
export { getContainerElementAndSetTemplate } from './dom'; | ||
//# sourceMappingURL=index.js.map |
@@ -5,2 +5,2 @@ /** | ||
export * from './index'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXNwYS1hbmd1bGFyLWludGVybmFscy5qcyIsInNvdXJjZVJvb3QiOiIuLi8uLi8uLi8uLi9saWJzL3NpbmdsZS1zcGEtYW5ndWxhci9pbnRlcm5hbHMvc3JjLyIsInNvdXJjZXMiOlsic2luZ2xlLXNwYS1hbmd1bGFyLWludGVybmFscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2luZGV4JztcbiJdfQ== | ||
//# sourceMappingURL=single-spa-angular-internals.js.map |
export {}; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4vLi4vbGlicy9zaW5nbGUtc3BhLWFuZ3VsYXIvaW50ZXJuYWxzL3NyYy8iLCJzb3VyY2VzIjpbInR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBcHBQcm9wcyB9IGZyb20gJ3NpbmdsZS1zcGEnO1xuaW1wb3J0IHsgTmdNb2R1bGVSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuZXhwb3J0IHR5cGUgRG9tRWxlbWVudEdldHRlciA9ICgpID0+IEhUTUxFbGVtZW50O1xuXG5leHBvcnQgaW50ZXJmYWNlIEJhc2VTaW5nbGVTcGFBbmd1bGFyT3B0aW9ucyB7XG4gIHRlbXBsYXRlOiBzdHJpbmc7XG4gIGRvbUVsZW1lbnRHZXR0ZXI/KCk6IEhUTUxFbGVtZW50O1xuICBib290c3RyYXBGdW5jdGlvbihwcm9wczogQXBwUHJvcHMpOiBQcm9taXNlPE5nTW9kdWxlUmVmPGFueT4+O1xufVxuIl19 | ||
//# sourceMappingURL=types.js.map |
export { ParcelModule } from './parcel.module'; | ||
export { ParcelComponent } from './parcel.component'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4vLi4vbGlicy9zaW5nbGUtc3BhLWFuZ3VsYXIvcGFyY2VsL3NyYy8iLCJzb3VyY2VzIjpbImluZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sb0JBQW9CLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBQYXJjZWxNb2R1bGUgfSBmcm9tICcuL3BhcmNlbC5tb2R1bGUnO1xuZXhwb3J0IHsgUGFyY2VsQ29tcG9uZW50IH0gZnJvbSAnLi9wYXJjZWwuY29tcG9uZW50JztcbiJdfQ== | ||
//# sourceMappingURL=index.js.map |
@@ -94,2 +94,3 @@ import { Component, ElementRef, Input, ChangeDetectionStrategy, } from '@angular/core'; | ||
]; | ||
/** @nocollapse */ | ||
ParcelComponent.ctorParameters = () => [ | ||
@@ -107,2 +108,2 @@ { type: ElementRef } | ||
}; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parcel.component.js","sourceRoot":"../../../../libs/single-spa-angular/parcel/src/","sources":["parcel.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  ChangeDetectionStrategy,\n} from '@angular/core';\nimport { Parcel, ParcelConfig, AppProps } from 'single-spa';\n\nconst enum Action {\n  Mount = 'mount',\n  Update = 'update',\n  Unmount = 'unmount',\n}\n\n@Component({\n  selector: 'parcel',\n  template: '',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ParcelComponent implements OnChanges, OnInit, OnDestroy {\n  @Input() config: ParcelConfig | null = null;\n  @Input() mountParcel: AppProps['mountParcel'] | null = null;\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  private hasError = false;\n  private unmounted?: boolean;\n  private wrapper: HTMLElement | null = null;\n  private parcel: Parcel | null = null;\n  private task: Promise<void> | null = null;\n\n  constructor(private host: ElementRef<HTMLElement>) {}\n\n  ngOnChanges(): void {\n    this.scheduleTask(Action.Update, () => {\n      if (this.parcel !== null && this.parcel.update) {\n        return this.parcel.update(this.customProps);\n      }\n    });\n  }\n\n  ngOnInit(): void {\n    this.scheduleTask(Action.Mount, () => {\n      if (this.mountParcel === null) {\n        throw new Error(\n          'single-spa-angular: the [mountParcel] binding is required when using the <parcel> component. You can either (1) import mountRootParcel from single-spa or (2) use the mountParcel prop provided to single-spa applications.',\n        );\n      }\n\n      this.wrapper = document.createElement(this.wrapWith);\n\n      if (this.appendTo !== null) {\n        this.appendTo.appendChild(this.wrapper);\n      } else {\n        this.host.nativeElement.appendChild(this.wrapper);\n      }\n\n      this.parcel = this.mountParcel(this.config!, {\n        ...this.customProps,\n        domElement: this.wrapper,\n      });\n\n      if (this.onParcelMount !== null) {\n        this.parcel.mountPromise.then(this.onParcelMount);\n      }\n\n      this.unmounted = false;\n      return this.parcel.mountPromise;\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.scheduleTask(Action.Unmount, () => {\n      if (this.parcel !== null && this.parcel.getStatus() === 'MOUNTED') {\n        return this.parcel.unmount();\n      }\n    });\n\n    if (this.wrapper !== null) {\n      this.wrapper.parentNode!.removeChild(this.wrapper);\n    }\n\n    this.unmounted = true;\n  }\n\n  private scheduleTask(action: Action, task: () => void | Promise<any>): void {\n    if (this.hasError && action !== Action.Unmount) {\n      // In an error state, we don't do anything anymore except for unmounting\n      return;\n    }\n\n    this.task = (this.task || Promise.resolve())\n      .then(() => {\n        if (this.unmounted && action !== Action.Unmount) {\n          // Never do anything once the angular component unmounts\n          return;\n        }\n\n        return task();\n      })\n      .catch((error: Error) => {\n        this.task = Promise.resolve();\n        this.hasError = true;\n\n        if (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"]} | ||
//# sourceMappingURL=parcel.component.js.map |
@@ -11,2 +11,2 @@ import { NgModule } from '@angular/core'; | ||
]; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFyY2VsLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIuLi8uLi8uLi8uLi9saWJzL3NpbmdsZS1zcGEtYW5ndWxhci9wYXJjZWwvc3JjLyIsInNvdXJjZXMiOlsicGFyY2VsLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRXpDLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQU1yRCxNQUFNLE9BQU8sWUFBWTs7O1lBSnhCLFFBQVEsU0FBQztnQkFDUixZQUFZLEVBQUUsQ0FBQyxlQUFlLENBQUM7Z0JBQy9CLE9BQU8sRUFBRSxDQUFDLGVBQWUsQ0FBQzthQUMzQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbmltcG9ydCB7IFBhcmNlbENvbXBvbmVudCB9IGZyb20gJy4vcGFyY2VsLmNvbXBvbmVudCc7XG5cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1BhcmNlbENvbXBvbmVudF0sXG4gIGV4cG9ydHM6IFtQYXJjZWxDb21wb25lbnRdLFxufSlcbmV4cG9ydCBjbGFzcyBQYXJjZWxNb2R1bGUge31cbiJdfQ== | ||
//# sourceMappingURL=parcel.module.js.map |
@@ -5,2 +5,2 @@ /** | ||
export * from './index'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXNwYS1hbmd1bGFyLXBhcmNlbC5qcyIsInNvdXJjZVJvb3QiOiIuLi8uLi8uLi8uLi9saWJzL3NpbmdsZS1zcGEtYW5ndWxhci9wYXJjZWwvc3JjLyIsInNvdXJjZXMiOlsic2luZ2xlLXNwYS1hbmd1bGFyLXBhcmNlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2luZGV4JztcbiJdfQ== | ||
//# sourceMappingURL=single-spa-angular-parcel.js.map |
@@ -6,2 +6,2 @@ /** | ||
export { SingleSpaPlatformLocation as ɵa } from './src/extra-providers'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXNwYS1hbmd1bGFyLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uL2xpYnMvc2luZ2xlLXNwYS1hbmd1bGFyLyIsInNvdXJjZXMiOlsic2luZ2xlLXNwYS1hbmd1bGFyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxTQUFTLENBQUM7QUFFeEIsT0FBTyxFQUFDLHlCQUF5QixJQUFJLEVBQUUsRUFBQyxNQUFNLHVCQUF1QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2luZGV4JztcblxuZXhwb3J0IHtTaW5nbGVTcGFQbGF0Zm9ybUxvY2F0aW9uIGFzIMm1YX0gZnJvbSAnLi9zcmMvZXh0cmEtcHJvdmlkZXJzJzsiXX0= | ||
//# sourceMappingURL=single-spa-angular.js.map |
@@ -9,39 +9,4 @@ import { Injectable, Inject } from '@angular/core'; | ||
this.skipNextPopState = false; | ||
// 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; | ||
this.source = 'Window.addEventListener:popstate'; | ||
} | ||
destroyApplication(zoneIdentifier) { | ||
// TLDR: Angular adds `popstate` event listener and then doesn't remove it when application gets destroyed. | ||
// Basically, Angular has a potentional memory leak. The `ɵBrowserPlatformLocation` | ||
// has `onPopState` method which adds `popstate` event listener and forgets, see here: | ||
// https://github.com/angular/angular/blob/14be55c9facf3e47b8c97df4502dc3f0f897da03/packages/common/src/location/platform_location.ts#L126 | ||
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); | ||
} | ||
} | ||
this.zoneToOnPopStateListenersMap.delete(zone); | ||
} | ||
pushState(state, title, url) { | ||
@@ -65,3 +30,3 @@ this.skipNextPopState = true; | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
fn = zone.wrap(fn, `${this.source++}`); | ||
fn = zone.wrap(fn, this.source); | ||
const onPopStateListener = (event) => { | ||
@@ -80,14 +45,4 @@ // The `LocationChangeEvent` doesn't have the `singleSpa` property, since it's added | ||
}; | ||
this.storeOnPopStateListener(zone, onPopStateListener); | ||
super.onPopState(onPopStateListener); | ||
return 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. | ||
const onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone) || []; | ||
onPopStateListeners.push(onPopStateListener); | ||
if (!this.zoneToOnPopStateListenersMap.has(zone)) { | ||
this.zoneToOnPopStateListenersMap.set(zone, onPopStateListeners); | ||
} | ||
} | ||
} | ||
@@ -106,3 +61,2 @@ SingleSpaPlatformLocation.decorators = [ | ||
provide: SingleSpaPlatformLocation, | ||
useClass: SingleSpaPlatformLocation, | ||
deps: [[new Inject(DOCUMENT)]], | ||
@@ -116,2 +70,2 @@ }, | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extra-providers.js","sourceRoot":"../../../libs/single-spa-angular/","sources":["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"]} | ||
//# sourceMappingURL=extra-providers.js.map |
export { singleSpaAngular } from './single-spa-angular'; | ||
export { getSingleSpaExtraProviders } from './extra-providers'; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIuLi8uLi8uLi9saWJzL3NpbmdsZS1zcGEtYW5ndWxhci8iLCJzb3VyY2VzIjpbInNyYy9wdWJsaWNfYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBQ3hELE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLG1CQUFtQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgc2luZ2xlU3BhQW5ndWxhciB9IGZyb20gJy4vc2luZ2xlLXNwYS1hbmd1bGFyJztcbmV4cG9ydCB7IGdldFNpbmdsZVNwYUV4dHJhUHJvdmlkZXJzIH0gZnJvbSAnLi9leHRyYS1wcm92aWRlcnMnO1xuIl19 | ||
//# sourceMappingURL=public_api.js.map |
import { __awaiter } from "tslib"; | ||
import { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled, } from 'single-spa-angular/internals'; | ||
import { getContainerElementAndSetTemplate } from 'single-spa-angular/internals'; | ||
import { SingleSpaPlatformLocation } from './extra-providers'; | ||
@@ -12,3 +12,2 @@ const defaultOptions = { | ||
domElementGetter: undefined, | ||
AnimationEngine: undefined, | ||
updateFunction: () => Promise.resolve(), | ||
@@ -101,7 +100,2 @@ bootstrappedModule: null, | ||
skipLocationChangeOnNonImperativeRoutingTriggers(module, options); | ||
// Cleanup resources, especially remove event listeners thus they will not be added | ||
// twice when application gets bootstrapped the second time. | ||
module.onDestroy(() => { | ||
singleSpaPlatformLocation.destroyApplication(zoneIdentifier); | ||
}); | ||
} | ||
@@ -116,45 +110,9 @@ bootstrappedOptions.bootstrappedNgZone = ngZone; | ||
} | ||
// eslint-disable-next-line @typescript-eslint/no-unused-vars | ||
function unmount(options, props) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
if (options.Router) { | ||
// Workaround for https://github.com/angular/angular/issues/19079 | ||
const router = options.bootstrappedModule.injector.get(options.Router); | ||
router.dispose(); | ||
} | ||
function unmount(options) { | ||
return Promise.resolve().then(() => { | ||
if (options.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', options.routingEventListener); | ||
} | ||
if (options.AnimationEngine) { | ||
/* | ||
The BrowserAnimationsModule does not clean up after itself :'(. When you unmount/destroy the main module, the | ||
BrowserAnimationsModule uses an AnimationRenderer thing to remove dom elements from the page. But the AnimationRenderer | ||
defers the actual work to the TransitionAnimationEngine to do this, and the TransitionAnimationEngine doesn't actually | ||
remove the dom node, but just calls "markElementAsRemoved()". | ||
See https://github.com/angular/angular/blob/db62ccf9eb46ee89366ade586365ea027bb93eb1/packages/animations/browser/src/render/transition_animation_engine.ts#L717 | ||
What markAsRemovedDoes is put it into an array called "collectedLeaveElements", which is all the elements that should be removed | ||
after the DOM has had a chance to do any animations. | ||
See https://github.com/angular/angular/blob/master/packages/animations/browser/src/render/transition_animation_engine.ts#L525 | ||
The actual dom nodes aren't removed until the TransitionAnimationEngine "flushes". | ||
See https://github.com/angular/angular/blob/db62ccf9eb46ee89366ade586365ea027bb93eb1/packages/animations/browser/src/render/transition_animation_engine.ts#L851 | ||
Unfortunately, though, that "flush" will never happen, since the entire module is being destroyed and there will be no more flushes. | ||
So what we do in this code is force one more flush of the animations after the module is destroyed. | ||
Ideally, we would do this by getting the TransitionAnimationEngine directly and flushing it. Unfortunately, though, it's private class | ||
that cannot be imported and is not provided to the dependency injector. So, instead, we get its wrapper class, AnimationEngine, and then | ||
access its private variable reference to the TransitionAnimationEngine so that we can call flush. | ||
*/ | ||
const animationEngine = options.bootstrappedModule.injector.get(options.AnimationEngine); | ||
animationEngine._transitionEngine.flush(); | ||
} | ||
options.bootstrappedModule.destroy(); | ||
options.bootstrappedModule = null; | ||
// TODO: this is not an issue anymore and should be removed in the future. | ||
removeApplicationFromDOMIfIvyEnabled(options, props); | ||
}); | ||
@@ -189,2 +147,2 @@ } | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-spa-angular.js","sourceRoot":"../../../libs/single-spa-angular/","sources":["src/single-spa-angular.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgModuleRef, NgZone } from '@angular/core';\nimport { Subscription } from 'rxjs';\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  bootstrappedModule: null,\n};\n\nexport function singleSpaAngular<T>(userOptions: SingleSpaAngularOptions<T>): LifeCycles<T> {\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  if (options.Router && !options.NavigationStart) {\n    // We call `console.warn` except of throwing `new Error()` since this will not\n    // be a breaking change.\n    console.warn(`single-spa-angular must be passed the NavigationStart 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(`\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      skipLocationChangeOnNonImperativeRoutingTriggers(module, options);\n\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(() => {\n        singleSpaPlatformLocation.destroyApplication(zoneIdentifier);\n      });\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  options.bootstrappedModule = null;\n\n  // TODO: this is not an issue anymore and should be removed in the future.\n  removeApplicationFromDOMIfIvyEnabled(options, props);\n}\n\nfunction skipLocationChangeOnNonImperativeRoutingTriggers(\n  module: NgModuleRef<any>,\n  options: SingleSpaAngularOptions,\n): void {\n  if (!options.NavigationStart) {\n    // As discussed we don't do anything right now if the developer doesn't provide\n    // `options.NavigationStart` since this might be a breaking change.\n    return;\n  }\n\n  const router = module.injector.get(options.Router);\n  const subscription: Subscription = router.events.subscribe((event: any) => {\n    if (event instanceof options.NavigationStart!) {\n      const currentNavigation = router.getCurrentNavigation();\n      // This listener will be set up for each Angular application\n      // that has routing capabilities.\n      // We set `skipLocationChange` for each non-imperative navigation,\n      // Angular router checks under the hood if it has to change\n      // the browser URL or not.\n      // If `skipLocationChange` is truthy then Angular router will not call\n      // `setBrowserUrl()` which calls `history.replaceState()` and dispatches `popstate` event.\n      if (currentNavigation.trigger !== 'imperative') {\n        currentNavigation.extras.skipLocationChange = true;\n        currentNavigation.extras.replaceUrl = false;\n      }\n    }\n  });\n\n  module.onDestroy(() => {\n    subscription.unsubscribe();\n  });\n}\n"]} | ||
//# sourceMappingURL=single-spa-angular.js.map |
export {}; | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiLi4vLi4vLi4vbGlicy9zaW5nbGUtc3BhLWFuZ3VsYXIvIiwic291cmNlcyI6WyJzcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlUmVmLCBUeXBlLCBOZ1pvbmUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEFwcFByb3BzIH0gZnJvbSAnc2luZ2xlLXNwYSc7XG5pbXBvcnQgeyBCYXNlU2luZ2xlU3BhQW5ndWxhck9wdGlvbnMgfSBmcm9tICdzaW5nbGUtc3BhLWFuZ3VsYXIvaW50ZXJuYWxzJztcblxuZXhwb3J0IGludGVyZmFjZSBTaW5nbGVTcGFBbmd1bGFyT3B0aW9uczxUID0ge30+IGV4dGVuZHMgQmFzZVNpbmdsZVNwYUFuZ3VsYXJPcHRpb25zIHtcbiAgLy8gVGhpcyBtaWdodCBiZSBgbm9vcGAgaWYgdGhlIHJvb3QgbW9kdWxlIGlzIGJvb3RzdHJhcHBlZFxuICAvLyB3aXRoIGB7IG5nWm9uZTogJ25vb3AnIH1gIG9wdGlvbnMuXG4gIE5nWm9uZTogdHlwZW9mIE5nWm9uZSB8ICdub29wJztcbiAgdXBkYXRlRnVuY3Rpb24/KHByb3BzOiBBcHBQcm9wcyk6IFByb21pc2U8YW55PjtcbiAgLy8gYFJvdXRlcmAgYW5kIGBOYXZpZ2F0aW9uU3RhcnRgIHNob3VsZCBiZSBhbHdheXMgcHJvdmlkZWQgdG9nZXRoZXIuXG4gIFJvdXRlcj86IFR5cGU8YW55PjtcbiAgTmF2aWdhdGlvblN0YXJ0PzogVHlwZTxhbnk+O1xuICBBbmltYXRpb25FbmdpbmU/OiBUeXBlPGFueT47XG4gIGJvb3RzdHJhcEZ1bmN0aW9uKHByb3BzOiBBcHBQcm9wcyAmIFQpOiBQcm9taXNlPE5nTW9kdWxlUmVmPGFueT4+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEJvb3RzdHJhcHBlZFNpbmdsZVNwYUFuZ3VsYXJPcHRpb25zIGV4dGVuZHMgU2luZ2xlU3BhQW5ndWxhck9wdGlvbnMge1xuICBib290c3RyYXBwZWRNb2R1bGU6IE5nTW9kdWxlUmVmPGFueT4gfCBudWxsO1xuICAvLyBBbGwgYmVsb3cgcHJvcGVydGllcyBjYW4gYmUgb3B0aW9uYWwgaW4gY2FzZSBvZlxuICAvLyBgU2luZ2xlU3BhQW5ndWxhck9wdHMuTmdab25lYCBpcyBhIGBub29wYCBzdHJpbmcgYW5kIG5vdCBhbiBgTmdab25lYCBjbGFzcy5cbiAgYm9vdHN0cmFwcGVkTmdab25lPzogTmdab25lO1xuICByb3V0aW5nRXZlbnRMaXN0ZW5lcj86ICgpID0+IHZvaWQ7XG4gIHpvbmVJZGVudGlmaWVyPzogc3RyaW5nO1xufVxuIl19 | ||
//# sourceMappingURL=types.js.map |
@@ -1,12 +0,1 @@ | ||
function removeApplicationFromDOMIfIvyEnabled(options, props) { | ||
if (ivyEnabled()) { | ||
const domElementGetter = chooseDomElementGetter(options, props); | ||
const domElement = getContainerElement(domElementGetter); | ||
// View Engine removes all nodes automatically when calling `NgModuleRef.destroy()`, | ||
// which calls `ComponentRef.destroy()`. | ||
// Basically this will remove `app-root` or any other selector from the container element. | ||
while (domElement.firstChild) | ||
domElement.removeChild(domElement.firstChild); | ||
} | ||
} | ||
function getContainerElementAndSetTemplate(options, props) { | ||
@@ -56,16 +45,2 @@ const domElementGetter = chooseDomElementGetter(options, props); | ||
} | ||
function ivyEnabled() { | ||
try { | ||
// `ɵivyEnabled` variable is exposed starting from version 8. | ||
// We use `require` here except of a single `import { ɵivyEnabled }` because the | ||
// developer can use Angular version that doesn't expose it (all versions <8). | ||
// The `catch` statement will handle those cases. | ||
// eslint-disable-next-line | ||
const { ɵivyEnabled } = require('@angular/core'); | ||
return !!ɵivyEnabled; | ||
} | ||
catch (_a) { | ||
return false; | ||
} | ||
} | ||
@@ -76,3 +51,3 @@ /** | ||
export { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled }; | ||
export { getContainerElementAndSetTemplate }; | ||
//# sourceMappingURL=single-spa-angular-internals.js.map |
@@ -95,2 +95,3 @@ import { Component, ChangeDetectionStrategy, ElementRef, Input, NgModule } from '@angular/core'; | ||
]; | ||
/** @nocollapse */ | ||
ParcelComponent.ctorParameters = () => [ | ||
@@ -97,0 +98,0 @@ { type: ElementRef } |
import { __awaiter } from 'tslib'; | ||
import { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled } from 'single-spa-angular/internals'; | ||
import { getContainerElementAndSetTemplate } from 'single-spa-angular/internals'; | ||
import { Injectable, Inject } from '@angular/core'; | ||
@@ -12,39 +12,4 @@ import { ɵBrowserPlatformLocation, DOCUMENT, PlatformLocation } from '@angular/common'; | ||
this.skipNextPopState = false; | ||
// 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; | ||
this.source = 'Window.addEventListener:popstate'; | ||
} | ||
destroyApplication(zoneIdentifier) { | ||
// TLDR: Angular adds `popstate` event listener and then doesn't remove it when application gets destroyed. | ||
// Basically, Angular has a potentional memory leak. The `ɵBrowserPlatformLocation` | ||
// has `onPopState` method which adds `popstate` event listener and forgets, see here: | ||
// https://github.com/angular/angular/blob/14be55c9facf3e47b8c97df4502dc3f0f897da03/packages/common/src/location/platform_location.ts#L126 | ||
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); | ||
} | ||
} | ||
this.zoneToOnPopStateListenersMap.delete(zone); | ||
} | ||
pushState(state, title, url) { | ||
@@ -68,3 +33,3 @@ this.skipNextPopState = true; | ||
// See https://github.com/single-spa/single-spa-angular/issues/94 for more details | ||
fn = zone.wrap(fn, `${this.source++}`); | ||
fn = zone.wrap(fn, this.source); | ||
const onPopStateListener = (event) => { | ||
@@ -83,14 +48,4 @@ // The `LocationChangeEvent` doesn't have the `singleSpa` property, since it's added | ||
}; | ||
this.storeOnPopStateListener(zone, onPopStateListener); | ||
super.onPopState(onPopStateListener); | ||
return 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. | ||
const onPopStateListeners = this.zoneToOnPopStateListenersMap.get(zone) || []; | ||
onPopStateListeners.push(onPopStateListener); | ||
if (!this.zoneToOnPopStateListenersMap.has(zone)) { | ||
this.zoneToOnPopStateListenersMap.set(zone, onPopStateListeners); | ||
} | ||
} | ||
} | ||
@@ -109,3 +64,2 @@ SingleSpaPlatformLocation.decorators = [ | ||
provide: SingleSpaPlatformLocation, | ||
useClass: SingleSpaPlatformLocation, | ||
deps: [[new Inject(DOCUMENT)]], | ||
@@ -128,3 +82,2 @@ }, | ||
domElementGetter: undefined, | ||
AnimationEngine: undefined, | ||
updateFunction: () => Promise.resolve(), | ||
@@ -217,7 +170,2 @@ bootstrappedModule: null, | ||
skipLocationChangeOnNonImperativeRoutingTriggers(module, options); | ||
// Cleanup resources, especially remove event listeners thus they will not be added | ||
// twice when application gets bootstrapped the second time. | ||
module.onDestroy(() => { | ||
singleSpaPlatformLocation.destroyApplication(zoneIdentifier); | ||
}); | ||
} | ||
@@ -232,45 +180,9 @@ bootstrappedOptions.bootstrappedNgZone = ngZone; | ||
} | ||
// eslint-disable-next-line @typescript-eslint/no-unused-vars | ||
function unmount(options, props) { | ||
return __awaiter(this, void 0, void 0, function* () { | ||
if (options.Router) { | ||
// Workaround for https://github.com/angular/angular/issues/19079 | ||
const router = options.bootstrappedModule.injector.get(options.Router); | ||
router.dispose(); | ||
} | ||
function unmount(options) { | ||
return Promise.resolve().then(() => { | ||
if (options.routingEventListener) { | ||
window.removeEventListener('single-spa:routing-event', options.routingEventListener); | ||
} | ||
if (options.AnimationEngine) { | ||
/* | ||
The BrowserAnimationsModule does not clean up after itself :'(. When you unmount/destroy the main module, the | ||
BrowserAnimationsModule uses an AnimationRenderer thing to remove dom elements from the page. But the AnimationRenderer | ||
defers the actual work to the TransitionAnimationEngine to do this, and the TransitionAnimationEngine doesn't actually | ||
remove the dom node, but just calls "markElementAsRemoved()". | ||
See https://github.com/angular/angular/blob/db62ccf9eb46ee89366ade586365ea027bb93eb1/packages/animations/browser/src/render/transition_animation_engine.ts#L717 | ||
What markAsRemovedDoes is put it into an array called "collectedLeaveElements", which is all the elements that should be removed | ||
after the DOM has had a chance to do any animations. | ||
See https://github.com/angular/angular/blob/master/packages/animations/browser/src/render/transition_animation_engine.ts#L525 | ||
The actual dom nodes aren't removed until the TransitionAnimationEngine "flushes". | ||
See https://github.com/angular/angular/blob/db62ccf9eb46ee89366ade586365ea027bb93eb1/packages/animations/browser/src/render/transition_animation_engine.ts#L851 | ||
Unfortunately, though, that "flush" will never happen, since the entire module is being destroyed and there will be no more flushes. | ||
So what we do in this code is force one more flush of the animations after the module is destroyed. | ||
Ideally, we would do this by getting the TransitionAnimationEngine directly and flushing it. Unfortunately, though, it's private class | ||
that cannot be imported and is not provided to the dependency injector. So, instead, we get its wrapper class, AnimationEngine, and then | ||
access its private variable reference to the TransitionAnimationEngine so that we can call flush. | ||
*/ | ||
const animationEngine = options.bootstrappedModule.injector.get(options.AnimationEngine); | ||
animationEngine._transitionEngine.flush(); | ||
} | ||
options.bootstrappedModule.destroy(); | ||
options.bootstrappedModule = null; | ||
// TODO: this is not an issue anymore and should be removed in the future. | ||
removeApplicationFromDOMIfIvyEnabled(options, props); | ||
}); | ||
@@ -277,0 +189,0 @@ } |
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; |
export * from './types'; | ||
export { getContainerElementAndSetTemplate, removeApplicationFromDOMIfIvyEnabled } from './dom'; | ||
export { getContainerElementAndSetTemplate } from './dom'; |
@@ -11,2 +11,2 @@ { | ||
"name": "single-spa-angular/internals" | ||
} | ||
} |
@@ -1,1 +0,1 @@ | ||
{"__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"} | ||
{"__symbolic":"module","version":4,"metadata":{"DomElementGetter":{"__symbolic":"interface"},"BaseSingleSpaAngularOptions":{"__symbolic":"interface"},"getContainerElementAndSetTemplate":{"__symbolic":"function"}},"origins":{"DomElementGetter":"./types","BaseSingleSpaAngularOptions":"./types","getContainerElementAndSetTemplate":"./dom"},"importAs":"single-spa-angular/internals"} |
@@ -1,6 +0,5 @@ | ||
import { BrowserBuilderOptions } from '@angular-devkit/build-angular'; | ||
export interface DefaultExtraOptions { | ||
removeMiniCssExtract: boolean; | ||
} | ||
interface Options extends Partial<BrowserBuilderOptions> { | ||
interface Options { | ||
customWebpackConfig?: { | ||
@@ -10,4 +9,9 @@ libraryName?: string; | ||
}; | ||
main?: string; | ||
sourceMap?: boolean | { | ||
scripts?: boolean; | ||
vendor?: boolean; | ||
}; | ||
} | ||
declare const _default: (config: any, options?: Options | undefined, extraOptions?: DefaultExtraOptions | undefined) => any; | ||
export default _default; |
@@ -6,2 +6,3 @@ "use strict"; | ||
const find_up_1 = require("@angular/cli/utilities/find-up"); | ||
const remove_mini_css_extract_1 = require("./webpack-5/remove-mini-css-extract"); | ||
const defaultExtraOptions = { | ||
@@ -18,3 +19,4 @@ removeMiniCssExtract: true, | ||
libraryTarget: (_b = (_a = options === null || options === void 0 ? void 0 : options.customWebpackConfig) === null || _a === void 0 ? void 0 : _a.libraryTarget) !== null && _b !== void 0 ? _b : 'umd', | ||
jsonpFunction: 'webpackJsonp' + libraryName, | ||
// `output.jsonpFunction` has been renamed to `output.chunkLoadingGlobal` in Webpack 5. | ||
chunkLoadingGlobal: 'webpackJsonp' + libraryName, | ||
devtoolNamespace: libraryName, | ||
@@ -76,12 +78,4 @@ }, | ||
function removeMiniCssExtract(config) { | ||
remove_mini_css_extract_1.removeMiniCssExtractRules(config); | ||
removePluginByName(config.plugins, 'MiniCssExtractPlugin'); | ||
config.module.rules.forEach((rule) => { | ||
if (rule.use) { | ||
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) { | ||
rule.use[cssMiniExtractIndex] = { loader: 'style-loader' }; | ||
} | ||
} | ||
}); | ||
} | ||
@@ -88,0 +82,0 @@ function getLibraryName(options) { |
{ | ||
"$schema": "../node_modules/ng-packagr/ng-package.schema.json", | ||
"name": "single-spa-angular", | ||
"version": "4.9.2", | ||
"version": "5.0.0", | ||
"description": "Helpers for building single-spa applications which use Angular 2", | ||
@@ -27,3 +27,3 @@ "schematics": "./schematics/schematics.json", | ||
"peerDependencies": { | ||
"@angular/core": ">=9", | ||
"@angular/core": ">=12", | ||
"json5": "*", | ||
@@ -40,4 +40,4 @@ "single-spa": ">=4.0.0" | ||
"dependencies": { | ||
"tslib": "^2.0.0" | ||
"tslib": "^2.1.0" | ||
} | ||
} | ||
} |
@@ -11,2 +11,2 @@ { | ||
"name": "single-spa-angular/parcel" | ||
} | ||
} |
@@ -9,3 +9,3 @@ import { ElementRef, OnChanges, OnDestroy, OnInit } from '@angular/core'; | ||
wrapWith: string; | ||
customProps: object; | ||
customProps: Record<string, unknown>; | ||
appendTo: Node | null; | ||
@@ -12,0 +12,0 @@ handleError: (error: Error) => void; |
@@ -1,7 +0,3 @@ | ||
import { Rule, SchematicContext, Tree } from '@angular-devkit/schematics'; | ||
import { Rule } from '@angular-devkit/schematics'; | ||
import { Schema as NgAddOptions } from './schema'; | ||
export default function (options: NgAddOptions): Rule; | ||
export declare function addDependencies(): Rule; | ||
export declare function createMainEntry(options: NgAddOptions): Rule; | ||
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 tslib_1 = require("tslib"); | ||
const core_1 = require("@angular-devkit/core"); | ||
const schematics_1 = require("@angular-devkit/schematics"); | ||
const dependencies_1 = require("@schematics/angular/utility/dependencies"); | ||
const core_1 = require("@angular-devkit/core"); | ||
// The JSON5 format supports comments and all Angular projects, | ||
// starting from version 10, contain comments in `tsconfig` files. | ||
const json5_1 = require("json5"); | ||
const add_scripts_1 = require("./add-scripts"); | ||
const dependencies_2 = require("./dependencies"); | ||
const workspace_1 = require("./workspace"); | ||
const add_dependencies_1 = require("./rules/add-dependencies"); | ||
const create_main_entry_1 = require("./rules/create-main-entry"); | ||
const update_configuration_1 = require("./rules/update-configuration"); | ||
const add_npm_scripts_1 = require("./rules/add-npm-scripts"); | ||
const show_warning_if_routing_is_enabled_1 = require("./rules/show-warning-if-routing-is-enabled"); | ||
function default_1(options) { | ||
return schematics_1.chain([ | ||
addDependencies(), | ||
createMainEntry(options), | ||
updateConfiguration(options), | ||
addNPMScripts(options), | ||
]); | ||
return (tree) => tslib_1.__awaiter(this, void 0, void 0, function* () { | ||
const { workspace, project, host } = yield getWorkspace(tree, options.project); | ||
return schematics_1.chain([ | ||
add_dependencies_1.addDependencies(), | ||
create_main_entry_1.createMainEntry(project, options), | ||
update_configuration_1.updateConfiguration(workspace, project, host, options), | ||
add_npm_scripts_1.addNPMScripts(workspace, project, host, options), | ||
show_warning_if_routing_is_enabled_1.showWarningIfRoutingIsEnabled(options), | ||
]); | ||
}); | ||
} | ||
exports.default = default_1; | ||
function addDependencies() { | ||
const dependencies = [ | ||
dependencies_2.getSingleSpaDependency(), | ||
dependencies_2.getSingleSpaAngularDependency(), | ||
dependencies_2.getAngularBuildersCustomWebpackDependency(), | ||
]; | ||
return (tree, context) => tslib_1.__awaiter(this, void 0, void 0, function* () { | ||
var e_1, _a; | ||
try { | ||
for (var dependencies_3 = tslib_1.__asyncValues(dependencies), dependencies_3_1; dependencies_3_1 = yield dependencies_3.next(), !dependencies_3_1.done;) { | ||
const dependency = dependencies_3_1.value; | ||
dependencies_1.addPackageJsonDependency(tree, dependency); | ||
context.logger.info(`Added '${dependency.name}' as a dependency`); | ||
} | ||
function getWorkspace(tree, projectName) { | ||
return tslib_1.__awaiter(this, void 0, void 0, function* () { | ||
const host = createVirtualHost(tree); | ||
const { workspace } = yield core_1.workspaces.readWorkspace('/', host); | ||
projectName = projectName || workspace.extensions.defaultProject; | ||
const project = workspace.projects.get(projectName); | ||
if (!project) { | ||
throw new schematics_1.SchematicsException(`Invalid project name: ${projectName}`); | ||
} | ||
catch (e_1_1) { e_1 = { error: e_1_1 }; } | ||
finally { | ||
try { | ||
if (dependencies_3_1 && !dependencies_3_1.done && (_a = dependencies_3.return)) yield _a.call(dependencies_3); | ||
} | ||
finally { if (e_1) throw e_1.error; } | ||
} | ||
return { workspace, project, host }; | ||
}); | ||
} | ||
exports.addDependencies = addDependencies; | ||
function createMainEntry(options) { | ||
return (host, context) => { | ||
const project = getClientProject(host, options); | ||
const path = core_1.normalize(project.workspace.root); | ||
const templateSource = schematics_1.apply(schematics_1.url('./_files'), [ | ||
schematics_1.applyTemplates({ | ||
prefix: project.workspace.prefix, | ||
routing: options.routing, | ||
usingBrowserAnimationsModule: options.usingBrowserAnimationsModule, | ||
}), | ||
schematics_1.move(path), | ||
]); | ||
const rule = schematics_1.mergeWith(templateSource, schematics_1.MergeStrategy.Overwrite); | ||
context.logger.info(`Generated 'main.single-spa.ts`); | ||
context.logger.info(`Generated 'single-spa-props.ts`); | ||
context.logger.info(`Generated asset-url.ts`); | ||
context.logger.info(`Generated extra-webpack.config.js`); | ||
return rule(host, context); | ||
function createVirtualHost(tree) { | ||
return { | ||
readFile(path) { | ||
return tslib_1.__awaiter(this, void 0, void 0, function* () { | ||
const data = tree.read(path); | ||
if (!data) { | ||
throw new schematics_1.SchematicsException('File not found.'); | ||
} | ||
return core_1.virtualFs.fileBufferToString(data); | ||
}); | ||
}, | ||
writeFile(path, data) { | ||
return tslib_1.__awaiter(this, void 0, void 0, function* () { | ||
return tree.overwrite(path, data); | ||
}); | ||
}, | ||
isDirectory(path) { | ||
return tslib_1.__awaiter(this, void 0, void 0, function* () { | ||
return !tree.exists(path) && tree.getDir(path).subfiles.length > 0; | ||
}); | ||
}, | ||
isFile(path) { | ||
return tslib_1.__awaiter(this, void 0, void 0, function* () { | ||
return tree.exists(path); | ||
}); | ||
}, | ||
}; | ||
} | ||
exports.createMainEntry = createMainEntry; | ||
function updateConfiguration(options) { | ||
return (host, context) => { | ||
const workspace = workspace_1.getWorkspace(host); | ||
const project = getClientProject(host, options); | ||
const clientProject = workspace.projects[project.name]; | ||
if (!clientProject.architect) { | ||
throw new Error('Client project architect not found.'); | ||
} | ||
const workspacePath = workspace_1.getWorkspacePath(host); | ||
updateProjectNewAngular(context, clientProject, project.name); | ||
updateTSConfig(host, clientProject); | ||
host.overwrite(workspacePath, JSON.stringify(workspace, null, 2)); | ||
context.logger.info(`Updated angular.json configuration`); | ||
// @ts-ignore | ||
context.logger.info(clientProject.architect.build.builder); | ||
return host; | ||
}; | ||
} | ||
exports.updateConfiguration = updateConfiguration; | ||
function updateProjectNewAngular(context, clientProject, projectName) { | ||
context.logger.info('Using @angular-devkit/custom-webpack builder.'); | ||
const buildTarget = clientProject.architect.build; | ||
const browserBuilder = '@angular-builders/custom-webpack:browser'; | ||
buildTarget.builder = browserBuilder; | ||
buildTarget.options.main = core_1.join(core_1.normalize(clientProject.root), core_1.normalize('src/main.single-spa.ts')); | ||
buildTarget.options.customWebpackConfig = { | ||
path: core_1.join(core_1.normalize(clientProject.root), 'extra-webpack.config.js'), | ||
libraryName: projectName, | ||
libraryTarget: 'umd', | ||
}; | ||
updateConfigurationsAndDisableOutputHashing(clientProject); | ||
const devServerBuilder = '@angular-builders/custom-webpack:dev-server'; | ||
clientProject.architect.serve.builder = devServerBuilder; | ||
} | ||
function updateTSConfig(host, clientProject) { | ||
const tsConfigPath = clientProject.architect.build.options.tsConfig; | ||
const buffer = host.read(tsConfigPath); | ||
if (buffer === null) { | ||
return; | ||
} | ||
const tsConfig = json5_1.parse(buffer.toString()); | ||
if (!Array.isArray(tsConfig.files)) { | ||
return; | ||
} | ||
// The "files" property will only contain path to `main.single-spa.ts` file, | ||
// because we remove `polyfills` from Webpack `entry` property. | ||
tsConfig.files = [core_1.normalize('src/main.single-spa.ts')]; | ||
host.overwrite(tsConfigPath, JSON.stringify(tsConfig, null, 2)); | ||
} | ||
function addNPMScripts(options) { | ||
return (host) => { | ||
const pkgPath = '/package.json'; | ||
const buffer = host.read(pkgPath); | ||
if (buffer === null) { | ||
throw new schematics_1.SchematicsException('Could not find package.json'); | ||
} | ||
add_scripts_1.addScripts(host, pkgPath, JSON.parse(buffer.toString()), options.project); | ||
}; | ||
} | ||
exports.addNPMScripts = addNPMScripts; | ||
function getClientProject(host, options) { | ||
const workspace = workspace_1.getWorkspace(host); | ||
let project = options.project; | ||
if (!options.project) { | ||
project = Object.keys(workspace.projects)[0]; | ||
} | ||
const clientProject = workspace.projects[project]; | ||
if (!clientProject) { | ||
throw new schematics_1.SchematicsException(`Client app ${options.project} not found.`); | ||
} | ||
return { name: project, workspace: clientProject }; | ||
} | ||
function updateConfigurationsAndDisableOutputHashing(clientProject) { | ||
const configurations = clientProject.architect.build.configurations; | ||
// If the user doesn't have any `configurations` then just skip this step. | ||
if (typeof configurations !== 'object') { | ||
return; | ||
} | ||
for (const configuration of Object.values(configurations)) { | ||
configuration.outputHashing = 'none'; | ||
} | ||
} | ||
//# sourceMappingURL=index.js.map |
export interface Schema { | ||
project?: string; | ||
routing?: boolean; | ||
usingBrowserAnimationsModule?: boolean; | ||
// This actually will still infer to `number` type, this is done to specify | ||
// that `4200` is used by default. | ||
port?: 4200 | number; | ||
} |
{ | ||
"$schema": "http://json-schema.org/schema", | ||
"id": "single-spa-ng-add", | ||
"$id": "single-spa-ng-add", | ||
"title": "single-spa ng-add schematic", | ||
@@ -20,7 +20,7 @@ "type": "object", | ||
}, | ||
"usingBrowserAnimationsModule": { | ||
"type": "boolean", | ||
"description": "A value of true means the angular application uses BrowserAnimationsModule", | ||
"default": false, | ||
"x-prompt": "Does your application use BrowserAnimationsModule?" | ||
"port": { | ||
"type": "number", | ||
"description": "Port to listen on", | ||
"default": 4200, | ||
"x-prompt": "What port should your project run on?" | ||
} | ||
@@ -27,0 +27,0 @@ }, |
@@ -1,1 +0,1 @@ | ||
{"__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"}}]}},"origins":{"ɵa":"./src/extra-providers","singleSpaAngular":"./src/single-spa-angular","getSingleSpaExtraProviders":"./src/extra-providers"},"importAs":"single-spa-angular"} | ||
{"__symbolic":"module","version":4,"metadata":{"ɵa":{"__symbolic":"class","extends":{"__symbolic":"reference","module":"@angular/common","name":"ɵBrowserPlatformLocation","line":12,"character":47},"decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Injectable","line":11,"character":1}}],"members":{"pushState":[{"__symbolic":"method"}],"replaceState":[{"__symbolic":"method"}],"onPopState":[{"__symbolic":"method"}]}},"singleSpaAngular":{"__symbolic":"function"},"getSingleSpaExtraProviders":{"__symbolic":"function","parameters":[],"value":[{"provide":{"__symbolic":"reference","name":"ɵa"},"deps":[[{"__symbolic":"new","expression":{"__symbolic":"reference","module":"@angular/core","name":"Inject","line":69,"character":18},"arguments":[{"__symbolic":"reference","module":"@angular/common","name":"DOCUMENT","line":69,"character":25}]}]]},{"provide":{"__symbolic":"reference","module":"@angular/common","name":"PlatformLocation","line":72,"character":15},"useExisting":{"__symbolic":"reference","name":"ɵa"}}]}},"origins":{"ɵa":"./src/extra-providers","singleSpaAngular":"./src/single-spa-angular","getSingleSpaExtraProviders":"./src/extra-providers"},"importAs":"single-spa-angular"} |
import { StaticProvider } from '@angular/core'; | ||
import { ɵBrowserPlatformLocation, LocationChangeEvent } from '@angular/common'; | ||
declare type OnPopStateListener = (event: LocationChangeEvent) => void; | ||
import { ɵBrowserPlatformLocation, LocationChangeListener } from '@angular/common'; | ||
export declare class SingleSpaPlatformLocation extends ɵBrowserPlatformLocation { | ||
private skipNextPopState; | ||
private zoneToOnPopStateListenersMap; | ||
private source; | ||
destroyApplication(zoneIdentifier: string): void; | ||
private readonly source; | ||
pushState(state: any, title: string, url: string): void; | ||
replaceState(state: any, title: string, url: string): void; | ||
onPopState(fn: OnPopStateListener): void; | ||
private storeOnPopStateListener; | ||
onPopState(fn: LocationChangeListener): VoidFunction; | ||
} | ||
@@ -20,2 +16,1 @@ /** | ||
export declare function getSingleSpaExtraProviders(): StaticProvider[]; | ||
export {}; |
import { NgModuleRef, Type, NgZone } from '@angular/core'; | ||
import { AppProps } from 'single-spa'; | ||
import { BaseSingleSpaAngularOptions } from 'single-spa-angular/internals'; | ||
export interface SingleSpaAngularOptions<T = {}> extends BaseSingleSpaAngularOptions { | ||
export interface SingleSpaAngularOptions<T = Record<string, unknown>> extends BaseSingleSpaAngularOptions { | ||
NgZone: typeof NgZone | 'noop'; | ||
@@ -9,3 +9,2 @@ updateFunction?(props: AppProps): Promise<any>; | ||
NavigationStart?: Type<any>; | ||
AnimationEngine?: Type<any>; | ||
bootstrapFunction(props: AppProps & T): Promise<NgModuleRef<any>>; | ||
@@ -12,0 +11,0 @@ } |
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
Sorry, the diff of this file is not supported yet
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
132
0
309281
2831
Updatedtslib@^2.1.0