UNPKG

@alauda-fe/common

Version:

Alauda frontend team common codes.

158 lines 20.4 kB
import { __decorate, __metadata } from "tslib"; /** * @packageDocumentation * @module utils */ import { FormItemControlDirective } from '@alauda/ui'; import { CdkPortalOutlet, ComponentPortal, TemplatePortal, } from '@angular/cdk/portal'; import { ChangeDetectionStrategy, Component, ComponentFactoryResolver, Directive, HostBinding, Input, Optional, Self, TemplateRef, ViewContainerRef, } from '@angular/core'; import { NgControl, } from '@angular/forms'; import { Observable, Subject, combineLatest, distinctUntilChanged, filter, map, startWith, takeUntil, } from 'rxjs'; import { ObservableInput } from '../utils/decorators'; import * as i0 from "@angular/core"; import * as i1 from "@angular/forms"; import * as i2 from "@alauda/ui"; import * as i3 from "../pipes/field-not-available.pipe"; export class ReadonlyFieldDirective { get isHidden() { return this.hidden || this.aclReadonlyField; } constructor(control, // eslint-disable-next-line sonar/deprecation cfr, viewContainerRef, // 仅考虑当前control,子control无法知晓父FormItemControl元素 ,包裹了多少个control,以及各自处于什么状态,该情况自行处理 controlDirective) { this.control = control; this.cfr = cfr; this.viewContainerRef = viewContainerRef; this.controlDirective = controlDirective; this.isControlRequired = false; this.templateContext = {}; this.destroy$ = new Subject(); } setValue(value) { if (this.template) { Object.assign(this.templateContext, { $implicit: value, ...this.context, }); } else { this.spanComponentRef.instance.setData(value); } } ngOnInit() { const controlEntity = this.control.control; // formControl will compose multi validators to 1, no matter defined in template nor FormBuilder const { validator, asyncValidator } = controlEntity; this.isControlRequired = this.controlDirective?.required || false; const portalOutlet = new CdkPortalOutlet(this.cfr, this.viewContainerRef); const portal = this.template ? new TemplatePortal(this.template, this.viewContainerRef, this.templateContext) : new ComponentPortal(SpanComponent, this.viewContainerRef); const containerAttached$ = this.aclReadonlyField$.pipe(distinctUntilChanged(), map(isUpdate => { if (!isUpdate) { // 重置 if (this.controlDirective) { this.controlDirective.required = this.isControlRequired; } if (portalOutlet.hasAttached()) { portalOutlet.detach(); } this.applyValidators(controlEntity, validator, asyncValidator); return false; } if (this.controlDirective) { this.controlDirective.required = false; } const attachPoint = portalOutlet.attach(portal); if (!this.template) { this.spanComponentRef = attachPoint; } this.clearValidators(controlEntity); return true; })); const sourceValue$ = this.control.valueChanges.pipe(startWith(this.control.value)); combineLatest([sourceValue$, containerAttached$]) .pipe(takeUntil(this.destroy$), filter(([, attached]) => !!attached)) .subscribe(([value]) => { this.setValue(value); }); } applyValidators(control, rawValidator, rawAsyncValidator) { if (control.validator === rawValidator && control.asyncValidator === rawAsyncValidator) { return; } control.setValidators(rawValidator); control.setAsyncValidators(rawAsyncValidator); control.updateValueAndValidity(); } clearValidators(control) { // delay checking validators for ngModel requestAnimationFrame(() => { if (!control.validator && !control.asyncValidator) { return; } control.clearValidators(); control.clearAsyncValidators(); control.updateValueAndValidity(); }); } ngOnDestroy() { this.destroy$.next(); } static { this.ɵfac = function ReadonlyFieldDirective_Factory(t) { return new (t || ReadonlyFieldDirective)(i0.ɵɵdirectiveInject(i1.NgControl), i0.ɵɵdirectiveInject(i0.ComponentFactoryResolver), i0.ɵɵdirectiveInject(i0.ViewContainerRef), i0.ɵɵdirectiveInject(i2.FormItemControlDirective, 10)); }; } static { this.ɵdir = /*@__PURE__*/ i0.ɵɵdefineDirective({ type: ReadonlyFieldDirective, selectors: [["", "aclReadonlyField", ""]], hostVars: 1, hostBindings: function ReadonlyFieldDirective_HostBindings(rf, ctx) { if (rf & 2) { i0.ɵɵhostProperty("hidden", ctx.isHidden); } }, inputs: { aclReadonlyField: "aclReadonlyField", hidden: "hidden", template: [0, "aclReadonlyFieldTemplate", "template"], context: [0, "aclReadonlyFieldTemplateContext", "context"] }, standalone: true }); } } __decorate([ ObservableInput(), __metadata("design:type", Observable) ], ReadonlyFieldDirective.prototype, "aclReadonlyField$", void 0); (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(ReadonlyFieldDirective, [{ type: Directive, args: [{ selector: '[aclReadonlyField]', standalone: true, }] }], () => [{ type: i1.NgControl }, { type: i0.ComponentFactoryResolver }, { type: i0.ViewContainerRef }, { type: i2.FormItemControlDirective, decorators: [{ type: Self }, { type: Optional }] }], { aclReadonlyField: [{ type: Input }], aclReadonlyField$: [], hidden: [{ type: Input }], template: [{ type: Input, args: ['aclReadonlyFieldTemplate'] }], context: [{ type: Input, args: ['aclReadonlyFieldTemplateContext'] }], isHidden: [{ type: HostBinding, args: ['hidden'] }] }); })(); export class SpanComponent { setData(value) { this.data = value; } static { this.ɵfac = function SpanComponent_Factory(t) { return new (t || SpanComponent)(); }; } static { this.ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: SpanComponent, selectors: [["ng-component"]], decls: 3, vars: 3, template: function SpanComponent_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelementStart(0, "span"); i0.ɵɵtext(1); i0.ɵɵpipe(2, "aclFieldNotAvailable"); i0.ɵɵelementEnd(); } if (rf & 2) { i0.ɵɵadvance(); i0.ɵɵtextInterpolate(i0.ɵɵpipeBind1(2, 1, ctx.data)); } }, dependencies: [i3.FieldNotAvailablePipe], styles: ["span[_ngcontent-%COMP%]{word-break:break-all}"] }); } } (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(SpanComponent, [{ type: Component, args: [{ template: `<span>{{ data | aclFieldNotAvailable }}</span>`, changeDetection: ChangeDetectionStrategy.Default, styles: ["span{word-break:break-all}\n"] }] }], null, null); })(); (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassDebugInfo(SpanComponent, { className: "SpanComponent" }); })(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"readonly-field.directive.js","sourceRoot":"","sources":["../../../../../../libs/common/src/core/directives/readonly-field.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @packageDocumentation\n * @module utils\n */\n\nimport { FormItemControlDirective } from '@alauda/ui';\nimport {\n  CdkPortalOutlet,\n  ComponentPortal,\n  TemplatePortal,\n} from '@angular/cdk/portal';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  ComponentFactoryResolver,\n  ComponentRef,\n  Directive,\n  HostBinding,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Self,\n  TemplateRef,\n  ViewContainerRef,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  AsyncValidatorFn,\n  NgControl,\n  ValidatorFn,\n} from '@angular/forms';\nimport {\n  Observable,\n  Subject,\n  combineLatest,\n  distinctUntilChanged,\n  filter,\n  map,\n  startWith,\n  takeUntil,\n} from 'rxjs';\n\nimport { ObservableInput } from '../utils/decorators';\n\n@Directive({\n  selector: '[aclReadonlyField]',\n  standalone: true,\n})\nexport class ReadonlyFieldDirective implements OnInit, OnDestroy {\n  @Input()\n  aclReadonlyField: boolean;\n\n  @ObservableInput()\n  aclReadonlyField$: Observable<boolean>;\n\n  @Input()\n  hidden: boolean;\n\n  private isControlRequired = false;\n\n  private spanComponentRef: ComponentRef<SpanComponent>;\n\n  private readonly templateContext = {};\n\n  private readonly destroy$ = new Subject<void>();\n\n  // 自定义模板\n  @Input('aclReadonlyFieldTemplate')\n  template: TemplateRef<any>;\n\n  // 自定义模板\n  // eslint-disable-next-line @angular-eslint/no-input-rename\n  @Input('aclReadonlyFieldTemplateContext')\n  context: object;\n\n  @HostBinding('hidden')\n  get isHidden() {\n    return this.hidden || this.aclReadonlyField;\n  }\n\n  constructor(\n    private readonly control: NgControl,\n    // eslint-disable-next-line sonar/deprecation\n    private readonly cfr: ComponentFactoryResolver,\n    private readonly viewContainerRef: ViewContainerRef,\n    // 仅考虑当前control，子control无法知晓父FormItemControl元素 ，包裹了多少个control，以及各自处于什么状态，该情况自行处理\n    @Self()\n    @Optional()\n    private readonly controlDirective: FormItemControlDirective,\n  ) {}\n\n  private setValue(value: unknown) {\n    if (this.template) {\n      Object.assign(this.templateContext, {\n        $implicit: value,\n        ...this.context,\n      });\n    } else {\n      this.spanComponentRef.instance.setData(value);\n    }\n  }\n\n  ngOnInit() {\n    const controlEntity = this.control.control;\n    // formControl will compose multi validators to 1, no matter defined in template nor FormBuilder\n    const { validator, asyncValidator } = controlEntity;\n    this.isControlRequired = this.controlDirective?.required || false;\n    const portalOutlet = new CdkPortalOutlet(this.cfr, this.viewContainerRef);\n    const portal = this.template\n      ? new TemplatePortal(\n          this.template,\n          this.viewContainerRef,\n          this.templateContext,\n        )\n      : new ComponentPortal(SpanComponent, this.viewContainerRef);\n\n    const containerAttached$ = this.aclReadonlyField$.pipe(\n      distinctUntilChanged(),\n      map(isUpdate => {\n        if (!isUpdate) {\n          // 重置\n          if (this.controlDirective) {\n            this.controlDirective.required = this.isControlRequired;\n          }\n          if (portalOutlet.hasAttached()) {\n            portalOutlet.detach();\n          }\n          this.applyValidators(controlEntity, validator, asyncValidator);\n          return false;\n        }\n\n        if (this.controlDirective) {\n          this.controlDirective.required = false;\n        }\n        const attachPoint = portalOutlet.attach(portal);\n        if (!this.template) {\n          this.spanComponentRef = attachPoint as ComponentRef<SpanComponent>;\n        }\n        this.clearValidators(controlEntity);\n        return true;\n      }),\n    );\n\n    const sourceValue$ = this.control.valueChanges.pipe(\n      startWith(this.control.value),\n    );\n\n    combineLatest([sourceValue$, containerAttached$])\n      .pipe(\n        takeUntil(this.destroy$),\n        filter(([, attached]) => !!attached),\n      )\n      .subscribe(([value]) => {\n        this.setValue(value);\n      });\n  }\n\n  private applyValidators(\n    control: AbstractControl,\n    rawValidator?: ValidatorFn,\n    rawAsyncValidator?: AsyncValidatorFn,\n  ) {\n    if (\n      control.validator === rawValidator &&\n      control.asyncValidator === rawAsyncValidator\n    ) {\n      return;\n    }\n    control.setValidators(rawValidator);\n    control.setAsyncValidators(rawAsyncValidator);\n    control.updateValueAndValidity();\n  }\n\n  private clearValidators(control: AbstractControl) {\n    // delay checking validators for ngModel\n    requestAnimationFrame(() => {\n      if (!control.validator && !control.asyncValidator) {\n        return;\n      }\n      control.clearValidators();\n      control.clearAsyncValidators();\n      control.updateValueAndValidity();\n    });\n  }\n\n  ngOnDestroy() {\n    this.destroy$.next();\n  }\n}\n\n@Component({\n  template: `<span>{{ data | aclFieldNotAvailable }}</span>`,\n  styles: [\n    `\n      span {\n        word-break: break-all;\n      }\n    `,\n  ],\n  // eslint-disable-next-line @angular-eslint/prefer-on-push-component-change-detection\n  changeDetection: ChangeDetectionStrategy.Default,\n})\nexport class SpanComponent<T = unknown> {\n  data: T;\n\n  setData(value: T) {\n    this.data = value;\n  }\n}\n"]}