在某些情况下,需要对已调度的动作进行去抖动并减少发送到我们的API的请求。 让我们考虑一个简单的应用程序,该应用程序呈现新闻列表并提供在所有新闻中进行搜索的功能:
class SearchNews {
static readonly type = '[News] Search news';
constructor(public title: string) {}
}
@Component({
selector: 'app-news-portal',
template: `
<app-news-search
(search)="search($event)"
[lastSearchedTitle]="lastSearchedTitle$ | async"
></app-news-search>
<app-news [news]="news$ | async"></app-news>
`
})
export class NewsPortalComponent implements OnDestroy {
@Select(NewsState.getNews) news$: Observable<News[]>;
lastSearchedTitle$ = this.store.selectOnce(NewsState.getLastSearchedTitle);
private destroy$ = new Subject<void>();
constructor(private store: Store, actions$: Actions) {
actions$
.pipe(
ofActionDispatched(SearchNews),
map((action: SearchNews) => action.title),
debounceTime(2000),
takeUntil(this.destroy$)
)
.subscribe(title => {
store.dispatch(new GetNews(title));
});
}
ngOnDestroy(): void {
this.destroy$.next();
this.destroy$.complete();
}
search(title: string): void {
this.store.dispatch(new SearchNews(title));
}
}
export interface NewsStateModel {
news: News[];
lastSearchedTitle: string | null;
}
export class GetNews {
static readonly type = '[News] Get news';
constructor(public title = '') {}
}
@State<NewsStateModel>({
name: 'news',
defaults: {
news: [],
lastSearchedTitle: null
}
})
@Injectable()
export class NewsState {
@Selector()
static getNews(state: NewsStateModel): News[] {
return state.news;
}
@Selector()
static getLastSearchedTitle(state: NewsStateModel): string | null {
return state.lastSearchedTitle;
}
constructor(private http: HttpClient) {}
@Action(GetNews)
getNews(ctx: StateContext<NewsStateModel>, { title }: GetNews) {
return this.http.get<News[]>(`/api/news?search=${title}`).pipe(
tap(news => {
ctx.setState({ news, lastSearchedTitle: title });
})
);
}
}