Skip to content

Commit d583ebe

Browse files
feat(pagination): Add pagination sample using mat-paginator
1 parent f8afdc0 commit d583ebe

8 files changed

+298
-83
lines changed

apps/demo/src/app/app.component.html

+1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
<a mat-list-item routerLink="/flight-search">withRedux</a>
66
<a mat-list-item routerLink="/flight-search-data-service-simple">withDataService (Simple)</a>
77
<a mat-list-item routerLink="/flight-search-data-service-dynamic">withDataService (Dynamic)</a>
8+
<a mat-list-item routerLink="/flight-search-with-pagination">withPagination</a>
89
<a mat-list-item routerLink="/flight-search-redux-connector">Redux Connector</a>
910
<a mat-list-item routerLink="/todo-storage-sync">withStorageSync</a>
1011
</mat-nav-list>

apps/demo/src/app/app.routes.ts

-9
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,4 @@
11
import { Route } from '@angular/router';
2-
import { TodoComponent } from './todo/todo.component';
3-
import { FlightSearchComponent } from './flight-search/flight-search.component';
4-
import { FlightSearchSimpleComponent } from './flight-search-data-service-simple/flight-search-simple.component';
5-
import { FlightEditSimpleComponent } from './flight-search-data-service-simple/flight-edit-simple.component';
6-
import { FlightSearchDynamicComponent } from './flight-search-data-service-dynamic/flight-search.component';
7-
import { FlightEditDynamicComponent } from './flight-search-data-service-dynamic/flight-edit.component';
8-
import { TodoStorageSyncComponent } from './todo-storage-sync/todo-storage-sync.component';
9-
import { FlightSearchReducConnectorComponent } from './flight-search-redux-connector/flight-search.component';
10-
import { provideFlightStore } from './flight-search-redux-connector/+state/redux';
112

