The TanStack Query (also known as react-query) adapter for Angular applications
Get rid of granular state management, manual refetching, and async spaghetti code. TanStack Query gives you declarative, always-up-to-date auto-managed queries and mutations that improve your developer experience.
β
Backend agnostic
β
Dedicated Devtools
β
Auto Caching
β
Auto Refetching
β
Window Focus Refetching
β
Polling/Realtime Queries
β
Parallel Queries
β
Dependent Queries
β
Mutations API
β
Automatic Garbage Collection
β
Paginated/Cursor Queries
β
Load-More/Infinite Scroll Queries
β
Request Cancellation
β
Prefetching
β
Offline Support
β
Data Selectors
β
SSR Support
- Features
- Table of Contents
- Installation
- Queries
- Query Client
- Mutations
- Query Global Options
- Operators
- Utils
- Use Constructor DI
- Devtools
- SSR
- Created By
- Contributors β¨
npm
npm i -S @ngneat/query
Yarn
yarn add --save @ngneat/query
Inject the QueryClientService
provider to get access to the query client instance:
import { injectQueryClient } from '@ngneat/query';
@Injectable({ providedIn: 'root' })
export class TodosService {
private queryClient = injectQueryClient();
}
Inject the UseQuery
in your service. Using the hook is similar to the official hook, except the query function should return an observable
.
import { UseQuery } from '@ngneat/query';
@Injectable({ providedIn: 'root' })
export class TodosService {
private http = inject(HttpClient);
private useQuery = inject(UseQuery);
getTodos() {
return this.useQuery(['todos'], () => {
return this.http.get<Todo[]>(
'https://jsonplaceholder.typicode.com/todos'
);
});
}
getTodo(id: number) {
return this.useQuery(['todo', id], () => {
return this.http.get<Todo>(
`https://jsonplaceholder.typicode.com/todos/${id}`
);
});
}
}
Use it in your component:
import { SubscribeModule } from '@ngneat/subscribe';
@Component({
standalone: true,
imports: [NgIf, NgForOf, SpinnerComponent, SubscribeModule],
template: `
<ng-container *subscribe="todos$ as todos">
<ng-query-spinner *ngIf="todos.isLoading"></ng-query-spinner>
<p *ngIf="todos.isError">Error...</p>
<ul *ngIf="todos.isSuccess">
<li *ngFor="let todo of todos.data">
{{ todo.title }}
</li>
</ul>
</ng-container>
`,
})
export class TodosPageComponent {
todos$ = inject(TodosService).getTodos().result$;
}
Note that using the *subscribe
directive is optional. Subscriptions can be made using any method you choose.
Inject the UseInfiniteQuery
provider in your service. Using the hook is similar to the official hook, except the query function should return an observable
.
import { UseInfiniteQuery } from '@ngneat/query';
@Injectable({ providedIn: 'root' })
export class ProjectsService {
private useInfiniteQuery = inject(UseInfiniteQuery);
getProjects() {
return this.useInfiniteQuery(
['projects'],
({ pageParam = 0 }) => {
return getProjects(pageParam);
},
{
getNextPageParam(projects) {
return projects.nextId;
},
getPreviousPageParam(projects) {
return projects.previousId;
},
}
);
}
}
Checkout the complete example in our playground.
Use the UsePersistedQuery
provider when you want to use the keepPreviousData
feature. For example, to implement the pagination functionality:
import { inject, Injectable } from '@angular/core';
import {
UsePersistedQuery,
QueryClientService,
queryOptions,
} from '@ngneat/query';
import { firstValueFrom } from 'rxjs';
@Injectable({ providedIn: 'root' })
export class PaginationService {
private queryClient = inject(QueryClientService);
getProjects = inject(UsePersistedQuery)((queryKey: ['projects', number]) => {
return queryOptions({
queryKey,
queryFn: ({ queryKey }) => {
return fetchProjects(queryKey[1]);
},
});
});
prefetch(page: number) {
return this.queryClient.prefetchQuery(['projects', page], () =>
firstValueFrom(fetchProjects(page))
);
}
}
Checkout the complete example in our playground.
The official mutation
function can be a little verbose. Generally, you can use the following in-house simplified implementation.
import { HttpClient } from '@angular/common/http';
import { inject, Injectable } from '@angular/core';
import { QueryClientService } from '@ngneat/query';
@Injectable({ providedIn: 'root' })
export class TodosService {
private http = inject(HttpClient);
private queryClient = inject(QueryClientService);
addTodo({ title }: { title: string }) {
return this.http.post<{ success: boolean }>(`todos`, { title }).pipe(
tap((newTodo) => {
// Invalidate to refetch
this.queryClient.invalidateQueries(['todos']);
// Or update manually
this.queryClient.setQueryData<TodosResponse>(
['todos'],
addEntity('todos', newTodo)
);
})
);
}
}
And in the component:
import { QueryClientService, useMutationResult } from '@ngneat/query';
@Component({
template: `
<input #ref />
<button
(click)="addTodo({ title: ref.value })"
*subscribe="addTodoMutation.result$ as addTodoMutation"
>
Add todo {{ addTodoMutation.isLoading ? 'Loading' : '' }}
</button>
`,
})
export class TodosPageComponent {
private todosService = inject(TodosService);
addTodoMutation = useMutationResult();
addTodo({ title }) {
this.todosService
.addTodo({ title })
.pipe(this.addTodoMutation.track())
.subscribe();
}
}
You can use the original mutation
functionality if you prefer.
import { HttpClient } from '@angular/common/http';
import { inject, Injectable } from '@angular/core';
import { QueryClientService, UseMutation } from '@ngneat/query';
@Injectable({ providedIn: 'root' })
export class TodosService {
private http = inject(HttpClient);
private queryClient = inject(QueryClientService);
private useMutation = inject(UseMutation);
addTodo() {
return this.useMutation(({ title }: { title: string }) => {
return this.http.post<{ success: boolean }>(`todos`, { title }).pipe(
tap((newTodo) => {
// Invalidate to refetch
this.queryClient.invalidateQueries(['todos']);
// Or update manually
this.queryClient.setQueryData<TodosResponse>(
['todos'],
addEntity('todos', newTodo)
);
})
);
});
}
}
And in the component:
@Component({
template: `
<input #ref />
<button
(click)="addTodo({ title: ref.value })"
*subscribe="addTodoMutation.result$ as addTodoMutation"
>
Add todo {{ addTodoMutation.isLoading ? 'Loading' : '' }}
</button>
`,
})
export class TodosPageComponent {
private todosService = inject(TodosService);
addTodoMutation = this.todosService.addTodo();
addTodo({ title }) {
this.addTodoMutation$.mutate({ title }).then((res) => {
console.log(res.success);
});
}
}
You can provide the QUERY_CLIENT_OPTIONS
provider to set the global options of the query client instance:
import { provideQueryClientOptions } from '@ngneat/query';
bootstrapApplication(AppComponent, {
providers: [
provideQueryClientOptions({
defaultOptions: {
queries: {
staleTime: 3000,
},
},
}),
]
});
The filterSuccess
operator is a shortcut for filter((result) => result.isSuccess)
.
It's useful when you want to filter only successful results.
this.todosService.getTodos().result$.pipe(filterSuccess());
The filterError
operator is a shortcut for filter((result) => result.status === 'error')
.
It's useful when you want to filter only error results.
this.todosService.getTodos().result$.pipe(filterError());
The tapSuccess
operator is a shortcut for tap((result) => result.isSuccess && callback(result.data))
.
It's useful when you want to run a side effect only when the result is successful.
this.todosService.getTodos().result$.pipe(tapSuccess((data) => console.log(data)));
The tapError
operator is a shortcut for tap((result) => result.isError && callback(result.error))
.
It's useful when you want to run a side effect only when the result is successful.
this.todosService.getTodos().result$.pipe(tapError((error) => console.log(error)));
The mapResultData
operator like the name implies maps the data
property of the result
object.
Note: The data will only be mapped if the result is successful and otherwise just returned as is on any other state.
this.todosService.getTodos().pipe(
mapResultData((data) => {
return {
todos: data.todos.filter(predicate),
};
})
);
The intersectResults
operator is used to merge multiple queries into one.
It will return a new base query result that will merge the results of all the queries.
Note: The data will only be mapped if the result is successful and otherwise just returned as is on any other state.
const query = combineLatest({
todos: todos.result$,
posts: posts.result$,
}).pipe(
intersectResults(({ todos, posts }) => {
return { ... }
})
)
// --- or ---
const query = combineLatest([todos.result$, posts.result$]).pipe(
intersectResults(([todos, posts]) => {
return { ... }
})
)
Implementation of isFetching and isMutating.
import {
UseIsFetching,
UseIsMutating,
createSyncObserverResult
} from '@ngneat/query';
// How many queries are fetching?
const isFetching$ = inject(UseIsFetching)();
// How many queries matching the posts prefix are fetching?
const isFetchingPosts$ = inject(UseIsFetching)(['posts']);
// How many mutations are fetching?
const isMutating$ = inject(UseIsMutating)();
// How many mutations matching the posts prefix are fetching?
const isMutatingPosts$ = inject(UseIsMutating)(['posts']);
// Create sync successful observer in case we want to work with one interface
of(createSyncObserverResult(data, options?))
You can use the constructor
version instead of inject
:
QueryService.use(...)
PersistedQueryService.use(...)
InfiniteQueryService.use(...)
MutationService.use(...)
Install the @ngneat/query-devtools
package. Lazy load and use it only in development
environment:
import { provideQueryDevTools } from '@ngneat/query';
bootstrapApplication(AppComponent, {
providers: [
provideQueryDevTools(),
],
});
On the Server:
import { provideQueryClient } from '@ngneat/query';
import { QueryClient, dehydrate } from '@tanstack/query-core';
import { renderApplication } from '@angular/platform-server';
async function handleRequest(req, res) {
const queryClient = new QueryClient();
let html = await renderApplication(AppComponent, {
providers: [provideQueryClient(queryClient)],
});
const queryState = JSON.stringify(dehydrate(queryClient));
html = html.replace(
'</body>',
`<script>window.__QUERY_STATE__ = ${queryState}</script></body>`
);
res.send(html);
queryClient.clear();
}
Client:
import { importProvidersFrom } from '@angular/core';
import { bootstrapApplication, BrowserModule } from '@angular/platform-browser';
import { provideQueryClient } from '@ngneat/query';
import { QueryClient, hydrate } from '@tanstack/query-core';
const queryClient = new QueryClient();
const dehydratedState = JSON.parse(window.__QUERY_STATE__);
hydrate(queryClient, dehydratedState);
bootstrapApplication(AppComponent, {
providers: [
importProvidersFrom(
BrowserModule.withServerTransition({ appId: 'server-app' })
),
provideQueryClient(queryClient),
],
});
Netanel Basal |
Thank goes to all these wonderful people who contributed β€οΈ