Cell Selection
The TreeList component supports single or multiple cell selection. You can select a cell with mouse click anywhere in the cell. You can access the collection of selected cells, use this collection and manipulate it. You can subscribe to selection events.
Basics
To select a cell, click anywhere in it.
To select a range of cells in one or more columns, hold the Shift key, while clicking on the first and last cell of the range. To select or deselect multiple cells that don't belong to a range, hold the Ctrl key.
You can also select a cell range by holding and dragging the mouse cursor. The dragging motion defines the diagonal of a rectangle and the TreeList will select the cells under this rectangle. To allow this kind of cell selection, set the DragToSelect
parameter in TreeListSelectionSettings
. The Shift and Ctrl modifiers are not supported in drag-to-select mode.
To enable cell selection:
- Set the TreeList
SelectedCells
parameter to a collection of typeIEnumerable<TreeListSelectedCellDescriptor>
. The collection must be initialized in advance. SeeTreeListSelectedCellDescriptor
for infomation about the object properties. - Add a
<TreeListSelectionSettings>
tag to the<TreeListSettings>
tag, and set theSelectionType
parameter to theTreeListSelectionType.Cell
.
<TelerikTreeList Data="@TreeListData"
IdField="@nameof(Employee.Id)"
ParentIdField="@nameof(Employee.ParentId)"
SelectionMode="@TreeListSelectionMode.Multiple"
@bind-SelectedCells="@SelectedCells"
Pageable="true">
<TreeListSettings>
<TreeListSelectionSettings SelectionType="@TreeListSelectionType.Cell" DragToSelect="true" />
</TreeListSettings>
<TreeListColumns>
<TreeListColumn Field="@nameof(Employee.FirstName)" Title="First Name" Width="350px" Expandable="true" />
<TreeListColumn Field="@nameof(Employee.LastName)" Title="Last Name" />
<TreeListColumn Field="@nameof(Employee.Position)" Title="Position" Width="200px" />
</TreeListColumns>
</TelerikTreeList>
<h3>Selected Cells:</h3>
<ul>
@foreach (TreeListSelectedCellDescriptor cellDescriptor in SelectedCells)
{
<li>
Column <code>Field</code>: @cellDescriptor.ColumnField,
<code>EmployeeId</code>: @(((Employee)cellDescriptor.DataItem).Id)
</li>
}
</ul>
@code {
private List<Employee> TreeListData { get; set; } = new();
private IEnumerable<TreeListSelectedCellDescriptor> SelectedCells { get; set; } = Enumerable.Empty<TreeListSelectedCellDescriptor>();
protected override void OnInitialized()
{
for (int i = 1; i <= 9; i++)
{
TreeListData.Add(new Employee()
{
Id = i,
ParentId = i <= 3 ? null : i % 3 + 1,
FirstName = "First " + i,
LastName = "Last " + i,
Position = i <= 3 ? "Team Lead" : "Software Engineer"
});
}
SelectedCells = new List<TreeListSelectedCellDescriptor>() {
new TreeListSelectedCellDescriptor()
{
DataItem = TreeListData.ElementAt(2),
ColumnField = nameof(Employee.FirstName)
}
};
}
public class Employee
{
public int Id { get; set; }
public int? ParentId { get; set; }
public string FirstName { get; set; } = string.Empty;
public string LastName { get; set; } = string.Empty;
public string Position { get; set; } = string.Empty;
}
}
SelectedCellsChanged Event
You can respond to user selection actions through the SelectedCellsChanged
event. The event handler receives a collection of type IEnumerable<TreeListSelectedCellDescriptor>
. The collection may have multiple, single, or no objects in it, depending on the SelectionMode
and the last user selection.
@* Select cells and handle the SelectedCellsChanged event *@
<TelerikTreeList Data="@TreeListData"
IdField="@nameof(Employee.Id)"
ParentIdField="@nameof(Employee.ParentId)"
SelectionMode="@TreeListSelectionMode.Multiple"
SelectedCells="@SelectedCells"
SelectedCellsChanged="@( (IEnumerable<TreeListSelectedCellDescriptor> newSelected) => OnCellSelect(newSelected) )"
Pageable="true">
<TreeListSettings>
<TreeListSelectionSettings SelectionType="@TreeListSelectionType.Cell" DragToSelect="true" />
</TreeListSettings>
<TreeListColumns>
<TreeListColumn Field="@nameof(Employee.FirstName)" Title="First Name" Width="350px" Expandable="true" />
<TreeListColumn Field="@nameof(Employee.LastName)" Title="Last Name" />
<TreeListColumn Field="@nameof(Employee.Position)" Title="Position" Width="200px" />
</TreeListColumns>
</TelerikTreeList>
<p><code>SelectedItemsChanged</code> fired at: @SelectedCellsChangedLog</p>
<h3>Selected Cells:</h3>
<ul>
@foreach (TreeListSelectedCellDescriptor cellDescriptor in SelectedCells)
{
<li>
Column <code>Field</code>: @cellDescriptor.ColumnField,
<code>EmployeeId</code>: @(((Employee)cellDescriptor.DataItem).Id)
</li>
}
</ul>
@code {
private List<Employee> TreeListData { get; set; } = new();
private IEnumerable<TreeListSelectedCellDescriptor> SelectedCells { get; set; } = Enumerable.Empty<TreeListSelectedCellDescriptor>();
private string SelectedCellsChangedLog { get; set; } = string.Empty;
protected void OnCellSelect(IEnumerable<TreeListSelectedCellDescriptor> cellDescriptors)
{
// Update the SelectedCells collection manually.
// When using two-way binding, this happens automatically.
SelectedCells = cellDescriptors;
SelectedCellsChangedLog = DateTime.Now.ToLongTimeString();
}
protected override void OnInitialized()
{
for (int i = 1; i <= 15; i++)
{
TreeListData.Add(new Employee()
{
Id = i,
ParentId = i <= 3 ? null : i % 3 + 1,
FirstName = "First " + i,
LastName = "Last " + i,
Position = i <= 3 ? "Team Lead" : "Software Engineer"
});
}
SelectedCells = new List<TreeListSelectedCellDescriptor>() {
new TreeListSelectedCellDescriptor()
{
DataItem = TreeListData.ElementAt(2),
ColumnField = nameof(Employee.FirstName)
}
};
}
public class Employee
{
public int Id { get; set; }
public int? ParentId { get; set; }
public string FirstName { get; set; } = string.Empty;
public string LastName { get; set; } = string.Empty;
public string Position { get; set; } = string.Empty;
}
}
SelectedCellsChanged and Asynchronous Operations
The SelectedCellsChanged
event handler cannot be awaited. To execute asynchronous operations when the user selects rows, use the OnRowClick
or OnRowDoubleClick
event instead.
TreeListSelectedCellDescriptor
The TreeListSelectedCellDescriptor
type exposes the following properties:
Property Name | Type | Description |
---|---|---|
ColumnField |
string |
The value of the TreeList column Field parameter, if set. |
ColumnId |
string |
The value of the TreeList column Id parameter, if set. |
DataItem |
object |
The TreeList data item instance. Cast it to the actual TreeList model type before use. |
Selection When Data Changes
When the TreeList Data
collection changes, the SelectedCells
collection has the following behavior:
- When the user updates a selected cell and the item instance is replaced, you have to also replace the
DataItem
object in theSelectedCells
collection. Do that in the TreeListOnUpdate
event. - When the user deletes a row with selected cells, update the
SelectedCells
collection in the the TreeListOnDelete
event handler. - To select cells from a new item in the TreeList you can use the
OnCreate
event to update theSelectedCells
collection.
Equals Comparison
The items in SelectedCells
are compared against the items in the TreeList data in order to determine which cells will be highlighted. The default framework behavior is to compare objects by reference. The data item references may not match when:
- The
SelectedCells
are obtained from a different data source than the all TreeList items, for example, from a separate service.
In such cases, the selected cells may not appear as expected. You have to override the Equals
method of the TreeList model class so that the items are compared by a unique identifier rather than by reference. When you override Equals
, it is also recommended to override the GetHashCode
method.
Cell Selection and Other TreeList Features
The selection feature behavior may vary when the TreeList configuration combines cell selection and other TreeList features, such as editing, virtualization, paging, templates. In such cases you need to consider certain limitation or include some modications.
Selection and Editing Modes
When you want to edit a TreeList item, the cell selection has the following behavior:
- Cell selection is not supported with
Incell
edit mode due to the overlapping pointer events that trigger selection and editing. If both features are enabled, only the editing will work. -
Inline
edit mode andPopup
edit mode integrate with cell selection without limitations.
Selection and Virtual Scrolling
When the TreeList has virtual scrolling, the component is able to select a range of cells with Shift only if all rows in that range are currently rendered. Consider the following scenario:
- Select a cell.
- Scroll down, so that virtualization kicks in and the rendered rows are no longer the same.
- Select another cell with Shift.
In this case, the range selection will start from the first row that is currently rendered. Compare with Selection and paging below.
Selection and Paging
The SelectedCells
collection persists across paging.
Selection and Templates
When using TreeList templates with cell selection:
- If you are using a TreeList column template and you have a clickable element in the template, wrap this element in a container with a
@onclick:stopPropagation
directive. You can check the knowledge base article on how to prevent row selection when the user clicks another component in the Grid column template. It is for the Grid component, but the concept is identical and it applies for both row and cell selection. - If you are using a row template the TreeList does not support cell selection. The row template removes the built-in cell instances and the HTML markup may not even include the expected number of cells.