@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
242 lines • 42.7 kB
JavaScript
import { Component, EventEmitter, Input, Output } from '@angular/core';
import { ApplicationType } from '@c8y/client';
import { AlertService, GainsightService, PluginsService, gettext, HumanizeAppNamePipe } from '@c8y/ngx-components';
import { EcosystemService, PRODUCT_EXPERIENCE_ECOSYSTEM } from '@c8y/ngx-components/ecosystem/shared';
import { TranslateService } from '@ngx-translate/core';
import { BsModalService } from 'ngx-bootstrap/modal';
import { BehaviorSubject, firstValueFrom, Observable } from 'rxjs';
import { AppsToUpdateRemotesSelectComponent } from './apps-to-update-remotes-select.component';
import { pick } from 'lodash-es';
import * as i0 from "@angular/core";
import * as i1 from "@c8y/ngx-components/ecosystem/shared";
import * as i2 from "ngx-bootstrap/modal";
import * as i3 from "@c8y/ngx-components";
import * as i4 from "@ngx-translate/core";
import * as i5 from "@angular/common";
import * as i6 from "./plugin-list-item.component";
export class PluginListComponent {
constructor(ecosystemService, bsModalService, pluginsService, alertService, translateService, gainsightService, humanizeAppNamePipe) {
this.ecosystemService = ecosystemService;
this.bsModalService = bsModalService;
this.pluginsService = pluginsService;
this.alertService = alertService;
this.translateService = translateService;
this.gainsightService = gainsightService;
this.humanizeAppNamePipe = humanizeAppNamePipe;
this.CURRENT_LOCATION = location.href;
this.emptyListText = '';
this.hideSource = false;
/**
* Shows the install button for each plugin separately. Currently used in package-details view.
*/
this.installable = false;
this.selectedItems = new EventEmitter();
this.remotePlugins$ = new BehaviorSubject({});
this.selectedPlugins = {};
this.updatingPluginId = { install: '', uninstall: '' };
this.appsDisabled = new Set();
}
updateSelectedItems(selected, plugin) {
this.selectedPlugins[plugin.id] = selected ? plugin : undefined;
const onlyInstalledPlugins = Object.values(this.selectedPlugins).filter(Boolean);
this.selectedItems.emit(onlyInstalledPlugins);
}
async installPlugin(plugin) {
await this.updateAppRemotes(plugin, 'install');
}
async uninstallPlugin(plugin) {
await this.updateAppRemotes(plugin, 'uninstall');
}
async updateAppRemotes(plugin, updateType) {
this.updatingPluginId[updateType] = plugin?.id;
let initialState;
try {
const apps = await this.getAppsForUpdate(plugin, updateType);
initialState = {
apps,
updateType,
pluginName: plugin.name,
appsDisabled: this.appsDisabled
};
}
catch (e) {
this.alertService.addServerFailure(e);
this.updatingPluginId[updateType] = '';
return;
}
let selectedApps;
try {
selectedApps = await this.selectApps(initialState);
if (!selectedApps) {
this.updatingPluginId[updateType] = '';
return;
}
}
catch {
// unreached
}
if (updateType === 'install') {
const isArchived = await this.ecosystemService.verifyArchived([plugin]);
if (!isArchived) {
this.updatingPluginId[updateType] = '';
return;
}
const licensesVerifiedByUser = await this.ecosystemService.verifyLicenses([plugin]);
if (!licensesVerifiedByUser) {
this.updatingPluginId[updateType] = '';
return;
}
}
for (const app of selectedApps) {
try {
if (updateType === 'install') {
const versionIsCompatible = await this.ecosystemService.verifyPluginVersionsCompatibility([plugin], app);
if (!versionIsCompatible) {
continue;
}
}
await this.handleRemotesUpdate(app, plugin, updateType);
const humanizedAppName = await firstValueFrom(this.humanizeAppNamePipe.transform(app));
const successText = updateType === 'install'
? this.translateService.instant(gettext('Plugin installed to application "{{ appName }}".'), {
appName: humanizedAppName
})
: this.translateService.instant(gettext('Plugin uninstalled from application "{{ appName }}".'), { appName: humanizedAppName });
this.alertService.success(successText);
this.onUpdateEventHandleGS(plugin, app, updateType);
}
catch (error) {
this.onUpdateEventHandleGS(plugin, app, updateType, error);
}
}
this.updatingPluginId[updateType] = '';
}
onUpdateEventHandleGS(plugin, app, updateType, error) {
const pluginCustomEventInfo = pick(plugin, [
'name',
'contextPath',
'module',
'version',
'type',
'id'
]);
const gsEventResult = updateType === 'install'
? PRODUCT_EXPERIENCE_ECOSYSTEM.APPLICATIONS.RESULTS.PLUGIN_INSTALLED
: PRODUCT_EXPERIENCE_ECOSYSTEM.APPLICATIONS.RESULTS.PLUGIN_REMOVED;
const eventData = {
component: PRODUCT_EXPERIENCE_ECOSYSTEM.APPLICATIONS.COMPONENTS.PLUGIN_LIST,
result: error || gsEventResult,
url: this.CURRENT_LOCATION,
...pluginCustomEventInfo,
targetApplicationName: app.name,
targetApplicationContextPath: app.contextPath,
...(error && { error })
};
this.gainsightService.triggerEvent(PRODUCT_EXPERIENCE_ECOSYSTEM.APPLICATIONS.EVENTS.PACKAGE_PLUGINS, eventData);
}
async getAppsForUpdate(plugin, updateType) {
let apps = (await this.ecosystemService.getWebApplications()).filter(app => this.ecosystemService.isOwner(app) && app.type !== ApplicationType.EXTERNAL);
if (updateType === 'install') {
this.appsDisabled.clear();
for (const app of apps) {
if (this.isPluginInstalledInApp(plugin, app)) {
this.appsDisabled.add(app.id);
}
}
}
if (updateType === 'uninstall') {
const installedApps = [];
for (const app of apps) {
if (this.isPluginInstalledInApp(plugin, app)) {
installedApps.push(app);
}
}
apps = installedApps;
}
return apps;
}
isPluginInstalledInApp(plugin, app) {
const appRemotes = this.pluginsService.getMFRemotes(app) || {};
for (const [remoteName, modules] of Object.entries(appRemotes)) {
const pluginFromThisPackageIsInstalled = this.getPluginContextPathWithoutVersion(remoteName) === plugin.contextPath;
const specificPluginModuleIsInstalled = modules.some(module => module === plugin.module);
if (pluginFromThisPackageIsInstalled && specificPluginModuleIsInstalled) {
return true;
}
}
return false;
}
getPluginContextPathWithoutVersion(remote) {
return remote.split('@')[0];
}
async handleRemotesUpdate(application, plugin, updateType) {
try {
// When remotes object is not set in the configuration object of an application.
// Fallback to setInitialRemotes is triggered.
const { remotes, excludedRemotes } = await (updateType === 'install'
? this.pluginsService.addRemotes(application, plugin)
: this.pluginsService.removeRemotes(application, this.getAllPluginsToRemove(plugin)));
if (!application.config) {
application.config = {};
}
application.config.remotes = remotes;
application.config.excludedRemotes = excludedRemotes;
const actualRemotes = this.pluginsService.getMFRemotes(application);
return this.emitRemotes(actualRemotes);
}
catch (er) {
if (er) {
this.alertService.addServerFailure(er);
}
throw er;
}
}
getAllPluginsToRemove(plugin) {
return this.package.applicationVersions.map(av => ({
id: `${plugin.contextPath}@${av.version}/${plugin.module}`,
idLatest: `${plugin.contextPath}/${plugin.module}`,
module: plugin.module,
path: plugin.path
}));
}
emitRemotes(remotes) {
this.remotePlugins$.next(remotes);
return { ...this.remotePlugins$.value };
}
async selectApps(initialState) {
try {
return await this.bsModalService.show(AppsToUpdateRemotesSelectComponent, {
class: 'modal-sm',
ariaDescribedby: 'modal-body',
ariaLabelledBy: 'modal-title',
initialState,
ignoreBackdropClick: true,
keyboard: false
}).content.result;
}
catch (er) {
return;
}
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PluginListComponent, deps: [{ token: i1.EcosystemService }, { token: i2.BsModalService }, { token: i3.PluginsService }, { token: i3.AlertService }, { token: i4.TranslateService }, { token: i3.GainsightService }, { token: i3.HumanizeAppNamePipe }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: PluginListComponent, selector: "c8y-plugin-list", inputs: { plugins$: "plugins$", emptyListText: "emptyListText", selectable: "selectable", hideSource: "hideSource", installable: "installable", package: "package" }, outputs: { selectedItems: "selectedItems" }, ngImport: i0, template: "<c8y-list-group class=\"bg-inherit\">\n <ng-container *ngIf=\"(plugins$ | async)?.length !== 0; else emptyList\">\n <ng-container *ngFor=\"let plugin of plugins$ | async\">\n <c8y-li [ngClass]=\"{ disabled: plugin.installed }\" class=\"bg-inherit\">\n <c8y-plugin-list-item\n (isItemSelected)=\"updateSelectedItems($event, plugin)\"\n [plugin]=\"plugin\"\n [selectable]=\"selectable\"\n [hideSource]=\"hideSource\"\n class=\"d-flex\"\n ></c8y-plugin-list-item>\n <div class=\"p-l-40 m-t-4\">\n <button\n *ngIf=\"installable\"\n (click)=\"uninstallPlugin(plugin)\"\n [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.uninstall }\"\n [disabled]=\"updatingPluginId.uninstall && plugin.id !== updatingPluginId.uninstall\"\n class=\"btn btn-danger btn-sm m-l-4\"\n title=\"{{ 'Uninstall plugin' | translate }}\"\n data-cy=\"plugin-list--uninstall-plugin-button\"\n translate\n >\n Uninstall plugin\n </button>\n <button\n *ngIf=\"installable\"\n (click)=\"installPlugin(plugin)\"\n [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.install }\"\n [disabled]=\"updatingPluginId.install && plugin.id !== updatingPluginId.install\"\n class=\"btn btn-default btn-sm m-l-8\"\n title=\"{{ 'Install plugin' | translate }}\"\n data-cy=\"plugin-list--install-plugin-button\"\n translate\n >\n Install plugin\n </button>\n </div>\n </c8y-li>\n </ng-container>\n </ng-container>\n</c8y-list-group>\n<ng-template #emptyList>\n <div class=\"c8y-empty-state text-left\" *ngIf=\"emptyListText\">\n <h1 c8yIcon=\"plugin\"></h1>\n <p>\n {{ emptyListText | translate }}\n </p>\n </div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i3.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i3.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { 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: i6.PluginListItemComponent, selector: "c8y-plugin-list-item", inputs: ["plugin", "selectable", "hideSource"], outputs: ["isItemSelected"] }, { kind: "pipe", type: i3.C8yTranslatePipe, name: "translate" }, { kind: "pipe", type: i5.AsyncPipe, name: "async" }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PluginListComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-plugin-list', template: "<c8y-list-group class=\"bg-inherit\">\n <ng-container *ngIf=\"(plugins$ | async)?.length !== 0; else emptyList\">\n <ng-container *ngFor=\"let plugin of plugins$ | async\">\n <c8y-li [ngClass]=\"{ disabled: plugin.installed }\" class=\"bg-inherit\">\n <c8y-plugin-list-item\n (isItemSelected)=\"updateSelectedItems($event, plugin)\"\n [plugin]=\"plugin\"\n [selectable]=\"selectable\"\n [hideSource]=\"hideSource\"\n class=\"d-flex\"\n ></c8y-plugin-list-item>\n <div class=\"p-l-40 m-t-4\">\n <button\n *ngIf=\"installable\"\n (click)=\"uninstallPlugin(plugin)\"\n [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.uninstall }\"\n [disabled]=\"updatingPluginId.uninstall && plugin.id !== updatingPluginId.uninstall\"\n class=\"btn btn-danger btn-sm m-l-4\"\n title=\"{{ 'Uninstall plugin' | translate }}\"\n data-cy=\"plugin-list--uninstall-plugin-button\"\n translate\n >\n Uninstall plugin\n </button>\n <button\n *ngIf=\"installable\"\n (click)=\"installPlugin(plugin)\"\n [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.install }\"\n [disabled]=\"updatingPluginId.install && plugin.id !== updatingPluginId.install\"\n class=\"btn btn-default btn-sm m-l-8\"\n title=\"{{ 'Install plugin' | translate }}\"\n data-cy=\"plugin-list--install-plugin-button\"\n translate\n >\n Install plugin\n </button>\n </div>\n </c8y-li>\n </ng-container>\n </ng-container>\n</c8y-list-group>\n<ng-template #emptyList>\n <div class=\"c8y-empty-state text-left\" *ngIf=\"emptyListText\">\n <h1 c8yIcon=\"plugin\"></h1>\n <p>\n {{ emptyListText | translate }}\n </p>\n </div>\n</ng-template>\n" }]
}], ctorParameters: () => [{ type: i1.EcosystemService }, { type: i2.BsModalService }, { type: i3.PluginsService }, { type: i3.AlertService }, { type: i4.TranslateService }, { type: i3.GainsightService }, { type: i3.HumanizeAppNamePipe }], propDecorators: { plugins$: [{
type: Input
}], emptyListText: [{
type: Input
}], selectable: [{
type: Input
}], hideSource: [{
type: Input
}], installable: [{
type: Input
}], package: [{
type: Input
}], selectedItems: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin-list.component.js","sourceRoot":"","sources":["../../../../ecosystem/application-plugins/plugin-list.component.ts","../../../../ecosystem/application-plugins/plugin-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { ApplicationRemotePlugins, ApplicationType, IApplication } from '@c8y/client';\nimport {\n  AlertService,\n  ApplicationPlugin,\n  GainsightService,\n  PluginsService,\n  gettext,\n  HumanizeAppNamePipe\n} from '@c8y/ngx-components';\nimport {\n  EcosystemService,\n  PRODUCT_EXPERIENCE_ECOSYSTEM\n} from '@c8y/ngx-components/ecosystem/shared';\nimport { TranslateService } from '@ngx-translate/core';\nimport { BsModalService } from 'ngx-bootstrap/modal';\nimport { BehaviorSubject, firstValueFrom, Observable } from 'rxjs';\nimport { AppsToUpdateRemotesSelectComponent } from './apps-to-update-remotes-select.component';\nimport { UpdateType } from './apps-to-update-remotes-select.model';\nimport { pick } from 'lodash-es';\n\n@Component({\n  selector: 'c8y-plugin-list',\n  templateUrl: './plugin-list.component.html'\n})\nexport class PluginListComponent {\n  CURRENT_LOCATION = location.href;\n\n  @Input() plugins$: Observable<ApplicationPlugin[]>;\n  @Input() emptyListText = '';\n  @Input() selectable: boolean;\n  @Input() hideSource = false;\n  /**\n   * Shows the install button for each plugin separately. Currently used in package-details view.\n   */\n  @Input() installable = false;\n  @Input() package: IApplication;\n  @Output() selectedItems: EventEmitter<ApplicationPlugin[]> = new EventEmitter();\n  remotePlugins$: BehaviorSubject<ApplicationRemotePlugins> = new BehaviorSubject({});\n  selectedPlugins: { [key: string]: ApplicationPlugin } = {};\n  updatingPluginId: Record<UpdateType, string> = { install: '', uninstall: '' };\n  appsDisabled: Set<IApplication['id']> = new Set<IApplication['id']>();\n\n  constructor(\n    private ecosystemService: EcosystemService,\n    private bsModalService: BsModalService,\n    private pluginsService: PluginsService,\n    private alertService: AlertService,\n    private translateService: TranslateService,\n    private gainsightService: GainsightService,\n    private humanizeAppNamePipe: HumanizeAppNamePipe\n  ) {}\n\n  updateSelectedItems(selected: boolean, plugin: ApplicationPlugin) {\n    this.selectedPlugins[plugin.id] = selected ? plugin : undefined;\n    const onlyInstalledPlugins = Object.values(this.selectedPlugins).filter(Boolean);\n    this.selectedItems.emit(onlyInstalledPlugins);\n  }\n\n  async installPlugin(plugin: ApplicationPlugin) {\n    await this.updateAppRemotes(plugin, 'install');\n  }\n\n  async uninstallPlugin(plugin: ApplicationPlugin) {\n    await this.updateAppRemotes(plugin, 'uninstall');\n  }\n\n  private async updateAppRemotes(plugin: ApplicationPlugin, updateType: UpdateType) {\n    this.updatingPluginId[updateType] = plugin?.id;\n    let initialState: Pick<\n      AppsToUpdateRemotesSelectComponent,\n      'apps' | 'updateType' | 'pluginName' | 'appsDisabled'\n    >;\n    try {\n      const apps = await this.getAppsForUpdate(plugin, updateType);\n      initialState = {\n        apps,\n        updateType,\n        pluginName: plugin.name,\n        appsDisabled: this.appsDisabled\n      };\n    } catch (e) {\n      this.alertService.addServerFailure(e);\n      this.updatingPluginId[updateType] = '';\n      return;\n    }\n\n    let selectedApps: IApplication[];\n    try {\n      selectedApps = await this.selectApps(initialState);\n      if (!selectedApps) {\n        this.updatingPluginId[updateType] = '';\n        return;\n      }\n    } catch {\n      // unreached\n    }\n\n    if (updateType === 'install') {\n      const isArchived = await this.ecosystemService.verifyArchived([plugin]);\n      if (!isArchived) {\n        this.updatingPluginId[updateType] = '';\n        return;\n      }\n\n      const licensesVerifiedByUser = await this.ecosystemService.verifyLicenses([plugin]);\n      if (!licensesVerifiedByUser) {\n        this.updatingPluginId[updateType] = '';\n        return;\n      }\n    }\n\n    for (const app of selectedApps) {\n      try {\n        if (updateType === 'install') {\n          const versionIsCompatible = await this.ecosystemService.verifyPluginVersionsCompatibility(\n            [plugin],\n            app\n          );\n\n          if (!versionIsCompatible) {\n            continue;\n          }\n        }\n\n        await this.handleRemotesUpdate(app, plugin, updateType);\n        const humanizedAppName = await firstValueFrom(this.humanizeAppNamePipe.transform(app));\n        const successText =\n          updateType === 'install'\n            ? this.translateService.instant(\n                gettext('Plugin installed to application \"{{ appName }}\".'),\n                {\n                  appName: humanizedAppName\n                }\n              )\n            : this.translateService.instant(\n                gettext('Plugin uninstalled from application \"{{ appName }}\".'),\n                { appName: humanizedAppName }\n              );\n        this.alertService.success(successText);\n        this.onUpdateEventHandleGS(plugin, app, updateType);\n      } catch (error) {\n        this.onUpdateEventHandleGS(plugin, app, updateType, error);\n      }\n    }\n    this.updatingPluginId[updateType] = '';\n  }\n\n  private onUpdateEventHandleGS(\n    plugin: ApplicationPlugin,\n    app: IApplication,\n    updateType: UpdateType,\n    error?: unknown\n  ) {\n    const pluginCustomEventInfo = pick(plugin, [\n      'name',\n      'contextPath',\n      'module',\n      'version',\n      'type',\n      'id'\n    ] satisfies (keyof ApplicationPlugin)[]);\n\n    const gsEventResult =\n      updateType === 'install'\n        ? PRODUCT_EXPERIENCE_ECOSYSTEM.APPLICATIONS.RESULTS.PLUGIN_INSTALLED\n        : PRODUCT_EXPERIENCE_ECOSYSTEM.APPLICATIONS.RESULTS.PLUGIN_REMOVED;\n\n    const eventData = {\n      component: PRODUCT_EXPERIENCE_ECOSYSTEM.APPLICATIONS.COMPONENTS.PLUGIN_LIST,\n      result: error || gsEventResult,\n      url: this.CURRENT_LOCATION,\n      ...pluginCustomEventInfo,\n      targetApplicationName: app.name,\n      targetApplicationContextPath: app.contextPath,\n      ...(error && { error })\n    };\n\n    this.gainsightService.triggerEvent(\n      PRODUCT_EXPERIENCE_ECOSYSTEM.APPLICATIONS.EVENTS.PACKAGE_PLUGINS,\n      eventData\n    );\n  }\n\n  private async getAppsForUpdate(plugin: ApplicationPlugin, updateType: UpdateType) {\n    let apps = (await this.ecosystemService.getWebApplications()).filter(\n      app => this.ecosystemService.isOwner(app) && app.type !== ApplicationType.EXTERNAL\n    );\n\n    if (updateType === 'install') {\n      this.appsDisabled.clear();\n      for (const app of apps) {\n        if (this.isPluginInstalledInApp(plugin, app)) {\n          this.appsDisabled.add(app.id);\n        }\n      }\n    }\n\n    if (updateType === 'uninstall') {\n      const installedApps: IApplication[] = [];\n      for (const app of apps) {\n        if (this.isPluginInstalledInApp(plugin, app)) {\n          installedApps.push(app);\n        }\n      }\n      apps = installedApps;\n    }\n    return apps;\n  }\n\n  private isPluginInstalledInApp(plugin: ApplicationPlugin, app: IApplication): boolean {\n    const appRemotes = this.pluginsService.getMFRemotes(app) || {};\n\n    for (const [remoteName, modules] of Object.entries(appRemotes)) {\n      const pluginFromThisPackageIsInstalled =\n        this.getPluginContextPathWithoutVersion(remoteName) === plugin.contextPath;\n      const specificPluginModuleIsInstalled = modules.some(module => module === plugin.module);\n      if (pluginFromThisPackageIsInstalled && specificPluginModuleIsInstalled) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  private getPluginContextPathWithoutVersion(remote: string) {\n    return remote.split('@')[0];\n  }\n\n  private async handleRemotesUpdate(\n    application: IApplication,\n    plugin: ApplicationPlugin,\n    updateType: UpdateType\n  ) {\n    try {\n      // When remotes object is not set in the configuration object of an application.\n      // Fallback to setInitialRemotes is triggered.\n      const { remotes, excludedRemotes } = await (updateType === 'install'\n        ? this.pluginsService.addRemotes(application, plugin)\n        : this.pluginsService.removeRemotes(application, this.getAllPluginsToRemove(plugin)));\n      if (!application.config) {\n        application.config = {};\n      }\n      application.config.remotes = remotes;\n      application.config.excludedRemotes = excludedRemotes;\n      const actualRemotes = this.pluginsService.getMFRemotes(application);\n      return this.emitRemotes(actualRemotes);\n    } catch (er) {\n      if (er) {\n        this.alertService.addServerFailure(er);\n      }\n      throw er;\n    }\n  }\n\n  private getAllPluginsToRemove(plugin: ApplicationPlugin): ApplicationPlugin[] {\n    return this.package.applicationVersions.map(av => ({\n      id: `${plugin.contextPath}@${av.version}/${plugin.module}`,\n      idLatest: `${plugin.contextPath}/${plugin.module}`,\n      module: plugin.module,\n      path: plugin.path\n    }));\n  }\n\n  private emitRemotes(remotes: ApplicationRemotePlugins): ApplicationRemotePlugins {\n    this.remotePlugins$.next(remotes);\n    return { ...this.remotePlugins$.value };\n  }\n\n  private async selectApps(\n    initialState: Pick<\n      AppsToUpdateRemotesSelectComponent,\n      'apps' | 'updateType' | 'pluginName' | 'appsDisabled'\n    >\n  ): Promise<IApplication[]> {\n    try {\n      return await (\n        this.bsModalService.show(AppsToUpdateRemotesSelectComponent, {\n          class: 'modal-sm',\n          ariaDescribedby: 'modal-body',\n          ariaLabelledBy: 'modal-title',\n          initialState,\n          ignoreBackdropClick: true,\n          keyboard: false\n        }).content as AppsToUpdateRemotesSelectComponent\n      ).result;\n    } catch (er) {\n      return;\n    }\n  }\n}\n","<c8y-list-group class=\"bg-inherit\">\n  <ng-container *ngIf=\"(plugins$ | async)?.length !== 0; else emptyList\">\n    <ng-container *ngFor=\"let plugin of plugins$ | async\">\n      <c8y-li [ngClass]=\"{ disabled: plugin.installed }\" class=\"bg-inherit\">\n        <c8y-plugin-list-item\n          (isItemSelected)=\"updateSelectedItems($event, plugin)\"\n          [plugin]=\"plugin\"\n          [selectable]=\"selectable\"\n          [hideSource]=\"hideSource\"\n          class=\"d-flex\"\n        ></c8y-plugin-list-item>\n        <div class=\"p-l-40 m-t-4\">\n          <button\n            *ngIf=\"installable\"\n            (click)=\"uninstallPlugin(plugin)\"\n            [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.uninstall }\"\n            [disabled]=\"updatingPluginId.uninstall && plugin.id !== updatingPluginId.uninstall\"\n            class=\"btn btn-danger btn-sm m-l-4\"\n            title=\"{{ 'Uninstall plugin' | translate }}\"\n            data-cy=\"plugin-list--uninstall-plugin-button\"\n            translate\n          >\n            Uninstall plugin\n          </button>\n          <button\n            *ngIf=\"installable\"\n            (click)=\"installPlugin(plugin)\"\n            [ngClass]=\"{ 'btn-pending': plugin.id === updatingPluginId.install }\"\n            [disabled]=\"updatingPluginId.install && plugin.id !== updatingPluginId.install\"\n            class=\"btn btn-default btn-sm m-l-8\"\n            title=\"{{ 'Install plugin' | translate }}\"\n            data-cy=\"plugin-list--install-plugin-button\"\n            translate\n          >\n            Install plugin\n          </button>\n        </div>\n      </c8y-li>\n    </ng-container>\n  </ng-container>\n</c8y-list-group>\n<ng-template #emptyList>\n  <div class=\"c8y-empty-state text-left\" *ngIf=\"emptyListText\">\n    <h1 c8yIcon=\"plugin\"></h1>\n    <p>\n      {{ emptyListText | translate }}\n    </p>\n  </div>\n</ng-template>\n"]}