Tutorial 27: Creating a Customized Sorting User Interface

 

Scott Mitchell

May 2007

Summary: This is the Visual Basic tutorial. (Switch to the Visual C# tutorial.) When displaying a long list of sorted data, it can be very helpful to group related data by introducing separator rows. In this tutorial, we'll see how to create such a sorting user interface. (15 printed pages)

Download the code for this sample.

Contents of Tutorial 27 (Visual Basic)

Introduction
Step 1: Creating a Standard, Sortable GridView
Step 2: Exploring Techniques for Adding the Separator Rows
Adding Rows to the Data Bound to the GridView
Manipulating the GridView's Control Collection After It Has Been Data-Bound
Step 3: Adding the Separator Rows to the GridView's Control Hierarchy
Conclusion

Introduction

When displaying a long list of sorted data where there are only a handful of different values in the sorted column, an end user might find it hard to discern where, exactly, the difference boundaries occur. For example, there are 81 products in the database, but only nine different category choices (eight unique categories plus the NULL option). Consider the case of a user who is interested in examining the products that fall under the Seafood category. From a page that lists all of the products in a single GridView, the user might decide her best bet is to sort the results by category, which will group together all of the Seafood products together. After sorting by category, the user then must hunt through the list, looking for where the Seafood-grouped products start and end. Because the results are ordered alphabetically by the category name finding the Seafood products is not difficult, but it still requires closely scanning the list of items in the grid.

To help highlight the boundaries between sorted groups, many Web sites employ a user interface that adds a separator between such groups. Separators like the ones shown in Figure 1 enables a user to find a particular group and identify its boundaries more quickly, as well as ascertain what distinct groups exist in the data.

Figure 1. Each category group is identified clearly.

In this tutorial, we'll see how to create such a sorting user interface.

Step 1: Creating a Standard, Sortable GridView

Before we explore how to augment the GridView to provide the enhanced sorting interface, let's first create a standard, sortable GridView that lists the products. Start by opening the CustomSortingUI.aspx page in the PagingAndSorting folder. Add a GridView to the page, set its ID property to ProductList, and bind it to a new ObjectDataSource. Configure the ObjectDataSource to use the ProductsBLL class's GetProducts() method for selecting records.

Next, configure the GridView so that it only contains the ProductName, CategoryName, SupplierName, and UnitPrice BoundFields and the Discontinued CheckBoxField. Finally, configure the GridView to support sorting by checking the Enable Sorting check box in the GridView's smart tag (or by setting its AllowSorting property to true). After making these additions to the CustomSortingUI.aspx page, the declarative markup should look similar to the following:

<asp:GridView ID="ProductList" runat="server" AllowSorting="True"
    AutoGenerateColumns="False" DataKeyNames="ProductID"
    DataSourceID="ObjectDataSource1" EnableViewState="False">
    <Columns>
        <asp:BoundField DataField="ProductName" HeaderText="Product"
            SortExpression="ProductName" />
        <asp:BoundField DataField="CategoryName" HeaderText="Category"
            ReadOnly="True" SortExpression="CategoryName" />
        <asp:BoundField DataField="SupplierName" HeaderText="Supplier"
            ReadOnly="True" SortExpression="SupplierName" />
        <asp:BoundField DataField="UnitPrice" DataFormatString="{0:C}"
            HeaderText="Price" HtmlEncode="False" 
SortExpression="UnitPrice" />
        <asp:CheckBoxField DataField="Discontinued" HeaderText="Discontinued"
            SortExpression="Discontinued" />
    </Columns>
</asp:GridView>

<asp:ObjectDataSource ID="ObjectDataSource1" runat="server"
    OldValuesParameterFormatString="original_{0}" SelectMethod="GetProducts"
    TypeName="ProductsBLL"></asp:ObjectDataSource>

Take a moment to view our progress thus far in a browser. Figure 2 shows the sortable GridView when its data is sorted by category in alphabetical order.

Figure 2. The sortable GridView's data is ordered by category.

Step 2: Exploring Techniques for Adding the Separator Rows

With the generic, sortable GridView complete, all that remains is to be able to add the separator rows in the GridView before each unique sorted group. But how can such rows be injected into the GridView? Essentially, we must iterate through the GridView's rows, determine where the differences occur between the values in the sorted column, and then add the appropriate separator row. When thinking about this problem, it seems natural that the solution lies somewhere in the GridView's RowDataBound event handler. As we discussed in the Custom Formatting Based upon Data tutorial, this event handler is commonly used when applying row-level formatting based on the row's data. However, the RowDataBound event handler is not the solution here, as rows cannot be added to the GridView programmatically from this event handler. The GridView's Rows collection, in fact, is read-only.

To add additional rows to the GridView we have three choices:

  • Add these metadata separator rows to the actual data that is bound to the GridView.
  • After the GridView has been bound to the data, add additional TableRow instances to the GridView's control collection.
  • Create a custom server control that extends the GridView control and overrides those methods responsible for constructing the GridView's structure.

Creating a custom server control would be the best approach if this functionality was needed on many Web pages or across several Web sites. However, it would entail quite a bit of code and a thorough exploration into the depths of the GridView's internal workings. Therefore, we'll not consider that option for this tutorial.

The other two options—adding separator rows to the actual data being bound to the GridView and manipulating the GridView's control collection after it has been bound—attack the problem differently and merit a discussion.

Adding Rows to the Data Bound to the GridView

When the GridView is bound to a data source, it creates a GridViewRow for each record returned by the data source. Therefore, we can inject the needed separator rows by adding "separator records" to the data source before binding it to the GridView. Figure 3 illustrates this concept.

Figure 3. One technique involves adding separator rows to the data source.

I use the term "separator records" in quotation marks, because there is no special separator record; instead, we must somehow flag that a particular record in the data source serves as a separator, instead of a normal data row. For our examples, we're binding a ProductsDataTable instance to the GridView, which is composed of ProductRows. We might flag a record as a "separator row" by setting its CategoryID property to -1 (because such a value couldn't exist normally).

