@alauda-fe/common
Version:
Alauda frontend team common codes.
198 lines • 27.2 kB
JavaScript
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"]}