123
export const appRoutes: Route[] = [
134
{
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
<h2 class="title">Flight Search (Pagination)</h2>
2+
3+
4+
<form (ngSubmit)="search()">
5+
<div>
6+
<mat-form-field>
7+
<mat-label>Name</mat-label>
8+
<input [(ngModel)]="searchParams.from" name="from" matInput />
9+
</mat-form-field>
10+
</div>
11+
12+
<div>
13+
<mat-form-field>
14+
<mat-label>Name</mat-label>
15+
<input [(ngModel)]="searchParams.to" name="to" matInput />
16+
</mat-form-field>
17+
</div>
18+
19+
<button mat-raised-button>Search</button>
20+
</form>
21+
22+
<mat-table [dataSource]="dataSource">
23+
<!-- From Column -->
24+
<ng-container matColumnDef="from">
25+
<mat-header-cell *matHeaderCellDef>From</mat-header-cell>
26+
<mat-cell *matCellDef="let element">{{ element.from }}</mat-cell>
27+
</ng-container>
28+
29+
<!-- To Column -->
30+
<ng-container matColumnDef="to">
31+
<mat-header-cell *matHeaderCellDef>To</mat-header-cell>
32+
<mat-cell *matCellDef="let element">{{ element.to }}</mat-cell>
33+
</ng-container>
34+
35+
<!-- Date Column -->
36+
<ng-container matColumnDef="date">
37+
<mat-header-cell mat-header-cell *matHeaderCellDef>Date</mat-header-cell>
38+
<mat-cell mat-cell *matCellDef="let element">{{
39+
element.date | date
40+
}}</mat-cell>
41+
</ng-container>
42+
43+
<mat-header-row *matHeaderRowDef="displayedColumns"></mat-header-row>
44+
<mat-row
45+
*matRowDef="let row; columns: displayedColumns"
46+
(click)="selection.toggle(row)"
47+
></mat-row>
48+
</mat-table>
49+
<mat-paginator [length]="flightStore.flightTotalCount()"
50+
[pageSize]="flightStore.flightPageSize()"
51+
[pageIndex]="flightStore.flightCurrentPage()"
52+
[showFirstLastButtons]="true"
53+
[pageSizeOptions]="[5, 10, 25]"
54+
(page)="handlePageEvent($event)"
55+
aria-label="Select page">
56+
</mat-paginator>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
import { Component, effect, inject } from '@angular/core';
2+
import { MatTableDataSource, MatTableModule } from '@angular/material/table';
3+
import { DatePipe } from '@angular/common';
4+
import { SelectionModel } from '@angular/cdk/collections';
5+
import { MatInputModule } from '@angular/material/input';
6+
import { FormsModule } from '@angular/forms';
7+
import { MatButtonModule } from '@angular/material/button';
8+
import { FlightBookingStore } from './flight-store';
9+
import { Flight } from '../shared/flight';
10+
import { MatPaginatorModule, PageEvent } from '@angular/material/paginator';
11+
12+
@Component({
13+
selector: 'demo-flight-search-with-pagination',
14+
templateUrl: 'flight-search-with-pagination.component.html',
15+
standalone: true,
16+
imports: [
17+
MatTableModule,
18+
MatPaginatorModule,
19+
DatePipe,
20+
MatInputModule,
21+
FormsModule,
22+
MatButtonModule,
23+
],
24+
providers: [FlightBookingStore]
25+
})
26+
export class FlightSearchWithPaginationComponent {
27+
searchParams: { from: string; to: string } = { from: 'Wien', to: '' };
28+
flightStore = inject(FlightBookingStore);
29+
30+
displayedColumns: string[] = ['from', 'to', 'date'];
31+
dataSource = new MatTableDataSource<Flight>([]);
32+
selection = new SelectionModel<Flight>(true, []);
33+
34+
constructor() {
35+
effect(() => {
36+
this.dataSource.data = this.flightStore.selectedPageFlightEntities();
37+
});
38+
this.flightStore.loadFlightEntities();
39+
}
40+
41+
search() {
42+
this.flightStore.updateFlightFilter(
43+
this.searchParams
44+
);
45+
this.flightStore.loadFlightEntities();
46+
}
47+
48+
handlePageEvent(e: PageEvent) {
49+
this.flightStore.setFlightPageSize(e.pageSize);
50+
this.flightStore.gotoFlightPage(e.pageIndex);
51+
}
52+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import { FlightService } from '../shared/flight.service';
2+
3+
import { signalStore, type } from '@ngrx/signals';
4+
5+
import { withEntities } from '@ngrx/signals/entities';
6+
import { withCallState, withDataService, withPagination } from 'ngrx-toolkit';
7+
import { Flight } from '../shared/flight';
8+
9+
export const FlightBookingStore = signalStore(
10+
withCallState({
11+
collection: 'flight',
12+
}),
13+
withEntities({
14+
entity: type<Flight>(),
15+
collection: 'flight',
16+
}),
17+
withDataService({
18+
dataServiceType: FlightService,
19+
filter: { from: 'Wien', to: '' },
20+
collection: 'flight',
21+
}),
22+
withPagination({
23+
entity: type<Flight>(),
24+
collection: 'flight',
25+
})
26+
);

apps/demo/src/app/lazy-routes.ts

+5
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import { FlightEditDynamicComponent } from './flight-search-data-service-dynamic
88
import { TodoStorageSyncComponent } from './todo-storage-sync/todo-storage-sync.component';
99
import { provideFlightStore } from './flight-search-redux-connector/+state/redux';
1010
import { FlightSearchReducConnectorComponent } from './flight-search-redux-connector/flight-search.component';
11+
import { FlightSearchWithPaginationComponent } from './flight-search-with-pagination/flight-search-with-pagination.component';
1112

1213
export const lazyRoutes: Route[] = [
1314
{ path: 'todo', component: TodoComponent },
@@ -21,6 +22,10 @@ export const lazyRoutes: Route[] = [
2122
path: 'flight-search-data-service-dynamic',
2223
component: FlightSearchDynamicComponent,
2324
},
25+
{
26+
path: 'flight-search-with-pagination',
27+
component: FlightSearchWithPaginationComponent
28+
},
2429
{ path: 'flight-edit-dynamic/:id', component: FlightEditDynamicComponent },
2530
{ path: 'todo-storage-sync', component: TodoStorageSyncComponent },
2631
{

libs/ngrx-toolkit/src/lib/with-pagination.spec.ts

+14-20
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,10 @@
11
import { patchState, signalStore, type } from '@ngrx/signals';
2-
import { createPageArray, gotoPage, setPageSize, withPagination } from './with-pagination';
2+
import {
3+
createPageArray,
4+
gotoPage,
5+
setPageSize,
6+
withPagination,
7+
} from './with-pagination';
38
import { setAllEntities, withEntities } from '@ngrx/signals/entities';
49

510
type Book = { id: number; title: string; author: string };
@@ -21,13 +26,13 @@ describe('withPagination', () => {
2126
const store = new Store();
2227

2328
patchState(store, setAllEntities(generateBooks(55)));
24-
expect(store.currentPage()).toBe(1);
29+
expect(store.currentPage()).toBe(0);
2530
expect(store.pageCount()).toBe(6);
2631
}),
2732
it('should use and update a pagination with collection', () => {
2833
const Store = signalStore(
2934
withEntities({ entity: type<Book>(), collection: 'books' }),
30-
withPagination({ collection: 'books' })
35+
withPagination({ entity: type<Book>(), collection: 'books' })
3136
);
3237

3338
const store = new Store();
@@ -37,41 +42,30 @@ describe('withPagination', () => {
3742
setAllEntities(generateBooks(55), { collection: 'books' })
3843
);
3944

40-
patchState(store, gotoPage(6, { collection: 'books' }));
41-
expect(store.booksCurrentPage()).toBe(6);
45+
patchState(store, gotoPage(5, { collection: 'books' }));
46+
expect(store.booksCurrentPage()).toBe(5);
4247
expect(store.selectedPageBooksEntities().length).toBe(5);
4348
expect(store.booksPageCount()).toBe(6);
4449
}),
4550
it('should react on enitiy changes', () => {
4651
const Store = signalStore(
47-
withEntities({ entity: type<Book>()}),
52+
withEntities({ entity: type<Book>() }),
4853
withPagination()
4954
);
5055

5156
const store = new Store();
5257

53-
patchState(
54-
store,
55-
setAllEntities(generateBooks(100))
56-
);
58+
patchState(store, setAllEntities(generateBooks(100)));
5759

5860
expect(store.pageCount()).toBe(10);
5961

60-
patchState(
61-
store,
62-
setAllEntities(generateBooks(20))
63-
);
62+
patchState(store, setAllEntities(generateBooks(20)));
6463

6564
expect(store.pageCount()).toBe(2);
6665

67-
68-
patchState(
69-
store,
70-
setPageSize(5)
71-
);
66+
patchState(store, setPageSize(5));
7267

7368
expect(store.pageCount()).toBe(4);
74-
7569
}),
7670
describe('internal pageNavigationArray', () => {
7771
it('should return an array of page numbers', () => {

0 commit comments

Comments
 (0)