@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
112 lines • 25 kB
JavaScript
import { Component, forwardRef, Input } from '@angular/core';
import { NG_VALUE_ACCESSOR } from '@angular/forms';
import { ApplicationService } from '@c8y/client';
import { gettext, PluginsService } from '@c8y/ngx-components';
import { BehaviorSubject, combineLatest, merge } from 'rxjs';
import { filter, map, shareReplay, switchMap } from 'rxjs/operators';
import * as i0 from "@angular/core";
import * as i1 from "@c8y/client";
import * as i2 from "@c8y/ngx-components";
import * as i3 from "@angular/common";
import * as i4 from "@angular/forms";
export class PackageVersionSelectComponent {
constructor(applicationService, pluginsService) {
this.applicationService = applicationService;
this.pluginsService = pluginsService;
this.label = gettext('Use plugin version');
this.onInput$ = new BehaviorSubject('');
this.isDisabled = false;
this.packageContextPath$ = new BehaviorSubject('');
this.packageId$ = new BehaviorSubject('');
const packageIdFromContextPath$ = this.packageContextPath$.pipe(filter(path => !!path), switchMap(path => this.getPackageIdForContextPath(path)));
const packageId$ = merge(packageIdFromContextPath$, this.packageId$.pipe(filter(id => !!id)));
const packageVersions$ = packageId$.pipe(switchMap(id => this.getPackageVersions(id)), shareReplay(1));
this.versions$ = combineLatest([packageVersions$, this.onInput$.asObservable()]).pipe(map(([resultList, filterStr]) => this.applyFilterToResultList(resultList, filterStr)));
}
writeValue(obj) {
this.selectedVersion = obj;
}
registerOnChange(fn) {
this.onChange = fn;
}
registerOnTouched(fn) {
this.onTouched = fn;
}
setDisabledState(isDisabled) {
this.isDisabled = isDisabled;
}
ngOnChanges(changes) {
if (changes.packageContextPath) {
this.packageContextPath$.next(this.packageContextPath);
}
if (changes.packageId) {
this.packageId$.next(this.packageId);
}
}
async getPackageVersions(packageId) {
return await this.applicationService.listVersions(packageId);
}
onVersionSelect(version) {
this.selectedVersion = version;
if (this.onChange) {
this.onChange(version);
}
if (this.onTouched) {
this.onTouched();
}
}
async getPackageIdForContextPath(contextPath) {
const { data: application } = await this.applicationService.getManifestOfContextPath(contextPath);
const packageAppId = application.id;
return packageAppId;
}
setInitialValueForInput(versions) {
if (!this.selectedVersion && versions.length > 0) {
const latest = versions.find(v => v.tags.includes('latest'));
this.selectedVersion = latest || versions[0];
if (this.onChange) {
this.onChange(this.selectedVersion);
}
}
}
filterAppVersions(appVersions, filterStr) {
return filterStr === ''
? appVersions
: appVersions.filter(appVersion => appVersion.version.includes(filterStr) ||
appVersion.tags?.some(tag => tag.includes(filterStr)));
}
applyFilterToResultList(resultList, filterStr) {
const versionsFilteredByStr = this.filterAppVersions(resultList.data, filterStr);
const sortedAppVersions = this.pluginsService.sortVersions({
list: versionsFilteredByStr,
path: ['version']
}, 'desc');
this.setInitialValueForInput(sortedAppVersions);
return { data: sortedAppVersions, res: resultList.res };
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PackageVersionSelectComponent, deps: [{ token: i1.ApplicationService }, { token: i2.PluginsService }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: PackageVersionSelectComponent, selector: "c8y-package-version-select", inputs: { label: "label", packageContextPath: "packageContextPath", packageId: "packageId" }, providers: [
{
provide: NG_VALUE_ACCESSOR,
multi: true,
useExisting: forwardRef(() => PackageVersionSelectComponent)
}
], usesOnChanges: true, ngImport: i0, template: "<label for=\"packageVersion\">{{ label | translate }}</label>\n<c8y-form-group>\n <c8y-typeahead\n [(ngModel)]=\"selectedVersion\"\n name=\"packageVersion\"\n (onSearch)=\"onInput$.next($event)\"\n placeholder=\"{{ 'Select below or start typing' | translate }}\"\n [displayProperty]=\"'version'\"\n [required]=\"true\"\n [disabled]=\"isDisabled\"\n [hideNew]=\"true\"\n [container]=\"'body'\"\n >\n <c8y-li\n *c8yFor=\"let appVersion of versions$; loadMore: 'auto'; notFound: notFoundTemplate\"\n (click)=\"onVersionSelect(appVersion)\"\n class=\"p-l-8 p-r-8 c8y-list__item--link\"\n [active]=\"selectedVersion === appVersion\"\n >\n <c8y-li-icon icon=\"big-parcel\"></c8y-li-icon>\n <span\n [ngStyle]=\"{\n display: 'flex',\n 'flex-direction': 'row',\n 'align-content': 'center',\n 'justify-content': 'space-between',\n 'align-items': 'center'\n }\"\n >\n <c8y-highlight\n [text]=\"appVersion.version || '--'\"\n [pattern]=\"onInput$ | async\"\n ></c8y-highlight>\n <span>\n <span *ngFor=\"let tag of appVersion.tags\" class=\"label label-info m-l-4\">\n {{ tag }}\n </span>\n </span>\n </span>\n </c8y-li>\n <ng-template #notFoundTemplate>\n <c8y-li\n class=\"bg-gray-lighter p-8\"\n *ngIf=\"(onInput$ | async)?.length > 0 && (versions$ | async)?.data?.length === 0\"\n >\n <span translate>No match found.</span>\n </c8y-li>\n </ng-template>\n </c8y-typeahead>\n</c8y-form-group>\n", dependencies: [{ kind: "directive", type: i2.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i2.ForOfDirective, selector: "[c8yFor]", inputs: ["c8yForOf", "c8yForLoadMore", "c8yForPipe", "c8yForNotFound", "c8yForMaxIterations", "c8yForLoadingTemplate", "c8yForLoadNextLabel", "c8yForLoadingLabel", "c8yForRealtime", "c8yForRealtimeOptions", "c8yForComparator", "c8yForEnableVirtualScroll", "c8yForVirtualScrollElementSize", "c8yForVirtualScrollStrategy", "c8yForVirtualScrollContainerHeight"], outputs: ["c8yForCount", "c8yForChange", "c8yForLoadMoreComponent"] }, { kind: "component", type: i2.HighlightComponent, selector: "c8y-highlight", inputs: ["pattern", "text", "elementClass", "shouldTrimPattern"] }, { kind: "component", type: i2.TypeaheadComponent, selector: "c8y-typeahead", inputs: ["required", "maxlength", "disabled", "allowFreeEntries", "placeholder", "displayProperty", "icon", "name", "autoClose", "hideNew", "container", "selected", "highlightFirstItem"], outputs: ["onSearch", "onIconClick"] }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i2.FormGroupComponent, selector: "c8y-form-group", inputs: ["hasError", "hasWarning", "hasSuccess", "novalidation", "status"] }, { kind: "component", type: i2.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i2.ListItemIconComponent, selector: "c8y-list-item-icon, c8y-li-icon", inputs: ["icon", "status"] }, { kind: "pipe", type: i2.C8yTranslatePipe, name: "translate" }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PackageVersionSelectComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-package-version-select', providers: [
{
provide: NG_VALUE_ACCESSOR,
multi: true,
useExisting: forwardRef(() => PackageVersionSelectComponent)
}
], template: "<label for=\"packageVersion\">{{ label | translate }}</label>\n<c8y-form-group>\n <c8y-typeahead\n [(ngModel)]=\"selectedVersion\"\n name=\"packageVersion\"\n (onSearch)=\"onInput$.next($event)\"\n placeholder=\"{{ 'Select below or start typing' | translate }}\"\n [displayProperty]=\"'version'\"\n [required]=\"true\"\n [disabled]=\"isDisabled\"\n [hideNew]=\"true\"\n [container]=\"'body'\"\n >\n <c8y-li\n *c8yFor=\"let appVersion of versions$; loadMore: 'auto'; notFound: notFoundTemplate\"\n (click)=\"onVersionSelect(appVersion)\"\n class=\"p-l-8 p-r-8 c8y-list__item--link\"\n [active]=\"selectedVersion === appVersion\"\n >\n <c8y-li-icon icon=\"big-parcel\"></c8y-li-icon>\n <span\n [ngStyle]=\"{\n display: 'flex',\n 'flex-direction': 'row',\n 'align-content': 'center',\n 'justify-content': 'space-between',\n 'align-items': 'center'\n }\"\n >\n <c8y-highlight\n [text]=\"appVersion.version || '--'\"\n [pattern]=\"onInput$ | async\"\n ></c8y-highlight>\n <span>\n <span *ngFor=\"let tag of appVersion.tags\" class=\"label label-info m-l-4\">\n {{ tag }}\n </span>\n </span>\n </span>\n </c8y-li>\n <ng-template #notFoundTemplate>\n <c8y-li\n class=\"bg-gray-lighter p-8\"\n *ngIf=\"(onInput$ | async)?.length > 0 && (versions$ | async)?.data?.length === 0\"\n >\n <span translate>No match found.</span>\n </c8y-li>\n </ng-template>\n </c8y-typeahead>\n</c8y-form-group>\n" }]
}], ctorParameters: () => [{ type: i1.ApplicationService }, { type: i2.PluginsService }], propDecorators: { label: [{
type: Input
}], packageContextPath: [{
type: Input
}], packageId: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"package-version-select.component.js","sourceRoot":"","sources":["../../../../../ecosystem/shared/package-version-select/package-version-select.component.ts","../../../../../ecosystem/shared/package-version-select/package-version-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, forwardRef, OnChanges, Input, SimpleChanges } from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { ApplicationService, IApplicationVersion, IResultList } from '@c8y/client';\nimport { gettext, PluginsService } from '@c8y/ngx-components';\nimport { BehaviorSubject, Observable, combineLatest, merge } from 'rxjs';\nimport { filter, map, shareReplay, switchMap } from 'rxjs/operators';\n\n@Component({\n  selector: 'c8y-package-version-select',\n  templateUrl: './package-version-select.component.html',\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      multi: true,\n      useExisting: forwardRef(() => PackageVersionSelectComponent)\n    }\n  ]\n})\nexport class PackageVersionSelectComponent implements OnChanges, ControlValueAccessor {\n  @Input() label = gettext('Use plugin version');\n  @Input() packageContextPath: string;\n  @Input() packageId: string | number;\n  selectedVersion: IApplicationVersion;\n  onInput$: BehaviorSubject<string> = new BehaviorSubject<string>('');\n  versions$: Observable<IResultList<IApplicationVersion>>;\n  isDisabled = false;\n  private packageContextPath$ = new BehaviorSubject('');\n  private packageId$ = new BehaviorSubject<string | number>('');\n  private onChange: (version: IApplicationVersion) => void;\n  private onTouched: () => void;\n\n  constructor(\n    private applicationService: ApplicationService,\n    private pluginsService: PluginsService\n  ) {\n    const packageIdFromContextPath$ = this.packageContextPath$.pipe(\n      filter(path => !!path),\n      switchMap(path => this.getPackageIdForContextPath(path))\n    );\n    const packageId$ = merge(packageIdFromContextPath$, this.packageId$.pipe(filter(id => !!id)));\n    const packageVersions$ = packageId$.pipe(\n      switchMap(id => this.getPackageVersions(id)),\n      shareReplay(1)\n    );\n    this.versions$ = combineLatest([packageVersions$, this.onInput$.asObservable()]).pipe(\n      map(([resultList, filterStr]) => this.applyFilterToResultList(resultList, filterStr))\n    );\n  }\n\n  writeValue(obj: any): void {\n    this.selectedVersion = obj;\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState?(isDisabled: boolean): void {\n    this.isDisabled = isDisabled;\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.packageContextPath) {\n      this.packageContextPath$.next(this.packageContextPath);\n    }\n\n    if (changes.packageId) {\n      this.packageId$.next(this.packageId);\n    }\n  }\n\n  async getPackageVersions(packageId: string | number): Promise<IResultList<IApplicationVersion>> {\n    return await this.applicationService.listVersions(packageId);\n  }\n\n  onVersionSelect(version: IApplicationVersion) {\n    this.selectedVersion = version;\n    if (this.onChange) {\n      this.onChange(version);\n    }\n    if (this.onTouched) {\n      this.onTouched();\n    }\n  }\n\n  private async getPackageIdForContextPath(contextPath: string): Promise<string | number> {\n    const { data: application } = await this.applicationService.getManifestOfContextPath(\n      contextPath\n    );\n    const packageAppId = application.id;\n    return packageAppId;\n  }\n\n  private setInitialValueForInput(versions: IApplicationVersion[]) {\n    if (!this.selectedVersion && versions.length > 0) {\n      const latest = versions.find(v => v.tags.includes('latest'));\n      this.selectedVersion = latest || versions[0];\n      if (this.onChange) {\n        this.onChange(this.selectedVersion);\n      }\n    }\n  }\n\n  private filterAppVersions(\n    appVersions: IApplicationVersion[],\n    filterStr: string\n  ): IApplicationVersion[] {\n    return filterStr === ''\n      ? appVersions\n      : appVersions.filter(\n          appVersion =>\n            appVersion.version.includes(filterStr) ||\n            appVersion.tags?.some(tag => tag.includes(filterStr))\n        );\n  }\n\n  private applyFilterToResultList(\n    resultList: IResultList<IApplicationVersion>,\n    filterStr: string\n  ): IResultList<IApplicationVersion> {\n    const versionsFilteredByStr = this.filterAppVersions(resultList.data, filterStr);\n    const sortedAppVersions = this.pluginsService.sortVersions(\n      {\n        list: versionsFilteredByStr,\n        path: ['version']\n      },\n      'desc'\n    );\n\n    this.setInitialValueForInput(sortedAppVersions);\n    return { data: sortedAppVersions, res: resultList.res };\n  }\n}\n","<label for=\"packageVersion\">{{ label | translate }}</label>\n<c8y-form-group>\n  <c8y-typeahead\n    [(ngModel)]=\"selectedVersion\"\n    name=\"packageVersion\"\n    (onSearch)=\"onInput$.next($event)\"\n    placeholder=\"{{ 'Select below or start typing' | translate }}\"\n    [displayProperty]=\"'version'\"\n    [required]=\"true\"\n    [disabled]=\"isDisabled\"\n    [hideNew]=\"true\"\n    [container]=\"'body'\"\n  >\n    <c8y-li\n      *c8yFor=\"let appVersion of versions$; loadMore: 'auto'; notFound: notFoundTemplate\"\n      (click)=\"onVersionSelect(appVersion)\"\n      class=\"p-l-8 p-r-8 c8y-list__item--link\"\n      [active]=\"selectedVersion === appVersion\"\n    >\n      <c8y-li-icon icon=\"big-parcel\"></c8y-li-icon>\n      <span\n        [ngStyle]=\"{\n          display: 'flex',\n          'flex-direction': 'row',\n          'align-content': 'center',\n          'justify-content': 'space-between',\n          'align-items': 'center'\n        }\"\n      >\n        <c8y-highlight\n          [text]=\"appVersion.version || '--'\"\n          [pattern]=\"onInput$ | async\"\n        ></c8y-highlight>\n        <span>\n          <span *ngFor=\"let tag of appVersion.tags\" class=\"label label-info m-l-4\">\n            {{ tag }}\n          </span>\n        </span>\n      </span>\n    </c8y-li>\n    <ng-template #notFoundTemplate>\n      <c8y-li\n        class=\"bg-gray-lighter p-8\"\n        *ngIf=\"(onInput$ | async)?.length > 0 && (versions$ | async)?.data?.length === 0\"\n      >\n        <span translate>No match found.</span>\n      </c8y-li>\n    </ng-template>\n  </c8y-typeahead>\n</c8y-form-group>\n"]}