To utilize this technique, we'd need to perform the following steps:

  1. Programmatically retrieve the data to bind to the GridView (a ProductsDataTable instance).
  2. Sort the data based on the GridView's SortExpression and SortDirection properties.
  3. Iterate through the ProductsRows in the ProductsDataTable, looking for where the differences in the sorted column lie.
  4. At each group boundary, inject a "separator record" ProductsRow instance into the DataTable, one that has its CategoryID set to -1 (or whatever designation was decided upon to mark a record as a "separator record").
  5. After injecting the "separator rows," programmatically bind the data to the GridView.

In addition to these five steps, we'd also need to provide an event handler for the GridView's RowDataBound event. Here, we'd check each DataRow and determine if it was a "separator row," one whose CategoryID setting was -1. If so, we'd probably want to adjust its formatting or the text displayed in the cell(s).

Using this technique for injecting the sorting group boundaries requires a bit more work than outlined earlier, as you must also provide an event handler for the GridView's Sorting event and keep track of the SortExpression and SortDirection values.

Manipulating the GridView's Control Collection After It Has Been Data-Bound

Instead of messaging the data before binding it to the GridView, we can add the separator rows after the data has been bound to the GridView. The process of data binding builds up the GridView's control hierarchy, which in reality is just a Table instance composed of a collection of rows, each of which is composed of a collection of cells. Specifically, the GridView's control collection contains a Table object at its root, a GridViewRow (which is derived from the TableRow class) for each record in the DataSource bound to the GridView, and a TableCell object in each GridViewRow instance for each data field in the DataSource.

To add separator rows between each sorting group, we can manipulate this control hierarchy directly, after it has been created. We can be confident that the GridView's control hierarchy has been created for the last time by the time the page is being rendered. Therefore, this approach overrides the Page class's Render method, at which point the GridView's final control hierarchy is updated to include the needed separator rows. Figure 4 illustrates this process.

Figure 4. An alternate technique manipulates the GridView's control hierarchy.

For this tutorial, we'll use this latter approach to customize the sorting user experience.

Note: The code I'm presenting in this tutorial is based on the example provided in Teemu Keiski's blog entry, Playing a Bit with GridView "Sort Grouping".

Step 3: Adding the Separator Rows to the GridView's Control Hierarchy

Because we only want to add the separator rows to the GridView's control hierarchy after its control hierarchy has been created and created for the last time on that page visit, we want to perform this addition at the end of the page lifecycle, but before the actual GridView control hierarchy has been rendered into HTML. The latest possible point at which we can accomplish this is the Page class's Render event, which we can override in our code-behind class using the following method signature:

Protected Overrides Sub Render(ByVal writer As HtmlTextWriter)
   ' Add code to manipulate the GridView control hierarchy
   MyBase.Render(writer)
End Sub

