Paging.NET
5.0.1-pre
dotnet add package Paging.NET --version 5.0.1-pre
NuGet\Install-Package Paging.NET -Version 5.0.1-pre
<PackageReference Include="Paging.NET" Version="5.0.1-pre" />
<PackageVersion Include="Paging.NET" Version="5.0.1-pre" />
<PackageReference Include="Paging.NET" />
paket add Paging.NET --version 5.0.1-pre
#r "nuget: Paging.NET, 5.0.1-pre"
#:package Paging.NET@5.0.1-pre
#addin nuget:?package=Paging.NET&version=5.0.1-pre&prerelease
#tool nuget:?package=Paging.NET&version=5.0.1-pre&prerelease
Paging.NET
Paging.NET is a lightweight and flexible library for server-side paging and incremental data loading. Large datasets can be handled more efficiently by retrieving items in smaller, predictable chunks, making data access easier to manage. The library set consists of the following NuGet packages:
Paging.NET: Core library containing the main paging models such asPagingInfoandPaginationSet.Paging.Queryable.NET: Extension library providingIQueryablesupport for paging, sorting, and filtering.Paging.MAUI: Add-on for .NET MAUI moble apps for implementing incremental loading and infinite scrolling scenarios.
Download and Install Paging.NET
This library is available on nuget.org:
| Package | Version | Downlods |
|---|---|---|
| Paging.NET | ||
| Paging.Queryable.NET | ||
| Paging.MAUI |
Getting Started
Paging or pagination is the process of splitting a collection into smaller subsets of items in order to improve performance and reduce the amount of data transferred at once. In practice, paging is usually combined with sorting, searching, and filtering.
In Paging.NET, the client sends a paging request as a PagingInfo, and the service responds with a PaginationSet<T>.
How to Use Paging.NET
Core Types
Paging.NET is built around two core models:
PagingInfois the paging request model. It specifies which page should be loaded, how many items should be returned, and which sorting, search, or filtering options should be applied.
| Property | Description |
|---|---|
FirstPageIndex |
The first valid page index for the request. Allowed values are 0 and 1. |
CurrentPage |
The currently selected page.<br/> The default value is PagingInfo.DefaultFirstPageIndex, which means it initially matches FirstPageIndex. |
ItemsPerPage |
Number of items returned per page. <br/>null disables paging and returns all items, 0 returns totals only, and positive values enable normal paging. The static PagingInfo.DefaultItemsPerPage defaults to null. |
SortBy |
Comma-separated sort expression such as "Name Asc" or "Year Desc, Name Asc". |
Sorting |
Dictionary-based sort definition as an alternative to SortBy string. |
Reverse |
Reverses the final sort order. |
Search |
Free-text search value that can be applied by the target data source. |
Filter |
Property-based filter values that can be applied by the target data source. |
PaginationSet<T>is the paged response model. It contains the items of the current page together with metadata describing the complete result set.
| Property | Description |
|---|---|
FirstPageIndex |
The first page index used for the request and response. |
CurrentPage |
The current page number of the returned result. This value is relative to FirstPageIndex. |
TotalPages |
Total number of pages available for the current filter and search criteria. |
TotalCount |
Total number of items matching the current filter and search criteria. |
TotalCountUnfiltered |
Total number of items before filter or search is applied. |
Items |
The items contained in the current page. |
Basic Example
The following example shows a simple request using the core models:
var pagingInfo = new PagingInfo
{
FirstPageIndex = PagingInfo.DefaultFirstPageIndex,
CurrentPage = 1,
ItemsPerPage = 10,
SortBy = "Name Asc",
Search = "Model Desc"
};
Paging Defaults and Modes
You can configure the global default page size once for your process:
Disable Paging By Default
PagingInfo.DefaultItemsPerPage = null; // null means return all items - no paging is used
Choose the First Page Index (0- or 1-based)
PagingInfo.DefaultFirstPageIndex is a global library default. Allowed values are 0 and 1.
If you want zero-based paging by default, set PagingInfo.DefaultFirstPageIndex = 0. You can still override
FirstPageIndex per request.
PagingInfo.DefaultFirstPageIndex = 0;
Or set it directly on a request:
var pagingInfo = new PagingInfo
{
FirstPageIndex = 0,
CurrentPage = 0,
ItemsPerPage = 10
};
FirstPageIndex is part of the PagingInfo contract and is serialized automatically when it differs from
PagingInfo.DefaultFirstPageIndex.
ItemsPerPage has three explicit modes:
null: disable paging and return all matching items0: return totals only and zero items> 0: return the requested page with the requested number of items
CurrentPage is interpreted relative to FirstPageIndex, which may be 0 or 1. FirstPageIndex is part of the
request and response contract and is serialized over JSON and query strings when it differs from PagingInfo.DefaultFirstPageIndex.
Service Example
A service can then use this request to return a page of Car items. The following example is intentionally kept as
pseudo code to illustrate the general flow:
public PaginationSet<Car> GetCars(PagingInfo pagingInfo)
{
var query = LoadCars();
// Apply search, filtering, sorting, grouping of data
// Apply Skip(...).Take(...) for the requested page
return new PaginationSet<Car>(pagingInfo, pageItems, totalCount, totalCount);
}
If you are working with an IQueryable<T>, the section below shows how Paging.Queryable.NET can perform all necessary steps
to create a PaginationSet<T> through the ToPaginationSet(...) extension method.
How to Use Paging.Queryable.NET
Paging.Queryable.NET provides extensions for applying paging directly to an IQueryable<T>.
This is useful for backend code working with Entity Framework or any other LINQ provider.
The main entry point is ToPaginationSet(...). It applies the PagingInfo request to an
IQueryable<TEntity> and returns a PaginationSet<TEntity>:
IQueryable<Car> queryable = dbContext.Cars;
var pagingInfo = new PagingInfo
{
CurrentPage = 1,
ItemsPerPage = 10,
SortBy = "Year Desc, Name Asc",
Filter =
{
{ "Year", 2024 }
}
};
var paginationSet = queryable.ToPaginationSet(pagingInfo);
In this example:
Filterapplies property-based constraints before counting and paging.SortBydefines the ordering before paging is applied.- The result is returned as a
PaginationSet<Car>. ItemsPerPage = 0returns counts without materializing page items.ItemsPerPage = nullskipsSkip(...).Take(...)and returns all matching items.
Without further configuration, ToPaginationSet(pagingInfo) behaves permissively:
- Every entity property is sortable and filterable by its (dotted) property path, e.g.
"Name"or"Owner.Name". Property names are matched case-insensitively. - Sort or filter names which cannot be resolved are silently skipped.
PagingInfo.Searchis ignored (a search predicate must be configured viaPagingOptions).- No default sort order is applied. Configure
PagingOptions.DefaultSortfor deterministic paging — without a stable sort order, the database may return rows in arbitrary order between page requests.
Configuring Sorting, Filtering and Mapping with PagingOptions
PagingOptions<TEntity> controls which properties are exposed for sorting and filtering, how external
(client-facing) names map to entity properties, computed sort expressions, custom filter predicates,
free-text search and the default sort order. PagingOptions<TEntity, TDto> additionally maps the queried
entities to DTOs.
var pagingOptions = new PagingOptions<Car, CarDto>(o =>
{
// Each property declares its capabilities explicitly;
// the external name defaults to the property path
o.Property(c => c.Name).Sortable().Filterable();
// Map an external name to an entity property (nested paths supported)
o.Property(c => c.Model).HasName("Brand").Sortable().Filterable();
o.Property(c => c.Owner.Name).HasName("Owner").Sortable(); // sort-only
// Computed sort key: the client sorts by "Age", which has no database column
o.Property("Age").Sortable(c => DateTime.UtcNow.Year - c.Year);
// Custom filter predicate for arbitrary filter values
o.Property("Electric").Filterable(value => value is bool isElectric
? (Expression<Func<Car, bool>>)(c => c.IsElectric == isElectric)
: null);
// Stable default sort: primary order when no sorting is requested
// AND tie-breaker appended after any requested sort order
o.DefaultSort(c => c.Id, SortOrder.Desc);
// Free-text search predicate; receives PagingInfo.Search
o.Search(s => c => c.Name.ToLower().Contains(s.ToLower()));
// Map entities to DTOs (mandatory on PagingOptions<TEntity, TDto>)
o.Map(cars => cars.Select(car => new CarDto
{
Id = car.Id,
Name = car.Name,
Model = car.Model,
Price = car.Price,
Year = car.Year
}));
});
var paginationSet = queryable.ToPaginationSet(pagingInfo, pagingOptions);
PagingOptions is frozen on first use and can safely be cached and reused across queries and threads.
Class-Based PagingOptions with Dependency Injection
Instead of configuring inline, PagingOptions can be subclassed and registered in dependency injection.
Constructor injection is the recommended way to feed runtime values (such as the current date/time)
into computed sort expressions. Use the factory overload of Sortable to evaluate the expression
freshly on every query:
public class CarPagingOptions : PagingOptions<Car, CarDto>
{
public CarPagingOptions(IDateTime dateTime, ICarMapper mapper)
{
this.Property(c => c.Name).Sortable().Filterable();
this.Property(c => c.Model).HasName("Brand").Sortable().Filterable();
this.Property("Age").Sortable(() =>
{
var now = dateTime.UtcNow;
return (Expression<Func<Car, int>>)(c => now.Year - c.Year);
});
this.DefaultSort(c => c.Id, SortOrder.Desc);
this.Map(mapper.MapCarsToCarDtos);
}
}
// Startup:
services.AddScoped<CarPagingOptions>();
// Controller:
var paginationSet = dbContext.Cars.ToPaginationSet(pagingInfo, this.carPagingOptions);
All sort and filter expressions remain LINQ expression trees, so Entity Framework translates them
to SQL — including computed sort keys (e.g. CASE expressions) and captured runtime values
(which become SQL parameters).
Handling of Unknown Property Names
When PagingOptions are used, unknown sort/filter property names throw a PagingException by default.
Web APIs typically translate this exception into an HTTP 400 (Bad Request) response.
The behavior is configurable, globally or separately for sorting and filtering:
o.UnknownProperties(UnknownPropertyHandling.Ignore); // sets both
o.UnknownSortProperties(UnknownPropertyHandling.Throw);
o.UnknownFilterProperties(UnknownPropertyHandling.Allow);
| Handling | Behavior |
|---|---|
Throw |
Throws PagingException (with PropertyName) for unknown property names. Default. |
Ignore |
Silently skips unknown property names. |
Allow |
Resolves unknown names as (dotted) entity property paths; unresolvable names are skipped. |
Filter Value Semantics
The Filter dictionary of PagingInfo supports the following value types:
| Filter value | Behavior |
|---|---|
{ "Name", "Tesla" } |
Case-insensitive contains search. |
{ "Price", ">=5000" } |
Comparison using a leading operator (>, >=, <, <=, =, ==). |
{ "Year", 2024 } |
Equality for numeric, bool and DateTime values. |
{ "Date", new Dictionary<string, object> { { ">", "2024-01-01" } } } |
Range comparisons with operator/value pairs. |
{ "Id", new[] { 1, 2, 3 } } |
IN-filter; matches any of the provided values. |
Invalid filter values (e.g. a non-numeric string compared to a numeric property) are leniently skipped.
Migration from 4.x to 5.0
| 4.x | 5.0 |
|---|---|
pagingInfo.CreatePaginationSet(queryable) |
queryable.ToPaginationSet(pagingInfo) |
pagingInfo.CreatePaginationSet(queryable, map, predicate) |
queryable.ToPaginationSet(pagingInfo, pagingOptions) with Map(...)/Search(...) configured in PagingOptions<TEntity, TDto> |
queryable.ApplyFilter(filter) / queryable.OrderBy(sorting, reverse) |
Removed; handled internally by ToPaginationSet. |
queryable.OrderByDefault() (implicit OrderBy("0")) |
Removed; configure PagingOptions.DefaultSort for deterministic paging. |
Dependency on System.Linq.Dynamic.Core |
Removed; Paging.Queryable.NET is dependency-free. |
How to Use Paging.MAUI
Paging.MAUI provides helpers for incremental loading and infinite scrolling in .NET MAUI apps.
The central type is InfiniteScrollCollection<T>. It is typically used together with a PagingInfo instance that keeps
track of the next page to load.
The following example is based on the MAUI sample app:
private readonly PagingInfo pagingInfo = new PagingInfo
{
CurrentPage = 1,
ItemsPerPage = 30,
};
private PaginationSet<Car>? lastPaginationSet;
public InfiniteScrollCollection<CarDto> Cars { get; } = new InfiniteScrollCollection<CarDto>();
public async Task InitializeAsync(ICarService carService)
{
this.Cars.OnCanLoadMore = () => !this.lastPaginationSet.StopScroll(this.pagingInfo);
this.Cars.OnLoadMore = async () =>
{
var paginationSet = await carService.GetCarsAsync(this.pagingInfo);
this.lastPaginationSet = paginationSet;
this.pagingInfo.CurrentPage++;
return paginationSet.Items.Select(car => new CarDto
{
Id = car.Id,
Name = car.Name,
Model = car.Model,
Price = car.Price,
Year = car.Year
});
};
await this.Cars.LoadMoreAsync();
}
This pattern assumes normal paging with ItemsPerPage > 0. For totals-only or unpaged requests, StopScroll(...)
returns true immediately.
In XAML, InfiniteScrollBehavior can be attached to a CollectionView
(xmlns paging referring to clr-namespace:Paging.MAUI;assembly=Paging.MAUI):
<CollectionView ItemsSource="{Binding Cars}">
<CollectionView.Behaviors>
<paging:InfiniteScrollBehavior
ItemsSource="{Binding Cars}"
IsLoadingMore="{Binding IsLoadingMore}"
RemainingItemsThreshold="5"/>
</CollectionView.Behaviors>
</CollectionView>
The behavior uses CollectionView's native RemainingItemsThresholdReached mechanism:
when the user scrolls close to the end of the list (RemainingItemsThreshold items remaining,
default 5), the next page is loaded automatically as long as OnCanLoadMore returns true.
The threshold set on the behavior overwrites any RemainingItemsThreshold set directly
on the CollectionView.
If you prefer commands over the behavior, CollectionView's built-in incremental loading
can be used directly with InfiniteScrollCollection — at the cost of guard logic in every
view model (RemainingItemsThresholdReached fires repeatedly while scrolling):
public IAsyncRelayCommand LoadMoreCommand => this.loadMoreCommand ??= new AsyncRelayCommand(async () =>
{
if (this.Cars.IsLoadingMore || !this.Cars.CanLoadMore)
{
return;
}
await this.Cars.LoadMoreAsync();
});
<CollectionView ItemsSource="{Binding Cars}"
RemainingItemsThreshold="5"
RemainingItemsThresholdReachedCommand="{Binding LoadMoreCommand}"/>
Legacy ListView support
The former ListView-based InfiniteScrollBehavior is still available in the namespace
Paging.MAUI.Compat (xmlns clr-namespace:Paging.MAUI.Compat;assembly=Paging.MAUI).
ListView is deprecated in .NET MAUI; this behavior will be removed in a future release.
Contribution
If you find a bug or want to propose a new feature, feel free to create a new issue here. Please use the predefined issue templates when submitting a new issue.
Thank You
Your contribution is valuable! Open source software isn’t just something you can pick up for free — it represents the hard work and dedication of many people who often not even know each other. We sincerely appreciate the time, effort, and dedication shown by everyone who helps keep this plugin going forward.
Links
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. net10.0 was computed. net10.0-android was computed. net10.0-browser was computed. net10.0-ios was computed. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. |
| .NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
| .NET Standard | netstandard2.0 is compatible. netstandard2.1 is compatible. |
| .NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
| MonoAndroid | monoandroid was computed. |
| MonoMac | monomac was computed. |
| MonoTouch | monotouch was computed. |
| Tizen | tizen40 was computed. tizen60 was computed. |
| Xamarin.iOS | xamarinios was computed. |
| Xamarin.Mac | xamarinmac was computed. |
| Xamarin.TVOS | xamarintvos was computed. |
| Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- System.Text.Json (>= 8.0.5)
-
.NETStandard 2.1
- System.Text.Json (>= 8.0.5)
NuGet packages (3)
Showing the top 3 NuGet packages that depend on Paging.NET:
| Package | Downloads |
|---|---|
|
Paging.Queryable.NET
Paging.Queryable is a lightweight library for seamless, incremental server-side data loading. |
|
|
Paging.Forms.NET
Paging.NET is a basic toolkit which provides incremental server-side data loads. |
|
|
Paging.MAUI
Paging.MAUI is a lightweight library for seamless, incremental server-side data loading. |
GitHub repositories
This package is not used by any popular GitHub repositories.
| Version | Downloads | Last Updated |
|---|---|---|
| 5.0.1-pre | 29 | 6/12/2026 |
| 5.0.0-pre | 45 | 6/12/2026 |
| 4.1.9 | 94 | 6/11/2026 |
| 4.1.6-pre | 93 | 6/11/2026 |
| 4.1.4-pre | 150 | 6/3/2026 |
| 4.1.3-pre | 130 | 6/3/2026 |
| 4.1.1-pre | 152 | 5/30/2026 |
| 4.0.6 | 166 | 5/29/2026 |
| 4.0.3-pre | 163 | 5/28/2026 |
| 4.0.1-pre | 264 | 4/15/2026 |
| 3.2.16-pre | 154 | 5/28/2026 |
| 3.2.15-pr.3533863989 | 61 | 5/28/2026 |
| 3.2.14-pre | 156 | 5/28/2026 |
| 3.2.13-pre | 161 | 4/15/2026 |
| 3.2.12-pr.3533863989 | 62 | 4/15/2026 |
| 3.2.11-pre | 150 | 4/15/2026 |
| 3.2.10-pre | 159 | 4/15/2026 |
| 3.2.9-pr.3533793184 | 71 | 4/15/2026 |
| 3.2.8-pre | 149 | 4/15/2026 |
| 3.2.7-pr.3533780164 | 60 | 4/15/2026 |
5.0
- Paging.Queryable: New entry point queryable.ToPaginationSet(pagingInfo, pagingOptions)
replaces pagingInfo.CreatePaginationSet(...).
- New PagingOptions<TEntity> and PagingOptions<TEntity, TDto>: sort/filter property allowlists,
external-to-internal property name mapping, computed sort expressions, custom filter predicates,
free-text search predicate, default sort order with tie-breaker, and DTO mapping.
- New UnknownPropertyHandling (Throw/Ignore/Allow) for unknown sort/filter property names.
- Breaking change: CreatePaginationSet, ApplyFilter, OrderBy, OrderByDefault and QueryExtensions
are removed from Paging.Queryable.
- Breaking change: The implicit OrderBy("0") default sort is removed;
configure PagingOptions.DefaultSort for deterministic paging.
- Removed dependency on System.Linq.Dynamic.Core.
- Paging.MAUI: New CollectionView-based InfiniteScrollBehavior using the native
RemainingItemsThresholdReached mechanism, with bindable RemainingItemsThreshold property.
- Breaking change: The ListView-based InfiniteScrollBehavior moved to namespace Paging.MAUI.Compat
and will be removed in a future release.
4.1
- Maintenance updates.
4.0
- Redesign paging semantics with explicit `ItemsPerPage` modes.
- Add `DefaultFirstPageIndex` to define 0-based or 1-based paging.
- Add `DefaultItemsPerPage` as the configurable global default page size.
3.2
- Update target framework to net9.0 and net10.0.
- Replace Newtonsoft.Json with System.Text.Json.ß
3.1
- Add support for nullable reference types.
3.0
- Drop support for Xamarin and .NET Framework.
- Add support for .NET MAUI.
2.2
- Maintenance updates.
2.0
- Add support for .NET standard.
1.0
- Initial release.