gpt4 book ai didi

javascript - 如何在异步验证器中使用 debounceTime() 和 distinctUntilChanged()

转载 作者:行者123 更新时间:2023-11-30 19:31:36 24 4
gpt4 key购买 nike

我想在我的异步验证器中添加 debounceTimedistinctUntilChanged

mockAsyncValidator(): AsyncValidatorFn {
return (control: FormControl): Observable<ValidationErrors | null> => {
return control.valueChanges.pipe(
debounceTime(500),
distinctUntilChanged(),
switchMap(value => {
console.log(value); // log works here
return this.mockService.checkValue(value).pipe(response => {
console.log(response); // log did not work here
if (response) {
return { invalid: true };
}
return null;
})
})
);
}

上面的代码不起作用,表单状态变为PENDING
但是当我在 this answer 中使用 timer 时,代码有效,但我不能使用 distinctUntilChanged

return timer(500).pipe(
switchMap(() => {
return this.mockService.checkValue(control.value).pipe(response => {
console.log(response); // log works here
if (response) {
return { invalid: true };
}
return null;
})
})
);

我尝试像这样使用 BehaviorSubject

debouncedSubject = new BehaviorSubject<string>('');

并在 AsyncValidatorFn 中使用它,但仍然不起作用,如下所示:

this.debouncedSubject.next(control.value);
return this.debouncedSubject.pipe(
debounceTime(500),
distinctUntilChanged(), // did not work
// I think maybe it's because of I next() the value
// immediately above
// but I don't know how to fix this
take(1), // have to add this, otherwise, the form is PENDING forever
// and this take(1) cannot add before debounceTime()
// otherwise debounceTime() won't work
switchMap(value => {
console.log(value); // log works here
return this.mockService.checkValue(control.value).pipe(response => {
console.log(response); // log works here
if (response) {
return { invalid: true };
}
return null;
}
);
})
);

最佳答案

问题是,当您在 validatorFn 中调用 pipe() 时,每次执行 validatorFn 时都会构建一个新管道。先前的值不会被捕获以用于 discinct 或 debounce 工作。你可以做的是在外部设置两个 BehaviourSubjects,在我的例子中是 termDebouncervalidationEmitter

您可以设置一个工厂方法来创建此验证器,从而重新使用它。您还可以扩展 AsyncValidator 并创建一个带有 DI 设置的类。我将在下面展示工厂方法。

export function AsyncValidatorFactory(mockService: MockService) { 
const termDebouncer = new BehaviorSubject('');
const validationEmitter = new BehaviorSubject<T>(null);
let prevTerm = '';
let prevValidity = null;

termDebouncer.pipe(
map(val => (val + '').trim()),
filter(val => val.length > 0),
debounceTime(500),
mergeMap(term => { const obs = term === prevTerm ? of(prevValidity) : mockService.checkValue(term);
prevTerm = term;
return obs; }),
map(respose => { invalid: true } : null),
tap(validity => prevValidity = validity)
).subscribe(validity => validationEmitter.next(validity))


return (control: AbstractControl) => {
termDebouncer.next(control.value)
return validationEmitter.asObservable().pipe(take(2))
}
}

编辑:此代码摘录来自 Angular 表单验证以外的用例(准确地说是 React 搜索小部件。)管道运算符可能需要更改以适合您的用例。

Edit2:take(1)first() 以确保可观察对象在发出验证消息后完成。 asObservable() 将确保在下一次调用时生成一个新的可观察对象。您也可以跳过 asObservable() 而只是 pipe(),因为管道运算符会分支异步管道并从那里开始创建一个新的可观察对象。您可能必须使用 take(2) 来克服 behaviorSubject 是有状态的并持有一个值这一事实。

Edit3:使用合并映射来处理 distinctUntilChanged() 将导致 observable 不发射和不完成的事实。

关于javascript - 如何在异步验证器中使用 debounceTime() 和 distinctUntilChanged(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56381335/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com