@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
143 lines • 40.5 kB
JavaScript
import { Component, ViewChild } from '@angular/core';
import { InventoryService } from '@c8y/client';
import { PluginsService } from '@c8y/ngx-components';
import { EcosystemService } from '@c8y/ngx-components/ecosystem/shared';
import { PluginsExportScopes } from '@c8y/ngx-components';
import { BehaviorSubject } from 'rxjs';
import * as i0 from "@angular/core";
import * as i1 from "@c8y/client";
import * as i2 from "@c8y/ngx-components/ecosystem/shared";
import * as i3 from "@c8y/ngx-components";
import * as i4 from "@angular/common";
import * as i5 from "../../application-plugins/plugin-list.component";
export class UpdateApplicationModalComponent {
constructor(inventoryService, ecosystemService, pluginsService) {
this.inventoryService = inventoryService;
this.ecosystemService = ecosystemService;
this.pluginsService = pluginsService;
this.isUpdateOngoing = false;
this.updateFailure = false;
this.result = new Promise((resolve, reject) => {
this._resolve = resolve;
this._reject = reject;
});
this.orphanedPlugins$ = new BehaviorSubject([]);
this.newPlugins$ = new BehaviorSubject([]);
}
ngOnInit() {
const hasDelta = this.getRemoteDelta();
if (!hasDelta) {
// start update directly if no remote delta detected
this.updateApplication();
}
}
/**
* Checks if there is a delta between the current application and the source package remotes.
* @returns {boolean} true if there is a delta between the current application and the source package remotes.
*/
getRemoteDelta() {
const currentRemotes = this.pluginsService.getMFExports(this.application, []);
const possibleNewRemotes = this.pluginsService.getMFExports(this.sourcePackage, []);
const installedRemotes = this.pluginsService.getMFRemotes(this.application);
const installedRemotesIds = PluginsService.convertInstalledRemotesToIds(installedRemotes);
const allRemotesToRemove = this.getAllRemotesToRemove(currentRemotes, possibleNewRemotes, installedRemotesIds);
const allRemotesToAdd = this.getAllRemotesToAdd(possibleNewRemotes, currentRemotes);
this.orphanedPlugins$.next(allRemotesToRemove);
this.newPlugins$.next(allRemotesToAdd);
return allRemotesToRemove.length !== 0 || allRemotesToAdd.length !== 0;
}
close() {
this._reject();
this.modal._dismiss();
}
done() {
if (!this.updateFailure) {
this._resolve();
this.modal._dismiss();
return;
}
this._reject();
this.modal._dismiss();
}
async updateApplication() {
try {
this.isUpdateOngoing = true;
const toInstallPlugins = this.newPlugins$.value.filter(({ selected }) => selected === true);
const toRemovePlugins = this.orphanedPlugins$.value;
this.orphanedPlugins$.next([]);
this.newPlugins$.next([]);
if (toRemovePlugins.length > 0) {
const remotes = await this.pluginsService.removeRemotes(this.application, toRemovePlugins);
this.application.config.remotes = remotes;
}
if (toInstallPlugins.length > 0) {
const remotes = await this.pluginsService.addRemotes(this.application, toInstallPlugins);
this.application.config.remotes = remotes;
}
const binaryMoId = this.sourcePackage.activeVersionId;
await this.inventoryService.detail(binaryMoId); // only trying if we can access it
await this.ecosystemService.uploadBinaryFromOtherPackage(this.sourcePackage, this.application, this.sourcePackage.activeVersionId);
}
catch (e) {
if (e.res?.status === 404) {
try {
this.updateFailure = !(await this.ecosystemService.fallbackToClone(this.application, this.sourcePackage));
}
catch (ex) {
this.updateFailure = true;
this.ecosystemService.alertError(e);
}
}
else {
this.updateFailure = true;
this.ecosystemService.alertError(e);
}
}
finally {
this.isUpdateOngoing = false;
}
}
getAllRemotesToAdd(possibleNewRemotes, currentRemotes) {
return possibleNewRemotes
.filter(possibleNewRemote => {
const isUnchanged = !!currentRemotes.some(currentRemote => PluginsService.createPluginId(this.application.contextPath, possibleNewRemote.module, '', true) === currentRemote.idLatest);
const isSelf = possibleNewRemote.scope === PluginsExportScopes.SELF ||
possibleNewRemote.scope === PluginsExportScopes.SELF_OPTIONAL;
return !isUnchanged && isSelf;
})
.map(newRemote => {
newRemote.contextPath = this.application.contextPath;
newRemote.id = `${newRemote.contextPath}/${newRemote.module}`;
newRemote.selected = newRemote.scope === PluginsExportScopes.SELF;
return newRemote;
});
}
getAllRemotesToRemove(currentRemotes, possibleNewRemotes, installedRemotesIds) {
return currentRemotes
.filter(currentRemote => {
const isUnchanged = !!possibleNewRemotes.some(newRemote => PluginsService.createPluginId(this.application.contextPath, newRemote.module, '', true) === currentRemote.idLatest);
const isInstalled = installedRemotesIds
? !!installedRemotesIds.some(currentInstalled => currentRemote.id === currentInstalled || currentRemote.idLatest === currentInstalled)
: false;
const isSelf = currentRemote.scope === PluginsExportScopes.SELF ||
currentRemote.scope === PluginsExportScopes.SELF_OPTIONAL;
return !isUnchanged && isInstalled && isSelf;
})
.map(toRemoveRemote => {
toRemoveRemote.contextPath = this.application.contextPath;
toRemoveRemote.id = `${toRemoveRemote.contextPath}/${toRemoveRemote.module}`;
toRemoveRemote.selected = true;
return toRemoveRemote;
});
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UpdateApplicationModalComponent, deps: [{ token: i1.InventoryService }, { token: i2.EcosystemService }, { token: i3.PluginsService }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: UpdateApplicationModalComponent, selector: "c8y-update-application-modal", viewQueries: [{ propertyName: "modal", first: true, predicate: ["modal"], descendants: true }], ngImport: i0, template: "<c8y-modal\n [title]=\"'Update application' | translate\"\n [headerClasses]=\"'dialog-header'\"\n [customFooter]=\"true\"\n #modal\n>\n <ng-container c8y-modal-title>\n <span class=\"dlt-c8y-icon-installing-updates\"></span>\n </ng-container>\n\n <ng-container\n *ngIf=\"\n (orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0;\n else updateProgress\n \"\n >\n <p\n class=\"text-center text-break-word p-24 text-14\"\n translate\n >\n Updating this blueprint will change the default plugins. Review the plugin changes before\n proceeding.\n </p>\n <c8y-list-group *ngIf=\"(orphanedPlugins$ | async).length > 0\">\n <c8y-li [collapsed]=\"true\">\n <c8y-li-icon>\n <span class=\"badge badge-danger\">{{ (orphanedPlugins$ | async).length }}</span>\n </c8y-li-icon>\n <c8y-li-body>\n <div translate>Orphaned plugins</div>\n </c8y-li-body>\n <c8y-li-footer translate>\n Some plugins are not contained in the new version of this blueprint and will therefore get\n removed.\n </c8y-li-footer>\n <c8y-li-collapse>\n <c8y-plugin-list\n class=\"m-t-16\"\n [emptyListText]=\"'No plugins available' | translate\"\n [plugins$]=\"orphanedPlugins$\"\n [selectable]=\"false\"\n [hideSource]=\"true\"\n ></c8y-plugin-list>\n </c8y-li-collapse>\n </c8y-li>\n </c8y-list-group>\n\n <c8y-list-group *ngIf=\"(newPlugins$ | async).length > 0\">\n <c8y-li [collapsed]=\"false\">\n <c8y-li-icon>\n <span class=\"badge badge-success\">{{ (newPlugins$ | async).length }}</span>\n </c8y-li-icon>\n <c8y-li-body>\n <div translate>New plugins added</div>\n </c8y-li-body>\n <c8y-li-footer translate>\n This blueprint will add new plugins. Please choose which you want to install.\n </c8y-li-footer>\n <c8y-li-collapse>\n <c8y-plugin-list\n class=\"m-t-16\"\n [emptyListText]=\"'No plugins available' | translate\"\n [plugins$]=\"newPlugins$\"\n [selectable]=\"true\"\n [hideSource]=\"false\"\n ></c8y-plugin-list>\n </c8y-li-collapse>\n </c8y-li>\n </c8y-list-group>\n </ng-container>\n\n <ng-container c8y-modal-footer-custom>\n <div\n class=\"modal-footer\"\n *ngIf=\"\n (orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0;\n else updateProgressButtons\n \"\n >\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n (click)=\"close()\"\n [disabled]=\"isUpdateOngoing\"\n >\n {{ 'Close' | translate }}\n </button>\n <button\n class=\"btn btn-primary\"\n title=\"{{ 'Continue' | translate }}\"\n (click)=\"updateApplication()\"\n *ngIf=\"(orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0\"\n [disabled]=\"isUpdateOngoing\"\n >\n {{ 'Continue' | translate }}\n </button>\n </div>\n </ng-container>\n\n <ng-template #updateProgressButtons>\n <ng-container c8y-modal-footer-custom>\n <div class=\"modal-footer\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n (click)=\"done()\"\n [disabled]=\"isUpdateOngoing\"\n >\n {{ 'Close' | translate }}\n </button>\n </div>\n </ng-container>\n </ng-template>\n\n <ng-template #updateProgress>\n <c8y-loading\n class=\"text-center d-block p-t-56 p-b-56 m-t-4 m-b-4\"\n style=\"min-height: 180px\"\n layout=\"application\"\n *ngIf=\"isUpdateOngoing\"\n [message]=\"'Updating\u2026' | translate\"\n ></c8y-loading>\n\n <c8y-operation-result\n type=\"success\"\n *ngIf=\"!isUpdateOngoing && !updateFailure\"\n text=\"{{ 'Update completed' | translate }}\"\n [size]=\"120\"\n [vertical]=\"true\"\n ></c8y-operation-result>\n <c8y-operation-result\n type=\"error\"\n *ngIf=\"!isUpdateOngoing && updateFailure\"\n text=\"{{ 'Failed to update application.' | translate }}\"\n [size]=\"120\"\n [vertical]=\"true\"\n ></c8y-operation-result>\n </ng-template>\n</c8y-modal>\n", dependencies: [{ kind: "directive", type: i3.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.LoadingComponent, selector: "c8y-loading", inputs: ["layout", "progress", "message"] }, { kind: "component", type: i3.OperationResultComponent, selector: "c8y-operation-result", inputs: ["text", "vertical", "size", "type"] }, { kind: "component", type: i3.ModalComponent, selector: "c8y-modal", inputs: ["disabled", "close", "dismiss", "title", "body", "customFooter", "headerClasses", "labels"], outputs: ["onDismiss", "onClose"] }, { kind: "component", type: i3.ListGroupComponent, selector: "c8y-list-group" }, { kind: "component", type: i3.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i3.ListItemIconComponent, selector: "c8y-list-item-icon, c8y-li-icon", inputs: ["icon", "status"] }, { kind: "component", type: i3.ListItemBodyComponent, selector: "c8y-list-item-body, c8y-li-body", inputs: ["body"] }, { kind: "component", type: i3.ListItemFooterComponent, selector: "c8y-list-item-footer, c8y-li-footer", inputs: ["footer"] }, { kind: "component", type: i3.ListItemCollapseComponent, selector: "c8y-list-item-collapse, c8y-li-collapse", inputs: ["collapseWay"] }, { kind: "component", type: i5.PluginListComponent, selector: "c8y-plugin-list", inputs: ["plugins$", "emptyListText", "selectable", "hideSource", "installable", "package"], outputs: ["selectedItems"] }, { kind: "pipe", type: i3.C8yTranslatePipe, name: "translate" }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UpdateApplicationModalComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-update-application-modal', template: "<c8y-modal\n [title]=\"'Update application' | translate\"\n [headerClasses]=\"'dialog-header'\"\n [customFooter]=\"true\"\n #modal\n>\n <ng-container c8y-modal-title>\n <span class=\"dlt-c8y-icon-installing-updates\"></span>\n </ng-container>\n\n <ng-container\n *ngIf=\"\n (orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0;\n else updateProgress\n \"\n >\n <p\n class=\"text-center text-break-word p-24 text-14\"\n translate\n >\n Updating this blueprint will change the default plugins. Review the plugin changes before\n proceeding.\n </p>\n <c8y-list-group *ngIf=\"(orphanedPlugins$ | async).length > 0\">\n <c8y-li [collapsed]=\"true\">\n <c8y-li-icon>\n <span class=\"badge badge-danger\">{{ (orphanedPlugins$ | async).length }}</span>\n </c8y-li-icon>\n <c8y-li-body>\n <div translate>Orphaned plugins</div>\n </c8y-li-body>\n <c8y-li-footer translate>\n Some plugins are not contained in the new version of this blueprint and will therefore get\n removed.\n </c8y-li-footer>\n <c8y-li-collapse>\n <c8y-plugin-list\n class=\"m-t-16\"\n [emptyListText]=\"'No plugins available' | translate\"\n [plugins$]=\"orphanedPlugins$\"\n [selectable]=\"false\"\n [hideSource]=\"true\"\n ></c8y-plugin-list>\n </c8y-li-collapse>\n </c8y-li>\n </c8y-list-group>\n\n <c8y-list-group *ngIf=\"(newPlugins$ | async).length > 0\">\n <c8y-li [collapsed]=\"false\">\n <c8y-li-icon>\n <span class=\"badge badge-success\">{{ (newPlugins$ | async).length }}</span>\n </c8y-li-icon>\n <c8y-li-body>\n <div translate>New plugins added</div>\n </c8y-li-body>\n <c8y-li-footer translate>\n This blueprint will add new plugins. Please choose which you want to install.\n </c8y-li-footer>\n <c8y-li-collapse>\n <c8y-plugin-list\n class=\"m-t-16\"\n [emptyListText]=\"'No plugins available' | translate\"\n [plugins$]=\"newPlugins$\"\n [selectable]=\"true\"\n [hideSource]=\"false\"\n ></c8y-plugin-list>\n </c8y-li-collapse>\n </c8y-li>\n </c8y-list-group>\n </ng-container>\n\n <ng-container c8y-modal-footer-custom>\n <div\n class=\"modal-footer\"\n *ngIf=\"\n (orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0;\n else updateProgressButtons\n \"\n >\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n (click)=\"close()\"\n [disabled]=\"isUpdateOngoing\"\n >\n {{ 'Close' | translate }}\n </button>\n <button\n class=\"btn btn-primary\"\n title=\"{{ 'Continue' | translate }}\"\n (click)=\"updateApplication()\"\n *ngIf=\"(orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0\"\n [disabled]=\"isUpdateOngoing\"\n >\n {{ 'Continue' | translate }}\n </button>\n </div>\n </ng-container>\n\n <ng-template #updateProgressButtons>\n <ng-container c8y-modal-footer-custom>\n <div class=\"modal-footer\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n (click)=\"done()\"\n [disabled]=\"isUpdateOngoing\"\n >\n {{ 'Close' | translate }}\n </button>\n </div>\n </ng-container>\n </ng-template>\n\n <ng-template #updateProgress>\n <c8y-loading\n class=\"text-center d-block p-t-56 p-b-56 m-t-4 m-b-4\"\n style=\"min-height: 180px\"\n layout=\"application\"\n *ngIf=\"isUpdateOngoing\"\n [message]=\"'Updating\u2026' | translate\"\n ></c8y-loading>\n\n <c8y-operation-result\n type=\"success\"\n *ngIf=\"!isUpdateOngoing && !updateFailure\"\n text=\"{{ 'Update completed' | translate }}\"\n [size]=\"120\"\n [vertical]=\"true\"\n ></c8y-operation-result>\n <c8y-operation-result\n type=\"error\"\n *ngIf=\"!isUpdateOngoing && updateFailure\"\n text=\"{{ 'Failed to update application.' | translate }}\"\n [size]=\"120\"\n [vertical]=\"true\"\n ></c8y-operation-result>\n </ng-template>\n</c8y-modal>\n" }]
}], ctorParameters: () => [{ type: i1.InventoryService }, { type: i2.EcosystemService }, { type: i3.PluginsService }], propDecorators: { modal: [{
type: ViewChild,
args: ['modal', { static: false }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"update-application-modal.component.js","sourceRoot":"","sources":["../../../../../ecosystem/application-properties/update-application-modal/update-application-modal.component.ts","../../../../../ecosystem/application-properties/update-application-modal/update-application-modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, ViewChild } from '@angular/core';\nimport { IApplication, InventoryService } from '@c8y/client';\nimport { ApplicationPlugin, PluginsService } from '@c8y/ngx-components';\nimport { EcosystemService } from '@c8y/ngx-components/ecosystem/shared';\nimport { PluginsExportScopes } from '@c8y/ngx-components';\nimport { BehaviorSubject } from 'rxjs';\n\n@Component({\n  selector: 'c8y-update-application-modal',\n  templateUrl: './update-application-modal.component.html'\n})\nexport class UpdateApplicationModalComponent implements OnInit {\n  @ViewChild('modal', { static: false }) private modal;\n  isUpdateOngoing = false;\n  updateFailure = false;\n  sourcePackage: IApplication;\n  application: IApplication;\n  result: Promise<void> = new Promise((resolve, reject) => {\n    this._resolve = resolve;\n    this._reject = reject;\n  });\n  orphanedPlugins$: BehaviorSubject<ApplicationPlugin[]> = new BehaviorSubject([]);\n  newPlugins$: BehaviorSubject<ApplicationPlugin[]> = new BehaviorSubject([]);\n\n  private _resolve: (value: void | PromiseLike<void>) => void;\n  private _reject: (value: void | PromiseLike<void>) => void;\n\n  constructor(\n    private inventoryService: InventoryService,\n    private ecosystemService: EcosystemService,\n    private pluginsService: PluginsService\n  ) {}\n\n  ngOnInit(): void {\n    const hasDelta = this.getRemoteDelta();\n    if (!hasDelta) {\n      // start update directly if no remote delta detected\n      this.updateApplication();\n    }\n  }\n\n  /**\n   * Checks if there is a delta between the current application and the source package remotes.\n   * @returns {boolean} true if there is a delta between the current application and the source package remotes.\n   */\n  getRemoteDelta() {\n    const currentRemotes = this.pluginsService.getMFExports(this.application, []);\n    const possibleNewRemotes = this.pluginsService.getMFExports(this.sourcePackage, []);\n    const installedRemotes = this.pluginsService.getMFRemotes(this.application);\n    const installedRemotesIds = PluginsService.convertInstalledRemotesToIds(installedRemotes);\n\n    const allRemotesToRemove = this.getAllRemotesToRemove(\n      currentRemotes,\n      possibleNewRemotes,\n      installedRemotesIds\n    );\n\n    const allRemotesToAdd = this.getAllRemotesToAdd(possibleNewRemotes, currentRemotes);\n\n    this.orphanedPlugins$.next(allRemotesToRemove);\n    this.newPlugins$.next(allRemotesToAdd);\n\n    return allRemotesToRemove.length !== 0 || allRemotesToAdd.length !== 0;\n  }\n\n  close() {\n    this._reject();\n    this.modal._dismiss();\n  }\n\n  done() {\n    if (!this.updateFailure) {\n      this._resolve();\n      this.modal._dismiss();\n      return;\n    }\n    this._reject();\n    this.modal._dismiss();\n  }\n\n  async updateApplication() {\n    try {\n      this.isUpdateOngoing = true;\n      const toInstallPlugins = this.newPlugins$.value.filter(({ selected }) => selected === true);\n      const toRemovePlugins = this.orphanedPlugins$.value;\n      this.orphanedPlugins$.next([]);\n      this.newPlugins$.next([]);\n      if (toRemovePlugins.length > 0) {\n        const remotes = await this.pluginsService.removeRemotes(this.application, toRemovePlugins);\n        this.application.config.remotes = remotes;\n      }\n      if (toInstallPlugins.length > 0) {\n        const remotes = await this.pluginsService.addRemotes(this.application, toInstallPlugins);\n        this.application.config.remotes = remotes;\n      }\n      const binaryMoId = this.sourcePackage.activeVersionId;\n      await this.inventoryService.detail(binaryMoId); // only trying if we can access it\n      await this.ecosystemService.uploadBinaryFromOtherPackage(\n        this.sourcePackage,\n        this.application,\n        this.sourcePackage.activeVersionId\n      );\n    } catch (e) {\n      if (e.res?.status === 404) {\n        try {\n          this.updateFailure = !(await this.ecosystemService.fallbackToClone(\n            this.application,\n            this.sourcePackage\n          ));\n        } catch (ex) {\n          this.updateFailure = true;\n          this.ecosystemService.alertError(e);\n        }\n      } else {\n        this.updateFailure = true;\n        this.ecosystemService.alertError(e);\n      }\n    } finally {\n      this.isUpdateOngoing = false;\n    }\n  }\n\n  private getAllRemotesToAdd(\n    possibleNewRemotes: ApplicationPlugin[],\n    currentRemotes: ApplicationPlugin[]\n  ) {\n    return possibleNewRemotes\n      .filter(possibleNewRemote => {\n        const isUnchanged = !!currentRemotes.some(\n          currentRemote =>\n            PluginsService.createPluginId(\n              this.application.contextPath,\n              possibleNewRemote.module,\n              '',\n              true\n            ) === currentRemote.idLatest\n        );\n\n        const isSelf =\n          possibleNewRemote.scope === PluginsExportScopes.SELF ||\n          possibleNewRemote.scope === PluginsExportScopes.SELF_OPTIONAL;\n\n        return !isUnchanged && isSelf;\n      })\n      .map(newRemote => {\n        newRemote.contextPath = this.application.contextPath;\n        newRemote.id = `${newRemote.contextPath}/${newRemote.module}`;\n        newRemote.selected = newRemote.scope === PluginsExportScopes.SELF;\n        return newRemote;\n      });\n  }\n\n  private getAllRemotesToRemove(\n    currentRemotes: ApplicationPlugin[],\n    possibleNewRemotes: ApplicationPlugin[],\n    installedRemotesIds: string[]\n  ) {\n    return currentRemotes\n      .filter(currentRemote => {\n        const isUnchanged = !!possibleNewRemotes.some(\n          newRemote =>\n            PluginsService.createPluginId(\n              this.application.contextPath,\n              newRemote.module,\n              '',\n              true\n            ) === currentRemote.idLatest\n        );\n        const isInstalled = installedRemotesIds\n          ? !!installedRemotesIds.some(\n              currentInstalled =>\n                currentRemote.id === currentInstalled || currentRemote.idLatest === currentInstalled\n            )\n          : false;\n\n        const isSelf =\n          currentRemote.scope === PluginsExportScopes.SELF ||\n          currentRemote.scope === PluginsExportScopes.SELF_OPTIONAL;\n\n        return !isUnchanged && isInstalled && isSelf;\n      })\n      .map(toRemoveRemote => {\n        toRemoveRemote.contextPath = this.application.contextPath;\n        toRemoveRemote.id = `${toRemoveRemote.contextPath}/${toRemoveRemote.module}`;\n        toRemoveRemote.selected = true;\n        return toRemoveRemote;\n      });\n  }\n}\n","<c8y-modal\n  [title]=\"'Update application' | translate\"\n  [headerClasses]=\"'dialog-header'\"\n  [customFooter]=\"true\"\n  #modal\n>\n  <ng-container c8y-modal-title>\n    <span class=\"dlt-c8y-icon-installing-updates\"></span>\n  </ng-container>\n\n  <ng-container\n    *ngIf=\"\n      (orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0;\n      else updateProgress\n    \"\n  >\n    <p\n      class=\"text-center text-break-word p-24 text-14\"\n      translate\n    >\n      Updating this blueprint will change the default plugins. Review the plugin changes before\n      proceeding.\n    </p>\n    <c8y-list-group *ngIf=\"(orphanedPlugins$ | async).length > 0\">\n      <c8y-li [collapsed]=\"true\">\n        <c8y-li-icon>\n          <span class=\"badge badge-danger\">{{ (orphanedPlugins$ | async).length }}</span>\n        </c8y-li-icon>\n        <c8y-li-body>\n          <div translate>Orphaned plugins</div>\n        </c8y-li-body>\n        <c8y-li-footer translate>\n          Some plugins are not contained in the new version of this blueprint and will therefore get\n          removed.\n        </c8y-li-footer>\n        <c8y-li-collapse>\n          <c8y-plugin-list\n            class=\"m-t-16\"\n            [emptyListText]=\"'No plugins available' | translate\"\n            [plugins$]=\"orphanedPlugins$\"\n            [selectable]=\"false\"\n            [hideSource]=\"true\"\n          ></c8y-plugin-list>\n        </c8y-li-collapse>\n      </c8y-li>\n    </c8y-list-group>\n\n    <c8y-list-group *ngIf=\"(newPlugins$ | async).length > 0\">\n      <c8y-li [collapsed]=\"false\">\n        <c8y-li-icon>\n          <span class=\"badge badge-success\">{{ (newPlugins$ | async).length }}</span>\n        </c8y-li-icon>\n        <c8y-li-body>\n          <div translate>New plugins added</div>\n        </c8y-li-body>\n        <c8y-li-footer translate>\n          This blueprint will add new plugins. Please choose which you want to install.\n        </c8y-li-footer>\n        <c8y-li-collapse>\n          <c8y-plugin-list\n            class=\"m-t-16\"\n            [emptyListText]=\"'No plugins available' | translate\"\n            [plugins$]=\"newPlugins$\"\n            [selectable]=\"true\"\n            [hideSource]=\"false\"\n          ></c8y-plugin-list>\n        </c8y-li-collapse>\n      </c8y-li>\n    </c8y-list-group>\n  </ng-container>\n\n  <ng-container c8y-modal-footer-custom>\n    <div\n      class=\"modal-footer\"\n      *ngIf=\"\n        (orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0;\n        else updateProgressButtons\n      \"\n    >\n      <button\n        class=\"btn btn-default\"\n        title=\"{{ 'Close' | translate }}\"\n        (click)=\"close()\"\n        [disabled]=\"isUpdateOngoing\"\n      >\n        {{ 'Close' | translate }}\n      </button>\n      <button\n        class=\"btn btn-primary\"\n        title=\"{{ 'Continue' | translate }}\"\n        (click)=\"updateApplication()\"\n        *ngIf=\"(orphanedPlugins$ | async).length > 0 || (newPlugins$ | async).length > 0\"\n        [disabled]=\"isUpdateOngoing\"\n      >\n        {{ 'Continue' | translate }}\n      </button>\n    </div>\n  </ng-container>\n\n  <ng-template #updateProgressButtons>\n    <ng-container c8y-modal-footer-custom>\n      <div class=\"modal-footer\">\n        <button\n          class=\"btn btn-default\"\n          title=\"{{ 'Close' | translate }}\"\n          (click)=\"done()\"\n          [disabled]=\"isUpdateOngoing\"\n        >\n          {{ 'Close' | translate }}\n        </button>\n      </div>\n    </ng-container>\n  </ng-template>\n\n  <ng-template #updateProgress>\n    <c8y-loading\n      class=\"text-center d-block p-t-56 p-b-56 m-t-4 m-b-4\"\n      style=\"min-height: 180px\"\n      layout=\"application\"\n      *ngIf=\"isUpdateOngoing\"\n      [message]=\"'Updating…' | translate\"\n    ></c8y-loading>\n\n    <c8y-operation-result\n      type=\"success\"\n      *ngIf=\"!isUpdateOngoing && !updateFailure\"\n      text=\"{{ 'Update completed' | translate }}\"\n      [size]=\"120\"\n      [vertical]=\"true\"\n    ></c8y-operation-result>\n    <c8y-operation-result\n      type=\"error\"\n      *ngIf=\"!isUpdateOngoing && updateFailure\"\n      text=\"{{ 'Failed to update application.' | translate }}\"\n      [size]=\"120\"\n      [vertical]=\"true\"\n    ></c8y-operation-result>\n  </ng-template>\n</c8y-modal>\n"]}