When the Page class's original Render method is invoked—base.Render(writer)—each of the controls in the page will be rendered, generating the markup based on their control hierarchy. Therefore, it is imperative that we both call base.Render(writer), so that the page is rendered, and manipulate the GridView's control hierarchy prior to calling base.Render(writer), so that the separator rows have been added to the GridView's control hierarchy before it has been rendered.

To inject the sort group headers, we first must ensure that the user has requested that the data be sorted. By default, the GridView's contents are not sorted, and therefore we don't need to enter any group sorting headers.

Note: If you want the GridView to be sorted by a particular column when the page is first loaded, call the GridView's Sort method on the first page visit (but not on subsequent postbacks). To accomplish this, add this call in the Page_Load event handler within an if (!Page.IsPostBack) conditional. Refer back to the Paging and Sorting Report Data tutorial information, for more on the Sort method.

Assuming that the data has been sorted, our next task is to determine what column the data was sorted by and then to scan the rows looking for differences in that column's values. The following code ensures that the data has been sorted and finds the column by which the data has been sorted:

Protected Overrides Sub Render(ByVal writer As HtmlTextWriter)
    ' Only add the sorting UI if the GridView is sorted
    If Not String.IsNullOrEmpty(ProductList.SortExpression) Then
        ' Determine the index and HeaderText of the column that
        'the data is sorted by
        Dim sortColumnIndex As Integer = -1
        Dim sortColumnHeaderText As String = String.Empty
        For i As Integer = 0 To ProductList.Columns.Count - 1
            If ProductList.Columns(i).SortExpression.CompareTo( _
                ProductList.SortExpression) = 0 Then
                sortColumnIndex = i
                sortColumnHeaderText = ProductList.Columns(i).HeaderText
                Exit For
            End If
        Next

        ' TODO: Scan the rows for differences in the sorted column's values
End Sub

If the GridView has yet to be sorted, the GridView's SortExpression property will not have been set. Therefore, we only want to add the separator rows if this property has some value. If it does, we next must determine the index of the column by which the data was sorted. This is accomplished by looping through the GridView's Columns collection, searching for the column whose SortExpression property equals the GridView's SortExpression property. In addition to the column's index, we also grab the HeaderText property, which is used when displaying the separator rows.

With the index of the column by which the data is sorted, the final step is to enumerate the rows of the GridView. For each row, we must determine whether the sorted column's value differs from the previous row's sorted column's value. If so, we must inject a new GridViewRow instance into the control hierarchy. This is accomplished with the following code:

Protected Overrides Sub Render(ByVal writer As HtmlTextWriter)
    ' Only add the sorting UI if the GridView is sorted
    If Not String.IsNullOrEmpty(ProductList.SortExpression) Then
        ' ... Code for finding the sorted column index removed for brevity ...


        ' Reference the Table the GridView has been rendered into
        Dim gridTable As Table = CType(ProductList.Controls(0), Table)

        ' Enumerate each TableRow, adding a sorting UI header if
        ' the sorted value has changed
        Dim lastValue As String = String.Empty
        For Each gvr As GridViewRow In ProductList.Rows
            Dim currentValue As String = gvr.Cells(sortColumnIndex).Text

            If lastValue.CompareTo(currentValue) <> 0 Then
                ' there's been a change in value in the sorted column
                Dim rowIndex As Integer = gridTable.Rows.GetRowIndex(gvr)

                ' Add a new sort header row
                Dim sortRow As New GridViewRow(rowIndex, rowIndex, _
                    DataControlRowType.DataRow, DataControlRowState.Normal)
                Dim sortCell As New TableCell()
                sortCell.ColumnSpan = ProductList.Columns.Count
                sortCell.Text = String.Format("{0}: {1}", _
                    sortColumnHeaderText, currentValue)
                sortCell.CssClass = "SortHeaderRowStyle"

                ' Add sortCell to sortRow, and sortRow to gridTable
                sortRow.Cells.Add(sortCell)
                gridTable.Controls.AddAt(rowIndex, sortRow)

                ' Update lastValue
                lastValue = currentValue
            End If
        Next
    End If

    MyBase.Render(writer)
End Sub

This code starts by programmatically referencing the Table object found at the root of the GridView's control hierarchy and creating a string variable named lastValue. lastValue is used to compare the current row's sorted column value with the previous row's value. Next, the GridView's Rows collection is enumerated and for each row the value of the sorted column is stored in the currentValue variable.

Note: To determine the value of the particular row's sorted column, I use the cell's Text property. This works well for BoundFields, but will not work as desired for TemplateFields, CheckBoxFields, and so on. We'll look at how to account for alternate GridView fields shortly.

