UNPKG

@alauda-fe/common

Version:

Alauda frontend team common codes.

198 lines 27.2 kB
import { coerceAttrBoolean } from '@alauda/ui'; import { ChangeDetectorRef, Directive, inject, Input, QueryList, ViewChildren, } from '@angular/core'; import { FormGroupDirective, NgControl, NgForm, } from '@angular/forms'; import { assocPath, identity, mergeDeepRight, path } from 'ramda'; import { ReplaySubject, startWith, switchMap, Subject, takeUntil, debounceTime, merge, combineLatest, filter, map, tap, take, of, } from 'rxjs'; import * as i0 from "@angular/core"; export function createNestedFormControl({ autoEmitChange = false, asyncValidator = false, } = {}) { return class NestedFormControl extends BaseNestedFormControl { constructor() { super(...arguments); this.autoEmitChange = autoEmitChange; this.asyncValidator = asyncValidator; } }; } export class BaseNestedFormControl { get disabled() { return this._disabled; } set disabled(val) { this._disabled = coerceAttrBoolean(val); } constructor() { this._disabled = false; this.model = null; this.submitted = false; this.autoEmitChange = false; this.asyncValidator = false; this.cdr = inject(ChangeDetectorRef); this.destroy$$ = new Subject(); this.hostForm = inject(NgForm, { optional: true }) ?? inject(FormGroupDirective, { optional: true }); if (this.hostForm) { this.hostForm.ngSubmit.pipe(takeUntil(this.destroy$$)).subscribe(() => { this.submitted = this.hostForm.submitted; this.onSubmitted?.(); this.cdr.markForCheck(); }); } } ngAfterViewInit() { this.controls.changes .pipe(startWith(this.controls), switchMap((controls) => controls.length ? merge(...controls.toArray().map(control => control.statusChanges)).pipe(startWith(null)) : of(null)), debounceTime(0)) .subscribe(() => { this.onValidatorChange?.(); }); if (this.autoEmitChange) { this.controls.changes .pipe(startWith(this.controls), switchMap((controls) => merge(...controls.toArray().map(control => control.valueChanges)))) .pipe(debounceTime(0)) .subscribe(() => { this.emitModel(this.model); }); } } ngOnDestroy() { this.destroy$$.next(); this.destroy$$.complete(); } registerOnValidatorChange(fn) { this.onValidatorChange = fn; } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(isDisabled) { this.disabled = isDisabled; this.cdr.markForCheck(); } writeValue(value) { this.model = this.valueIn(value); this.cdr.markForCheck(); } validate(_control) { const getErrors = () => { const errors = this.controls?.reduce((acc, control) => { if (control.errors) { return mergeDeepRight(acc, control.name ? { [control.name]: control.errors } : control.errors); } return acc; }, {}); return errors && Object.keys(errors).length ? errors : null; }; return this.asyncValidator ? combineLatest((this.controls?.toArray() ?? []).map(control => control.statusChanges.pipe(startWith(control.status)))).pipe(debounceTime(0), filter(statuses => statuses.includes('INVALID') || !statuses.includes('PENDING')), take(1), map(() => getErrors()), tap(() => { this.cdr.markForCheck(); })) : getErrors(); } emitModel(model = this.model) { this.model = model; this.emitValue(this.modelOut(this.model)); } emitValue(value) { if (this.onChange) { this.onChange(value); } } valueIn(value) { return value; } modelOut(model) { return model; } static { this.ɵfac = function BaseNestedFormControl_Factory(t) { return new (t || BaseNestedFormControl)(); }; } static { this.ɵdir = /*@__PURE__*/ i0.ɵɵdefineDirective({ type: BaseNestedFormControl, viewQuery: function BaseNestedFormControl_Query(rf, ctx) { if (rf & 1) { i0.ɵɵviewQuery(NgControl, 5); } if (rf & 2) { let _t; i0.ɵɵqueryRefresh(_t = i0.ɵɵloadQuery()) && (ctx.controls = _t); } }, inputs: { disabled: "disabled" } }); } } (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(BaseNestedFormControl, [{ type: Directive }], () => [], { controls: [{ type: ViewChildren, args: [NgControl] }], disabled: [{ type: Input }] }); })(); /** * @deprecated use {@link BaseNestedFormControl} instead. */ export class BaseNestedFormControlPure extends BaseNestedFormControl { constructor() { super(); this.model$ = new ReplaySubject(1); this.model$.subscribe(model => { this.model = model; this.cdr.markForCheck(); }); } writeValue(value) { this.model$.next(this.valueIn(value)); } emitValue(value) { if (this.onChange) { this.onChange(value); this.writeValue(value); } } static { this.ɵfac = function BaseNestedFormControlPure_Factory(t) { return new (t || BaseNestedFormControlPure)(); }; } static { this.ɵdir = /*@__PURE__*/ i0.ɵɵdefineDirective({ type: BaseNestedFormControlPure, features: [i0.ɵɵInheritDefinitionFeature] }); } } (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(BaseNestedFormControlPure, [{ type: Directive }], () => [], null); })(); function collectPaths(mapper, parent) { return Object.entries(mapper).reduce((acc, [key, value]) => { const path = typeof value === 'string' ? value.split('.') : value; if (Array.isArray(path)) { acc.push([parent.concat(key), path]); return acc; } return acc.concat(collectPaths(path, parent.concat(key))); }, []); } export function dataTransfer(mapper) { const tuples = collectPaths(mapper, []); let _source; let _target = {}; return [ source => { _source = source; return tuples.reduce((acc, [left, right]) => { const newValue = path(right, source); if (path(left, acc) === newValue) { return acc; } return (_target = assocPath(left, newValue, acc)); }, _target); }, target => { _target = target; return tuples.reduce((acc, [left, right]) => { const newValue = path(left, target); if (path(right, acc) === newValue) { return acc; } return (_source = assocPath(right, newValue, acc)); }, _source); }, ]; } function parsePath(path) { return typeof path === 'string' ? path.split('.') : path; } export function copyValue(from, to, mapper = identity) { const fromPath = parsePath(from); const toPath = parsePath(to); return (object) => assocPath(toPath, mapper(path(fromPath, object)), object); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-nested-form-control.js","sourceRoot":"","sources":["../../../../../../libs/common/src/core/abstract/base-nested-form-control.ts"],"names":[],"mappings":"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","sourcesContent":["import { coerceAttrBoolean } from '@alauda/ui';\nimport {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Directive,\n  inject,\n  Input,\n  OnDestroy,\n  QueryList,\n  ViewChildren,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  ControlValueAccessor,\n  FormGroupDirective,\n  NgControl,\n  NgForm,\n  ValidationErrors,\n} from '@angular/forms';\nimport { assocPath, identity, mergeDeepRight, path } from 'ramda';\nimport {\n  ReplaySubject,\n  startWith,\n  switchMap,\n  Subject,\n  takeUntil,\n  debounceTime,\n  merge,\n  Observable,\n  combineLatest,\n  filter,\n  map,\n  tap,\n  take,\n  of,\n} from 'rxjs';\n\nexport function createNestedFormControl<V, M = V>({\n  autoEmitChange = false,\n  asyncValidator = false,\n}: {\n  autoEmitChange?: boolean;\n  asyncValidator?: boolean;\n} = {}): // workaround: https://github.com/microsoft/TypeScript/issues/30355\nnew () => BaseNestedFormControl<V, M> {\n  return class NestedFormControl extends BaseNestedFormControl<V, M> {\n    protected override autoEmitChange = autoEmitChange;\n    protected override asyncValidator = asyncValidator;\n  };\n}\n\n@Directive()\nexport class BaseNestedFormControl<V, M = V>\n  implements AfterViewInit, OnDestroy, ControlValueAccessor\n{\n  @ViewChildren(NgControl)\n  protected controls: QueryList<NgControl>;\n\n  @Input()\n  get disabled() {\n    return this._disabled;\n  }\n\n  set disabled(val: boolean | '') {\n    this._disabled = coerceAttrBoolean(val);\n  }\n\n  private _disabled = false;\n\n  onValidatorChange: () => void;\n  onChange: (_: V) => void;\n  onTouched: () => void;\n  onSubmitted: () => void;\n\n  model: M = null;\n  submitted = false;\n\n  protected autoEmitChange = false;\n  protected asyncValidator = false;\n  protected cdr = inject(ChangeDetectorRef);\n  protected destroy$$ = new Subject<void>();\n  protected hostForm =\n    inject(NgForm, { optional: true }) ??\n    inject(FormGroupDirective, { optional: true });\n\n  constructor() {\n    if (this.hostForm) {\n      this.hostForm.ngSubmit.pipe(takeUntil(this.destroy$$)).subscribe(() => {\n        this.submitted = this.hostForm.submitted;\n        this.onSubmitted?.();\n        this.cdr.markForCheck();\n      });\n    }\n  }\n\n  ngAfterViewInit() {\n    this.controls.changes\n      .pipe(\n        startWith(this.controls),\n        switchMap((controls: QueryList<NgControl>) =>\n          controls.length\n            ? merge(\n                ...controls.toArray().map(control => control.statusChanges),\n              ).pipe(startWith(null))\n            : of(null),\n        ),\n        debounceTime(0),\n      )\n      .subscribe(() => {\n        this.onValidatorChange?.();\n      });\n\n    if (this.autoEmitChange) {\n      this.controls.changes\n        .pipe(\n          startWith(this.controls),\n          switchMap((controls: QueryList<NgControl>) =>\n            merge(...controls.toArray().map(control => control.valueChanges)),\n          ),\n        )\n        .pipe(debounceTime(0))\n        .subscribe(() => {\n          this.emitModel(this.model);\n        });\n    }\n  }\n\n  ngOnDestroy() {\n    this.destroy$$.next();\n    this.destroy$$.complete();\n  }\n\n  registerOnValidatorChange(fn: () => void) {\n    this.onValidatorChange = fn;\n  }\n\n  registerOnChange(fn: (_: V) => void) {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void) {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean) {\n    this.disabled = isDisabled;\n    this.cdr.markForCheck();\n  }\n\n  writeValue(value: V) {\n    this.model = this.valueIn(value);\n    this.cdr.markForCheck();\n  }\n\n  validate(\n    _control?: AbstractControl,\n  ): ValidationErrors | null | Observable<ValidationErrors | null> {\n    const getErrors = () => {\n      const errors = this.controls?.reduce((acc, control) => {\n        if (control.errors) {\n          return mergeDeepRight(\n            acc,\n            control.name ? { [control.name]: control.errors } : control.errors,\n          );\n        }\n        return acc;\n      }, {} as ValidationErrors);\n\n      return errors && Object.keys(errors).length ? errors : null;\n    };\n\n    return this.asyncValidator\n      ? combineLatest(\n          (this.controls?.toArray() ?? []).map(control =>\n            control.statusChanges.pipe(startWith(control.status)),\n          ),\n        ).pipe(\n          debounceTime(0),\n          filter(\n            statuses =>\n              statuses.includes('INVALID') || !statuses.includes('PENDING'),\n          ),\n          take(1),\n          map(() => getErrors()),\n          tap(() => {\n            this.cdr.markForCheck();\n          }),\n        )\n      : getErrors();\n  }\n\n  emitModel(model: M = this.model) {\n    this.model = model;\n    this.emitValue(this.modelOut(this.model));\n  }\n\n  emitValue(value: V) {\n    if (this.onChange) {\n      this.onChange(value);\n    }\n  }\n\n  protected valueIn(value: V): M {\n    return value as any;\n  }\n\n  protected modelOut(model: M): V {\n    return model as any;\n  }\n}\n\n/**\n * @deprecated use {@link BaseNestedFormControl} instead.\n */\n@Directive()\nexport class BaseNestedFormControlPure<V, M = V> extends BaseNestedFormControl<\n  V,\n  M\n> {\n  model$ = new ReplaySubject<M>(1);\n\n  constructor() {\n    super();\n\n    this.model$.subscribe(model => {\n      this.model = model;\n      this.cdr.markForCheck();\n    });\n  }\n\n  override writeValue(value: V) {\n    this.model$.next(this.valueIn(value));\n  }\n\n  override emitValue(value: V) {\n    if (this.onChange) {\n      this.onChange(value);\n      this.writeValue(value);\n    }\n  }\n}\n\ntype Path = string | Array<string | number>;\n\ninterface Mapper {\n  [x: string]: Mapper | Path;\n}\n\ntype PathTuple = [string[], string[]];\n\nfunction collectPaths(mapper: Mapper, parent: string[]): PathTuple[] {\n  return Object.entries(mapper).reduce((acc, [key, value]) => {\n    const path = typeof value === 'string' ? value.split('.') : value;\n    if (Array.isArray(path)) {\n      acc.push([parent.concat(key), path]);\n      return acc;\n    }\n    return acc.concat(collectPaths(path, parent.concat(key)));\n  }, []);\n}\n\nexport function dataTransfer<\n  S extends object,\n  T extends object,\n  M extends Record<keyof T, Path | Mapper> = Record<keyof T, Path | Mapper>,\n>(mapper: M): [(source: S) => T, (target: T) => S] {\n  const tuples = collectPaths(mapper, []);\n  let _source: S;\n  let _target: T = {} as T;\n  return [\n    source => {\n      _source = source;\n      return tuples.reduce((acc, [left, right]) => {\n        const newValue = path<unknown>(right, source);\n\n        if (path(left, acc) === newValue) {\n          return acc;\n        }\n\n        return (_target = assocPath<unknown, T>(left, newValue, acc));\n      }, _target);\n    },\n    target => {\n      _target = target;\n      return tuples.reduce((acc, [left, right]) => {\n        const newValue = path<unknown>(left, target);\n\n        if (path(right, acc) === newValue) {\n          return acc;\n        }\n\n        return (_source = assocPath<unknown, S>(right, newValue, acc));\n      }, _source);\n    },\n  ];\n}\n\nfunction parsePath(path: Path): Array<string | number> {\n  return typeof path === 'string' ? path.split('.') : path;\n}\n\nexport function copyValue<T extends object>(\n  from: Path,\n  to: Path,\n  mapper: (v: any) => any = identity,\n) {\n  const fromPath = parsePath(from);\n  const toPath = parsePath(to);\n  return (object: T) =>\n    assocPath(toPath, mapper(path(fromPath, object)), object);\n}\n"]}