UNPKG

@alauda-fe/common

Version:

Alauda frontend team common codes.

116 lines 15.8 kB
import { __decorate, __metadata } from "tslib"; import { ComponentPortal, DomPortalOutlet } from '@angular/cdk/portal'; import { ApplicationRef, ChangeDetectionStrategy, ChangeDetectorRef, Component, ComponentFactoryResolver, Directive, Injector, Input, Optional, Self, } from '@angular/core'; import { FormGroupDirective, NgControl, NgForm } from '@angular/forms'; import { Observable, Subject, merge, of, debounceTime, map, switchMap, takeUntil, } from 'rxjs'; import { ObservableInput } from '../../core/public-api'; import { BaseErrorsMapper } from './base-errors-mapper'; import * as i0 from "@angular/core"; import * as i1 from "@angular/forms"; export class ErrorsMapperDirective extends BaseErrorsMapper { constructor(injector, cdr, // eslint-disable-next-line sonar/deprecation cfr, appRef, control, ngForm, formGroup) { super(injector); this.injector = injector; this.cdr = cdr; this.cfr = cfr; this.appRef = appRef; this.control = control; this.ngForm = ngForm; this.formGroup = formGroup; this.destroy$ = new Subject(); } ngAfterViewInit() { if (!this.control) { throw new Error('Errors mapper is not properly configured. No valid control is found.'); } this.initPortal(); const statusChange = [this.control.statusChanges]; if (this.ngForm) { statusChange.push(this.ngForm.ngSubmit); } if (this.formGroup) { statusChange.push(this.formGroup.ngSubmit); } this.aclErrorsMapperDisabled$ .pipe(switchMap(isDisable => isDisable ? of(null) : merge(...statusChange, this.translate.locale$).pipe(debounceTime(50), map(() => this.getErrorMessage(this.control.errors, this.aclErrorsMapper || {}, this.aclErrorsMapperFn, this.aclErrorsMapperControlName)))), takeUntil(this.destroy$)) .subscribe(content => { this.updateContent(content); this.cdr.markForCheck(); }); } updateContent(content) { if (!content) { if (this.portalHost.hasAttached()) { this.portalHost.detach(); } return; } if (!this.portalHost.hasAttached()) { this.portalAttach = this.portalHost.attach(this.portal).instance; } this.portalAttach.data = content; this.portalAttach.cdr.markForCheck(); } initPortal() { this.portalHost = new DomPortalOutlet(this.aclErrorsMapperOutlet, this.cfr, this.appRef, this.injector); this.portal = new ComponentPortal(TextComponent); } ngOnDestroy() { this.destroy$.next(); this.portalHost.dispose(); } static { this.ɵfac = function ErrorsMapperDirective_Factory(t) { return new (t || ErrorsMapperDirective)(i0.ɵɵdirectiveInject(i0.Injector), i0.ɵɵdirectiveInject(i0.ChangeDetectorRef), i0.ɵɵdirectiveInject(i0.ComponentFactoryResolver), i0.ɵɵdirectiveInject(i0.ApplicationRef), i0.ɵɵdirectiveInject(i1.NgControl, 2), i0.ɵɵdirectiveInject(i1.NgForm, 8), i0.ɵɵdirectiveInject(i1.FormGroupDirective, 8)); }; } static { this.ɵdir = /*@__PURE__*/ i0.ɵɵdefineDirective({ type: ErrorsMapperDirective, selectors: [["", "aclErrorsMapper", ""]], inputs: { aclErrorsMapper: "aclErrorsMapper", aclErrorsMapperFn: "aclErrorsMapperFn", aclErrorsMapperDisabled: "aclErrorsMapperDisabled", aclErrorsMapperOutlet: "aclErrorsMapperOutlet", aclErrorsMapperControlName: "aclErrorsMapperControlName" }, standalone: true, features: [i0.ɵɵInheritDefinitionFeature] }); } } __decorate([ ObservableInput(), __metadata("design:type", Observable) ], ErrorsMapperDirective.prototype, "aclErrorsMapperDisabled$", void 0); (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(ErrorsMapperDirective, [{ type: Directive, args: [{ selector: '[aclErrorsMapper]', standalone: true, }] }], () => [{ type: i0.Injector }, { type: i0.ChangeDetectorRef }, { type: i0.ComponentFactoryResolver }, { type: i0.ApplicationRef }, { type: i1.NgControl, decorators: [{ type: Self }] }, { type: i1.NgForm, decorators: [{ type: Optional }] }, { type: i1.FormGroupDirective, decorators: [{ type: Optional }] }], { aclErrorsMapper: [{ type: Input }], aclErrorsMapperFn: [{ type: Input }], aclErrorsMapperDisabled: [{ type: Input }], aclErrorsMapperDisabled$: [], aclErrorsMapperOutlet: [{ type: Input }], aclErrorsMapperControlName: [{ type: Input }] }); })(); class TextComponent { constructor(cdr) { this.cdr = cdr; } static { this.ɵfac = function TextComponent_Factory(t) { return new (t || TextComponent)(i0.ɵɵdirectiveInject(i0.ChangeDetectorRef)); }; } static { this.ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: TextComponent, selectors: [["ng-component"]], standalone: true, features: [i0.ɵɵStandaloneFeature], decls: 1, vars: 1, template: function TextComponent_Template(rf, ctx) { if (rf & 1) { i0.ɵɵtext(0); } if (rf & 2) { i0.ɵɵtextInterpolate1(" ", ctx.data, " "); } }, encapsulation: 2, changeDetection: 0 }); } } (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(TextComponent, [{ type: Component, args: [{ template: ` {{ data }} `, changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, }] }], () => [{ type: i0.ChangeDetectorRef }], null); })(); (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassDebugInfo(TextComponent, { className: "TextComponent" }); })(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"errors-mapper.directive.js","sourceRoot":"","sources":["../../../../../../libs/common/src/form/errors-mapper/errors-mapper.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComponentPortal, DomPortalOutlet, Portal } from '@angular/cdk/portal';\nimport {\n  AfterViewInit,\n  ApplicationRef,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ComponentFactoryResolver,\n  Directive,\n  Injector,\n  Input,\n  OnDestroy,\n  Optional,\n  Self,\n} from '@angular/core';\nimport { FormGroupDirective, NgControl, NgForm } from '@angular/forms';\nimport {\n  Observable,\n  Subject,\n  merge,\n  of,\n  debounceTime,\n  map,\n  switchMap,\n  takeUntil,\n} from 'rxjs';\n\nimport { StringMap, ObservableInput } from '../../core/public-api';\n\nimport { BaseErrorsMapper } from './base-errors-mapper';\nimport { ErrorsMapperFn } from './types';\n\n@Directive({\n  selector: '[aclErrorsMapper]',\n  standalone: true,\n})\nexport class ErrorsMapperDirective\n  extends BaseErrorsMapper\n  implements AfterViewInit, OnDestroy\n{\n  @Input()\n  aclErrorsMapper: StringMap | '';\n\n  @Input()\n  aclErrorsMapperFn: ErrorsMapperFn;\n\n  @Input()\n  aclErrorsMapperDisabled: boolean;\n\n  /**\n   * disable errors mapper\n   */\n  @ObservableInput()\n  aclErrorsMapperDisabled$: Observable<boolean>;\n\n  /**\n   * specify outlet, consider this other than specific control entity to make sure di can always get ngForm\n   */\n  @Input()\n  aclErrorsMapperOutlet: Element;\n\n  @Input()\n  aclErrorsMapperControlName: string;\n\n  private portal: Portal<any>;\n  private portalHost: DomPortalOutlet;\n  private portalAttach: TextComponent;\n  private readonly destroy$ = new Subject<void>();\n\n  constructor(\n    public injector: Injector,\n    private readonly cdr: ChangeDetectorRef,\n    // eslint-disable-next-line sonar/deprecation\n    private readonly cfr: ComponentFactoryResolver,\n    private readonly appRef: ApplicationRef,\n    @Self() private readonly control: NgControl,\n    @Optional() private readonly ngForm: NgForm,\n    @Optional()\n    private readonly formGroup: FormGroupDirective,\n  ) {\n    super(injector);\n  }\n\n  ngAfterViewInit() {\n    if (!this.control) {\n      throw new Error(\n        'Errors mapper is not properly configured. No valid control is found.',\n      );\n    }\n    this.initPortal();\n    const statusChange = [this.control.statusChanges];\n\n    if (this.ngForm) {\n      statusChange.push(this.ngForm.ngSubmit);\n    }\n    if (this.formGroup) {\n      statusChange.push(this.formGroup.ngSubmit);\n    }\n\n    this.aclErrorsMapperDisabled$\n      .pipe(\n        switchMap(isDisable =>\n          isDisable\n            ? of(null)\n            : merge(...statusChange, this.translate.locale$).pipe(\n                debounceTime(50),\n                map(() =>\n                  this.getErrorMessage(\n                    this.control.errors,\n                    this.aclErrorsMapper || {},\n                    this.aclErrorsMapperFn,\n                    this.aclErrorsMapperControlName,\n                  ),\n                ),\n              ),\n        ),\n        takeUntil(this.destroy$),\n      )\n      .subscribe(content => {\n        this.updateContent(content);\n        this.cdr.markForCheck();\n      });\n  }\n\n  updateContent(content: string) {\n    if (!content) {\n      if (this.portalHost.hasAttached()) {\n        this.portalHost.detach();\n      }\n      return;\n    }\n    if (!this.portalHost.hasAttached()) {\n      this.portalAttach = this.portalHost.attach(this.portal).instance;\n    }\n    this.portalAttach.data = content;\n    this.portalAttach.cdr.markForCheck();\n  }\n\n  initPortal() {\n    this.portalHost = new DomPortalOutlet(\n      this.aclErrorsMapperOutlet,\n      this.cfr,\n      this.appRef,\n      this.injector,\n    );\n    this.portal = new ComponentPortal(TextComponent);\n  }\n\n  ngOnDestroy() {\n    this.destroy$.next();\n    this.portalHost.dispose();\n  }\n}\n\n@Component({\n  template: ` {{ data }} `,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n})\nclass TextComponent {\n  data: string;\n  constructor(public cdr: ChangeDetectorRef) {}\n}\n"]}