High Performance with RadGridView and Virtual Mode including Filtering, Sorting and Grouping
Date Posted | Product | Author |
---|---|---|
Q1 2013 | RadGridView | Georgi Georgiev |
Problem
You want to bind RadGridView to over a million items. In such cases it is recommended to use Virtual Mode, however this removes the Filtering, Sorting and Grouping capabilities of RadGridView since RadGridView has no direct access to its DataSource.
As of Q1 2016 (version 2016.1.112) Telerik UI for WinForms suite offers RadVirtualGrid control. It is a grid component developed on top of Telerik Presentation Framework which provides a convenient way to implement your own data management operations and optimizes the performance when interacting with large amounts of data.
Solution
We can create a special Data Layer which will manage our DataSource and only provide the needed data to the grid. This layer will also allow RadGridView to be responsive while the operations are being executed. Below you can see the speed at which the operations will be performed when *RadGridView *is bound to 2 million items:
NOTE: The projects below requires NET 4 since it uses some NET 4 features, such as Parallel LINQ and UI for Winforms version *Q1 2014 or above** due to the paging functionality and some API changes.*
First of all we need to create our Data Layer. It will be called ItemSource. It needs a few essential properties:
- DataSource – This will be the property which will keep the collection with all items which were initially passed to it.
- View – This will be the collection of items which will be represented by RadGridView. This collection will always be sorted or filtered according to the descriptors of RadGridView.
- BoundProperties – This collection will keep all the properties of the bound objects, so we can get or set the value when needed.
- Count – Will return the amount of items in the View.
- Indexer – Will return a value from the View by index.
- CurrentOperation – This will allow us to track what operation is executing at any given moment
The implementation of the above properties can be seen below:
It will also need to have methods which will allow us to get the values from the items in the DataSource, namely GetValue and SetValue:
Now, our ItemSource has to be able to perform the very important Sorting and Filtering operations. For the sorting we will use PLINQ. The queries will be executed over the view. Here is how our sorting method will look like:
For the filtering we will use the ExpressionParser which RadGridView uses internally to check whether an object passes certain filter:
The grouping will still be performed by RadGridView, due to its nature. That is why it is recommended to use the Paging functionality along with the PagingBeforeGrouping property set to true. Now we can filter or sort our data, however these methods are protected, moreover, even if we execute them they will be synchronous, which will block the UI thread. We will prevent that by implementing a queuing mechanism which will execute these methods and provide events for when the operations are complete. For this we will use a BackgroundWorker:
The WorkerCompleted, ProgressChanged and DoWork events will be used to respectively:
Start another cycle which will execute tasks if there are such and report that there are no more tasks to execute
Report that a task is finished
Dequeue tasks and execute them in order
We will use one class and one enumeration that in the process of execution of pending tasks:
And these are the event handlers of the background worker, described above:
After we have a mechanism which can execute tasks in order we need a way to start it up. We will expose one method which be responsible for this:
Now, we need to integrate this layer into RadGridView. To do this we will inherit from RadGridView and add a few properties:
ItemsSource – This will keep a reference to the ItemSource which will manage the data
VirtualDataSource – It will set the DataSource of the ItemSource and initialize the Rows and Columns according to the properties and the amount of items
ShowLoadingOverlay – Determines whether a waiting bar will be displayed in the middle of RadGridView while the operations are being performed in the background
AutomaticallyRetreiveCellValues – As we know, in VirtualMode we need to provide the values of the cells by ourselves. In this case this is not a trivial task, since many factors such as groups and pages should be taken into consideration, that is why we will provide a built-in way to do this.
AutomaticallyPushCellValues – The same as the previous property, however this time we will set the value of the data bound items.
LoadingOverlay – Provides reference to the waiting bar being displayed in the middle of RadGridView
The Initialize method is very essential as it manages the rows and columns of RadGridView. It should be executed either when the DataSource changes or when BindingContext changes:
Now we just need to replace some of the classes of RadGridView. First we start with the RadGridViewElement:
In the VirtualRadGridViewElement we will need to create a custom MasterTemplate:
And in the VirtualMasterGridViewTemplate, a custom ListSource:
In the ListSource we need to set the DataBoundItem to our rows manually, in order to be able to group by these rows and create a custom RadCollectionView:
The DataView actually creates the ItemSource, the GroupBuilder and the Indexer which simply keeps reference to the rows:
This is what the VirtualIndex requires to work properly:
And this is what our GroupBuilder requires:
NOTE: In order to see the full implementation of the classes you should download the source code below.
In order to use the grid it is enough to set the VirtualDataSource property to some collection. You can also use the OperationStarted and OperationCompleted events at will.
A complete solution in C# and VB.NET can be found here.