The currentValue and lastValue variables are then compared. If they differ, we must add a new separator row to the control hierarchy. This is accomplished by determining the index of the GridViewRow in the Table object's Rows collection, creating new GridViewRow and TableCell instances, and then adding the TableCell and GridViewRow to the control hierarchy.

Note that the separator row's lone TableCell is formatted so that it spans the entire width of the GridView, is formatted using the SortHeaderRowStyle CSS class, and has its Text property such that it shows both the sort group name (such as "Category") and the group's value (such as "Beverages"). Finally, lastValue is updated to the value of currentValue.

The CSS class used to format the sorting group header row—SortHeaderRowStyle—must be specified in the Styles.css file. Feel free to use whatever style settings appeal to you; I used the following:

.SortHeaderRowStyle
{
    background-color: #c00;
    text-align: left;
    font-weight: bold;
    color: White;
}

With the current code, the sorting interface adds sort group headers when sorting by any BoundField (see Figure 5, which shows a screen shot when sorting by supplier). However, when sorting by any other field type (such as a CheckBoxField or TemplateField), the sort group headers are nowhere to be found (see Figure 6).

Figure 5. The sorting interface includes sort group headers when sorting by BoundFields.

Figure 6. The sort group headers are missing when sorting a CheckBoxField.

The reason the sort group headers are missing when sorting by a CheckBoxField is that the code currently uses just the TableCell's Text property to determine the value of the sorted column for each row. For CheckBoxFields, the TableCell's Text property is an empty string; instead, the value is available through a CheckBox Web control that resides within the TableCell's Controls collection.

To handle field types other than BoundFields, we must augment the code where the currentValue variable is assigned to check for the existence of a CheckBox in the TableCell's Controls collection. Instead of using currentValue = gvr.Cells(sortColumnIndex).Text, replace this code with the following:

Dim currentValue As String = String.Empty
If gvr.Cells(sortColumnIndex).Controls.Count > 0 Then
    If TypeOf gvr.Cells(sortColumnIndex).Controls(0) Is CheckBox Then
        If CType(gvr.Cells(sortColumnIndex).Controls(0), CheckBox).Checked 
Then
            currentValue = "Yes"
        Else
            currentValue = "No"
        End If

        ' ... Add other checks here if using columns with other
        '      Web controls in them (Calendars, DropDownLists, etc.) ...
    End If
Else
    currentValue = gvr.Cells(sortColumnIndex).Text
End If

This code examines the sorted column TableCell for the current row to determine if there are any controls in the Controls collection. If there are, and the first control is a CheckBox, the currentValue variable is set to "Yes" or "No", depending on the CheckBox's Checked property. Otherwise, the value is taken from the TableCell's Text property. This logic can be replicated to handle sorting for any TemplateFields that might exist in the GridView.

With the preceding code addition, the sort group headers are now present when sorting by the Discontinued CheckBoxField (see Figure 7).

Figure 7. The sort group headers are now present when sorting a CheckBoxField.

Note: If you have products with NULL database values for the CategoryID, SupplierID, or UnitPrice fields, those values will appear as empty strings in the GridView by default, meaning the separator row's text for those products with NULL values will read like "Category:" (that is, there's no name after "Category:" like with "Category: Beverages"). If you want a value displayed here you can either set the BoundFields' NullDisplayText property to the text you want displayed or you can add a conditional statement in the Render method when assigning the currentValue to the separator row's Text property.

Conclusion

The GridView does not include many built-in options for customizing the sorting interface. However, with a bit of low-level code, it is possible to tweak the GridView's control hierarchy to create a more customized interface. In this tutorial, we saw how to add a sort group separator row for a sortable GridView, which more easily identifies the distinct groups and those groups' boundaries. For additional examples of customized sorting interfaces, check out Scott Guthrie's A Few ASP.NET 2.0 GridView Sorting Tips and Tricks blog entry.

Happy programming!

 

About the author

Scott Mitchell, author of seven ASP/ASP.NET books and founder of 4GuysFromRolla.com, has been working with Microsoft Web technologies since 1998. Scott works as an independent consultant, trainer, and writer. His latest book is Sams Teach Yourself ASP.NET 2.0 in 24 Hours. He can be reached at mitchell@4GuysFromRolla.com or via his blog, which can be found at http://ScottOnWriting.NET.

© Microsoft Corporation. All rights reserved.