@alauda-fe/common
Version:
Alauda frontend team common codes.
280 lines • 47.4 kB
JavaScript
import { Injectable, IterableDiffers, } from '@angular/core';
import { cloneDeep, isEqual, set, unset } from 'lodash-es';
import * as i0 from "@angular/core";
export const DUPLICATE_ERROR_KEY = 'duplicateError';
// IDENTICAL: 检测具有互反性,传递性。a与b重复,b与c重复,a必然与c重复,此程度上可进行优化
// FULL: 不要求检测具有互反性,传递性。a与b判别重复,b与c判别重复,a未必与c判别重复
// 主要是针对一些复杂的自定义匹配重复的使用场景,FULL模式采用全检查
export var DUPLICATION_JUSTIFY_STRATEGY;
(function (DUPLICATION_JUSTIFY_STRATEGY) {
DUPLICATION_JUSTIFY_STRATEGY["IDENTICAL"] = "identical";
DUPLICATION_JUSTIFY_STRATEGY["FULL"] = "full";
})(DUPLICATION_JUSTIFY_STRATEGY || (DUPLICATION_JUSTIFY_STRATEGY = {}));
export const STRATEGY_JUDGE_MAPPER = {
[DUPLICATION_JUSTIFY_STRATEGY.IDENTICAL]: identicalStrategyCheck,
[DUPLICATION_JUSTIFY_STRATEGY.FULL]: fullStrategyCheck,
};
// use in component lifecycle
export class ValidateRowDuplicateService {
/**
* @param differFactory: used to create support differ, which can be provided by user
*/
constructor(arrayDiffers) {
this.arrayDiffers = arrayDiffers;
// used to save last value, to check whether if control changed
this.controlValueMap = new WeakMap();
// used to save errors,to solve validators concurrently work(second validator can't get error from previous validator in one turn)
this.errorMap = new WeakMap();
}
// eslint-disable-next-line sonarjs/cognitive-complexity
createFormArrayValidator({ rowKeys, formArray, valueMapFn = defaultValueMapFn, initData = [], matchesFn = matchValue, strategy = DUPLICATION_JUSTIFY_STRATEGY.IDENTICAL, }) {
const keys = formatKeys(rowKeys);
if (!this.rowsDiffer) {
this.rowsDiffer = this.arrayDiffers.find([]).create();
}
this.initFormArraySizeChange(formArray, this.rowsDiffer, keys, valueMapFn, matchesFn, strategy, initData);
return (control) => {
const { controls } = formArray;
const lastValue = this.controlValueMap.get(control);
if (control.pristine) {
return null;
}
// nothing change
const currentValue = getValueInKeys(control.value, keys, valueMapFn);
this.controlValueMap.set(control, currentValue);
if (lastValue) {
let identical = true;
Object.entries(keys).forEach(([key, props]) => {
if (!equalsValue(props, currentValue[key], lastValue[key])) {
identical = false;
}
});
if (identical) {
return this.errorMap.get(control);
}
}
const diffKeys = lastValue
? diffKeysByValue(keys, currentValue, lastValue)
: keys;
// 和上次值存在差异,可将老数据作为删除行处理
if (lastValue) {
this.handleRemoveRowValue(diffKeys, controls, lastValue, valueMapFn, control, matchesFn, strategy, initData);
}
// 更新后的状态,可能导致其他行出现新的重复
const otherControls = controls.filter(ctrl => ctrl !== control);
const ctrlError = {};
otherControls.forEach(ctrl => {
const otherValue = getValueInKeys(ctrl.value, diffKeys, valueMapFn);
const othersError = {};
// keys 中分别比较 如果 otherValue 与 currentValue 在某个key下,数据相同,则说明此 key 存在重复错误
Object.entries(diffKeys).forEach(([key, props]) => {
if (matchesFn(props, otherValue[key], currentValue[key])) {
set(othersError, [key], currentValue[key]);
set(ctrlError, key, currentValue[key]);
}
});
this.addDuplicateError(othersError, ctrl, true);
});
(initData || []).forEach(initRow => {
Object.entries(diffKeys).forEach(([key, props]) => {
if (matchesFn(props, initRow[key], currentValue[key])) {
set(ctrlError, key, currentValue[key]);
}
});
});
this.addDuplicateError(ctrlError, control);
return this.errorMap.get(control);
};
}
handleRemoveRowValue(keys, controls, staleValue, mapFn, selfControl, matchesFn, strategy, initData) {
if (selfControl) {
this.removeDuplicateError(Object.keys(keys), selfControl);
}
const toRemoveError = STRATEGY_JUDGE_MAPPER[strategy](keys, controls, staleValue, mapFn, matchesFn, initData);
(toRemoveError || []).forEach(({ key, control }) => this.removeDuplicateError([key], control, true));
}
initFormArraySizeChange(formArray, rowsDiffer, keys, mapFn, matchesFn, strategy, initData) {
if (!this.formArraySizeSub) {
this.formArraySizeSub = formArray.valueChanges.subscribe(_ => {
const { controls } = formArray;
const rowsDiff = rowsDiffer.diff(controls);
if (rowsDiff) {
// on deleting row
rowsDiff.forEachRemovedItem(record => {
const rawValue = getValueInKeys(record.item.value, keys, mapFn);
this.handleRemoveRowValue(keys, controls, rawValue, mapFn, undefined, matchesFn, strategy, initData);
});
}
});
}
}
addDuplicateError(errorInfo, control, setControl = false) {
const errors = cloneDeep(this.errorMap.get(control)) || {};
// already has error,no need to add
if (!Object.keys(errorInfo).some(key => errors?.[DUPLICATE_ERROR_KEY]?.[key] == null)) {
return;
}
Object.keys(errorInfo).forEach(key => {
set(errors, [DUPLICATE_ERROR_KEY, key], errorInfo[key]);
});
this.errorMap.set(control, errors);
if (setControl) {
// 当设置重复报错时,需要注意保留原来的报错
const controlError = cloneDeep(control.errors);
unset(controlError, [DUPLICATE_ERROR_KEY]);
const error = { ...controlError, ...this.errorMap.get(control) };
control.setErrors(Object.keys(error).length === 0 ? null : error, {
emitEvent: false,
});
// control's error should be shown,it's not pristine any more
control.markAsDirty();
}
}
removeDuplicateError(errorInfo, control, setControl = false) {
const errors = cloneDeep(this.errorMap.get(control));
// no need to remove error
if (!errorInfo.some(key => errors?.[DUPLICATE_ERROR_KEY]?.[key] != null)) {
return;
}
errorInfo.forEach(key => {
unset(errors, [DUPLICATE_ERROR_KEY, key]);
});
this.errorMap.set(control, Object.keys(errors?.[DUPLICATE_ERROR_KEY] || {}).length ? errors : null);
if (setControl) {
const controlError = cloneDeep(control.errors) || {};
unset(controlError, [DUPLICATE_ERROR_KEY]);
const error = { ...controlError, ...this.errorMap.get(control) };
control.setErrors(Object.keys(error).length === 0 ? null : error, {
emitEvent: false,
});
}
}
ngOnDestroy() {
if (this.formArraySizeSub) {
this.formArraySizeSub.unsubscribe();
}
}
static { this.ɵfac = function ValidateRowDuplicateService_Factory(t) { return new (t || ValidateRowDuplicateService)(i0.ɵɵinject(i0.IterableDiffers)); }; }
static { this.ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: ValidateRowDuplicateService, factory: ValidateRowDuplicateService.ɵfac }); }
}
(() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(ValidateRowDuplicateService, [{
type: Injectable
}], () => [{ type: i0.IterableDiffers }], null); })();
// 仅返回keys规则下,currentValue与lastValue存在不同的部分key
function diffKeysByValue(keys, currentValue, lastValue) {
const obj = {};
Object.entries(keys).forEach(([key, props]) => {
if (!equalsValue(props, currentValue[key], lastValue[key])) {
obj[key] = props;
}
});
return obj;
}
// 指定属性下,value1 和 value2 对于所有的属性存在有效值,并且相同,此时可判定重复
function matchValue(props, value1, value2) {
return !props.some(prop => unCheckedValue(value1, prop) || value1[prop] !== value2[prop]);
}
// 指定属性下,value1 和 value2 对于所有的属性相同,此时可判定 value1 , value2 相等
function equalsValue(props, value1, value2) {
return !props.some(prop => !isEqual(value1[prop], value2[prop]));
}
/**
* @param keys ['protocol',{complex: ['containerPort','hostPort']}]
* @returns [{protocol: ['protocol']}, {complex: ['containerPort','hostPort']}]
*/
function formatKeys(keys = []) {
const obj = {};
keys.forEach(keyObj => {
if (typeof keyObj === 'string') {
obj[keyObj] = [keyObj];
return;
}
Object.keys(keyObj).forEach(key => {
obj[key] = keyObj[key];
});
});
return obj;
}
function unCheckedValue(value, key) {
return value?.[key] == null || value[key] === '';
}
function defaultValueMapFn(data, prop) {
return data?.[prop];
}
/**
* @param value input value,which should include all key behind keys
* @param keys formatted keys, key is error key, like [{'containerPort':['containerPort']},{'complexError':['containerPort','hostPort']}]
* @returns
*/
function getValueInKeys(value, keys, mapFn) {
const obj = {};
Object.entries(keys).forEach(([key, props]) => {
props.forEach(prop => {
set(obj, [key, prop], mapFn(value, prop));
});
});
return obj;
}
// 互反性检查
function identicalStrategyCheck(keys, controls, staleValue, mapFn, matchesFn, initData) {
const errorsCounter = {};
Object.entries(keys).forEach(([key, props]) => {
errorsCounter[key] = { controls: new Set(), count: 0 };
initData.forEach(initRow => {
if (matchValue(props, initRow[key], staleValue[key])) {
errorsCounter[key].count++;
}
});
controls.forEach(ctrl => {
const controlValue = getValueInKeys(ctrl.value, keys, mapFn);
// 互反性策略,要求和老数据相比,因此只有和老数据相同,但和任意当前其他数据不同的,才需要去掉之前的错误
if (matchesFn(props, controlValue[key], staleValue[key])) {
errorsCounter[key].count++;
errorsCounter[key].controls.add(ctrl);
}
});
});
const toRemoveError = [];
Object.entries(errorsCounter).forEach(([key, item]) => {
if (item.count === 1) {
item.controls.forEach(control => {
toRemoveError.push({ key, control });
});
}
});
return toRemoveError;
}
// 全检查
function fullStrategyCheck(keys, controls, _, mapFn, matchesFn, initData) {
const controlError = new Map();
Object.entries(keys).forEach(([key, props]) => {
controls.forEach(ctrl => {
const controlValue = getValueInKeys(ctrl.value, keys, mapFn);
// 全检查策略,最终错误 key 对应 count 为 0 的ctrl会去掉相应错误
controlError.set(ctrl, { key, count: 0 });
initData.forEach(initRow => {
if (matchValue(props, initRow[key], controlValue)) {
const rawCount = controlError.get(ctrl).count;
controlError.set(ctrl, { key, count: rawCount + 1 });
}
});
controls
.filter(otherControl => otherControl !== ctrl)
.forEach(otherControl => {
const otherValue = getValueInKeys(otherControl.value, keys, mapFn);
if (matchesFn(props, controlValue[key], otherValue[key])) {
const rawCount = controlError.get(ctrl).count;
controlError.set(ctrl, { key, count: rawCount + 1 });
}
});
});
});
const toRemoveError = [];
controlError.forEach(({ count, key }, control) => {
if (count === 0) {
toRemoveError.push({ key, control });
}
});
return toRemoveError;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-row-duplicate.service.js","sourceRoot":"","sources":["../../../../../../../libs/common/src/core/services/form-row-duplicate/form-row-duplicate.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Injectable,\n  IterableDiffer,\n  IterableDiffers,\n  OnDestroy,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  FormArray,\n  FormGroup,\n  ValidationErrors,\n  ValidatorFn,\n} from '@angular/forms';\nimport { cloneDeep, isEqual, set, unset } from 'lodash-es';\nimport { Subscription } from 'rxjs';\n\nexport const DUPLICATE_ERROR_KEY = 'duplicateError';\n\ntype RowKeys = Array<string | Record<string, string[]>>;\ntype FormatKeys = Record<string, string[]>;\nexport type FormatValue = Record<string, any>;\n\nexport type MatchValueFn = (\n  props: string[],\n  value1: FormatValue,\n  value2: FormatValue,\n) => boolean;\n\nexport type StrategyFn = (\n  keys: FormatKeys,\n  controls: AbstractControl[],\n  staleValue: FormatValue,\n  mapFn: ValueMapFn,\n  matchesFn: MatchValueFn,\n  initData: any[],\n) => Array<{ key: string; control: AbstractControl }>;\n\n// IDENTICAL: 检测具有互反性，传递性。a与b重复，b与c重复，a必然与c重复，此程度上可进行优化\n// FULL: 不要求检测具有互反性，传递性。a与b判别重复，b与c判别重复，a未必与c判别重复\n// 主要是针对一些复杂的自定义匹配重复的使用场景，FULL模式采用全检查\nexport enum DUPLICATION_JUSTIFY_STRATEGY {\n  IDENTICAL = 'identical',\n  FULL = 'full',\n}\n\nexport const STRATEGY_JUDGE_MAPPER: Record<\n  DUPLICATION_JUSTIFY_STRATEGY,\n  StrategyFn\n> = {\n  [DUPLICATION_JUSTIFY_STRATEGY.IDENTICAL]: identicalStrategyCheck,\n  [DUPLICATION_JUSTIFY_STRATEGY.FULL]: fullStrategyCheck,\n};\n\n// use in component lifecycle\n@Injectable()\nexport class ValidateRowDuplicateService implements OnDestroy {\n  // used to detect all controls data change\n  private rowsDiffer: IterableDiffer<AbstractControl>;\n\n  /**\n   * @param differFactory: used to create support differ, which can be provided by user\n   */\n  constructor(private readonly arrayDiffers: IterableDiffers) {}\n\n  // used to save last value, to check whether if control changed\n  private readonly controlValueMap = new WeakMap<\n    AbstractControl,\n    FormatValue\n  >();\n\n  // used to save errors，to solve validators concurrently work(second validator can't get error from previous validator in one turn)\n  private readonly errorMap = new WeakMap<AbstractControl, ValidationErrors>();\n\n  private formArraySizeSub: Subscription;\n\n  // eslint-disable-next-line sonarjs/cognitive-complexity\n  createFormArrayValidator({\n    rowKeys,\n    formArray,\n    valueMapFn = defaultValueMapFn,\n    initData = [],\n    matchesFn = matchValue,\n    strategy = DUPLICATION_JUSTIFY_STRATEGY.IDENTICAL,\n  }: {\n    rowKeys: RowKeys;\n    formArray: FormArray;\n    valueMapFn?: any;\n    initData?: any[];\n    matchesFn?: MatchValueFn;\n    strategy?: DUPLICATION_JUSTIFY_STRATEGY;\n  }): ValidatorFn {\n    const keys = formatKeys(rowKeys);\n    if (!this.rowsDiffer) {\n      this.rowsDiffer = this.arrayDiffers.find([]).create();\n    }\n\n    this.initFormArraySizeChange(\n      formArray,\n      this.rowsDiffer,\n      keys,\n      valueMapFn,\n      matchesFn,\n      strategy,\n      initData,\n    );\n\n    return (control: FormGroup) => {\n      const { controls } = formArray;\n      const lastValue = this.controlValueMap.get(control);\n      if (control.pristine) {\n        return null;\n      }\n      // nothing change\n      const currentValue = getValueInKeys(control.value, keys, valueMapFn);\n      this.controlValueMap.set(control, currentValue);\n      if (lastValue) {\n        let identical = true;\n        Object.entries(keys).forEach(([key, props]) => {\n          if (!equalsValue(props, currentValue[key], lastValue[key])) {\n            identical = false;\n          }\n        });\n        if (identical) {\n          return this.errorMap.get(control);\n        }\n      }\n      const diffKeys = lastValue\n        ? diffKeysByValue(keys, currentValue, lastValue)\n        : keys;\n      // 和上次值存在差异，可将老数据作为删除行处理\n      if (lastValue) {\n        this.handleRemoveRowValue(\n          diffKeys,\n          controls,\n          lastValue,\n          valueMapFn,\n          control,\n          matchesFn,\n          strategy,\n          initData,\n        );\n      }\n      // 更新后的状态，可能导致其他行出现新的重复\n      const otherControls = controls.filter(ctrl => ctrl !== control);\n      const ctrlError: Record<string, any> = {};\n      otherControls.forEach(ctrl => {\n        const otherValue = getValueInKeys(ctrl.value, diffKeys, valueMapFn);\n        const othersError: Record<string, any> = {};\n        // keys 中分别比较 如果 otherValue 与 currentValue 在某个key下，数据相同，则说明此 key 存在重复错误\n        Object.entries(diffKeys).forEach(([key, props]) => {\n          if (matchesFn(props, otherValue[key], currentValue[key])) {\n            set(othersError, [key], currentValue[key]);\n            set(ctrlError, key, currentValue[key]);\n          }\n        });\n        this.addDuplicateError(othersError, ctrl, true);\n      });\n      (initData || []).forEach(initRow => {\n        Object.entries(diffKeys).forEach(([key, props]) => {\n          if (matchesFn(props, initRow[key], currentValue[key])) {\n            set(ctrlError, key, currentValue[key]);\n          }\n        });\n      });\n      this.addDuplicateError(ctrlError, control);\n      return this.errorMap.get(control);\n    };\n  }\n\n  private handleRemoveRowValue(\n    keys: FormatKeys,\n    controls: AbstractControl[],\n    staleValue: FormatValue,\n    mapFn: ValueMapFn,\n    selfControl?: AbstractControl,\n    matchesFn?: MatchValueFn,\n    strategy?: DUPLICATION_JUSTIFY_STRATEGY,\n    initData?: any[],\n  ) {\n    if (selfControl) {\n      this.removeDuplicateError(Object.keys(keys), selfControl);\n    }\n    const toRemoveError = STRATEGY_JUDGE_MAPPER[strategy](\n      keys,\n      controls,\n      staleValue,\n      mapFn,\n      matchesFn,\n      initData,\n    );\n    (toRemoveError || []).forEach(({ key, control }) =>\n      this.removeDuplicateError([key], control, true),\n    );\n  }\n\n  private initFormArraySizeChange(\n    formArray: FormArray,\n    rowsDiffer: IterableDiffer<AbstractControl>,\n    keys: FormatKeys,\n    mapFn: ValueMapFn,\n    matchesFn: MatchValueFn,\n    strategy: DUPLICATION_JUSTIFY_STRATEGY,\n    initData?: any[],\n  ) {\n    if (!this.formArraySizeSub) {\n      this.formArraySizeSub = formArray.valueChanges.subscribe(_ => {\n        const { controls } = formArray;\n        const rowsDiff = rowsDiffer.diff(controls);\n        if (rowsDiff) {\n          // on deleting row\n          rowsDiff.forEachRemovedItem(record => {\n            const rawValue = getValueInKeys(record.item.value, keys, mapFn);\n            this.handleRemoveRowValue(\n              keys,\n              controls,\n              rawValue,\n              mapFn,\n              undefined,\n              matchesFn,\n              strategy,\n              initData,\n            );\n          });\n        }\n      });\n    }\n  }\n\n  private addDuplicateError(\n    errorInfo: Record<string, any>,\n    control: AbstractControl,\n    setControl = false,\n  ) {\n    const errors = cloneDeep(this.errorMap.get(control)) || {};\n    // already has error,no need to add\n    if (\n      !Object.keys(errorInfo).some(\n        key => errors?.[DUPLICATE_ERROR_KEY]?.[key] == null,\n      )\n    ) {\n      return;\n    }\n\n    Object.keys(errorInfo).forEach(key => {\n      set(errors, [DUPLICATE_ERROR_KEY, key], errorInfo[key]);\n    });\n    this.errorMap.set(control, errors);\n    if (setControl) {\n      // 当设置重复报错时,需要注意保留原来的报错\n      const controlError = cloneDeep(control.errors);\n      unset(controlError, [DUPLICATE_ERROR_KEY]);\n      const error = { ...controlError, ...this.errorMap.get(control) };\n      control.setErrors(Object.keys(error).length === 0 ? null : error, {\n        emitEvent: false,\n      });\n\n      // control's error should be shown,it's not pristine any more\n      control.markAsDirty();\n    }\n  }\n\n  private removeDuplicateError(\n    errorInfo: string[],\n    control: AbstractControl,\n    setControl = false,\n  ) {\n    const errors = cloneDeep(this.errorMap.get(control));\n    // no need to remove error\n    if (!errorInfo.some(key => errors?.[DUPLICATE_ERROR_KEY]?.[key] != null)) {\n      return;\n    }\n    errorInfo.forEach(key => {\n      unset(errors, [DUPLICATE_ERROR_KEY, key]);\n    });\n    this.errorMap.set(\n      control,\n      Object.keys(errors?.[DUPLICATE_ERROR_KEY] || {}).length ? errors : null,\n    );\n    if (setControl) {\n      const controlError: Record<string, any> = cloneDeep(control.errors) || {};\n      unset(controlError, [DUPLICATE_ERROR_KEY]);\n\n      const error = { ...controlError, ...this.errorMap.get(control) };\n      control.setErrors(Object.keys(error).length === 0 ? null : error, {\n        emitEvent: false,\n      });\n    }\n  }\n\n  ngOnDestroy() {\n    if (this.formArraySizeSub) {\n      this.formArraySizeSub.unsubscribe();\n    }\n  }\n}\n\n// 仅返回keys规则下，currentValue与lastValue存在不同的部分key\nfunction diffKeysByValue(\n  keys: FormatKeys,\n  currentValue: FormatValue,\n  lastValue: FormatValue,\n) {\n  const obj: FormatKeys = {};\n  Object.entries(keys).forEach(([key, props]) => {\n    if (!equalsValue(props, currentValue[key], lastValue[key])) {\n      obj[key] = props;\n    }\n  });\n  return obj;\n}\n\n// 指定属性下，value1 和 value2 对于所有的属性存在有效值，并且相同，此时可判定重复\nfunction matchValue(props: string[], value1: FormatValue, value2: FormatValue) {\n  return !props.some(\n    prop => unCheckedValue(value1, prop) || value1[prop] !== value2[prop],\n  );\n}\n\n// 指定属性下，value1 和 value2 对于所有的属性相同，此时可判定 value1 , value2 相等\nfunction equalsValue(\n  props: string[],\n  value1: FormatValue,\n  value2: FormatValue,\n) {\n  return !props.some(prop => !isEqual(value1[prop], value2[prop]));\n}\n\n/**\n * @param keys ['protocol',{complex: ['containerPort','hostPort']}]\n * @returns [{protocol: ['protocol']}, {complex: ['containerPort','hostPort']}]\n */\nfunction formatKeys(keys: RowKeys = []): FormatKeys {\n  const obj: FormatKeys = {};\n  keys.forEach(keyObj => {\n    if (typeof keyObj === 'string') {\n      obj[keyObj] = [keyObj];\n      return;\n    }\n    Object.keys(keyObj).forEach(key => {\n      obj[key] = keyObj[key];\n    });\n  });\n  return obj;\n}\n\nfunction unCheckedValue(value: FormatValue, key: string) {\n  return value?.[key] == null || value[key] === '';\n}\n\nfunction defaultValueMapFn(data: any, prop: string) {\n  return data?.[prop];\n}\n\ntype ValueMapFn = (data: any, prop?: string) => any;\n\n/**\n * @param value input value,which should include all key behind keys\n * @param keys formatted keys, key is error key, like [{'containerPort':['containerPort']},{'complexError':['containerPort','hostPort']}]\n * @returns\n */\nfunction getValueInKeys(\n  value: any,\n  keys: FormatKeys,\n  mapFn: ValueMapFn,\n): FormatValue {\n  const obj = {};\n  Object.entries(keys).forEach(([key, props]) => {\n    props.forEach(prop => {\n      set(obj, [key, prop], mapFn(value, prop));\n    });\n  });\n  return obj;\n}\n\n// 互反性检查\nfunction identicalStrategyCheck(\n  keys: FormatKeys,\n  controls: AbstractControl[],\n  staleValue: FormatValue,\n  mapFn: ValueMapFn,\n  matchesFn: MatchValueFn,\n  initData: any[],\n) {\n  const errorsCounter: Record<\n    string,\n    { controls: Set<AbstractControl>; count: number }\n  > = {};\n\n  Object.entries(keys).forEach(([key, props]) => {\n    errorsCounter[key] = { controls: new Set(), count: 0 };\n    initData.forEach(initRow => {\n      if (matchValue(props, initRow[key], staleValue[key])) {\n        errorsCounter[key].count++;\n      }\n    });\n    controls.forEach(ctrl => {\n      const controlValue = getValueInKeys(ctrl.value, keys, mapFn);\n      // 互反性策略，要求和老数据相比，因此只有和老数据相同，但和任意当前其他数据不同的，才需要去掉之前的错误\n      if (matchesFn(props, controlValue[key], staleValue[key])) {\n        errorsCounter[key].count++;\n        errorsCounter[key].controls.add(ctrl);\n      }\n    });\n  });\n  const toRemoveError: Array<{ key: string; control: AbstractControl }> = [];\n  Object.entries(errorsCounter).forEach(([key, item]) => {\n    if (item.count === 1) {\n      item.controls.forEach(control => {\n        toRemoveError.push({ key, control });\n      });\n    }\n  });\n  return toRemoveError;\n}\n\n// 全检查\nfunction fullStrategyCheck(\n  keys: FormatKeys,\n  controls: AbstractControl[],\n  _: FormatValue,\n  mapFn: ValueMapFn,\n  matchesFn: MatchValueFn,\n  initData: any[],\n) {\n  const controlError: Map<AbstractControl, { key: string; count: number }> =\n    new Map();\n\n  Object.entries(keys).forEach(([key, props]) => {\n    controls.forEach(ctrl => {\n      const controlValue = getValueInKeys(ctrl.value, keys, mapFn);\n      // 全检查策略，最终错误 key 对应 count 为 0 的ctrl会去掉相应错误\n      controlError.set(ctrl, { key, count: 0 });\n      initData.forEach(initRow => {\n        if (matchValue(props, initRow[key], controlValue)) {\n          const rawCount = controlError.get(ctrl).count;\n          controlError.set(ctrl, { key, count: rawCount + 1 });\n        }\n      });\n      controls\n        .filter(otherControl => otherControl !== ctrl)\n        .forEach(otherControl => {\n          const otherValue = getValueInKeys(otherControl.value, keys, mapFn);\n          if (matchesFn(props, controlValue[key], otherValue[key])) {\n            const rawCount = controlError.get(ctrl).count;\n            controlError.set(ctrl, { key, count: rawCount + 1 });\n          }\n        });\n    });\n  });\n  const toRemoveError: Array<{ key: string; control: AbstractControl }> = [];\n  controlError.forEach(({ count, key }, control) => {\n    if (count === 0) {\n      toRemoveError.push({ key, control });\n    }\n  });\n  return toRemoveError;\n}\n"]}