-
-
Notifications
You must be signed in to change notification settings - Fork 29
Row Selection
- Single Row Selection
- Multiple Row Selections
- Change Dynamically Single/Multiple Selections
- Mixing Single & Multiple Row Selections
- Disable Custom Rows Selections via
selectableOverride
- Disable External Button when having Empty Selection
- Change Row Selections
- Troubleshooting
For row selection, you can simply play with couple of grid options (see below) and subscribe to onSelectedRowsChanged
(a SlickGrid Event that is, it's not an Observable). However please note that onSelectedRowsChanged
is a function available on the Grid
object and you will need bind to (gridChanged)
to get the object when grid is ready. There are 2 types of row selection(s) which you can do.
Note: enableCheckboxSelector
and enableExcelCopyBuffer
do not work well together, this is because they both share the same Row.SelectionModel
and one cancels the other. It is recommended to not use enableExcelCopyBuffer
in that case.
For a single row selection, you need to have enableCellNavigation: true
, enableRowSelection: true
and multiSelect: false
and as described earlier, subscribe to onSelectedRowsChanged
(for that you need to bind to (gridChanged)
). There are 2 ways to choose for the implementation of a row selection, option 1. is the most common option and is the recommend way of doing it.
You can also do it through a delegate
since all SlickGrid events are exposed as delegate
. For more info see Wiki - OnEvents - 3. delegate
export class Example1 {
attached() {
this.initializeGrid();
this.dataset = this.loadData(500);
const gridContainerElm = document.querySelector<HTMLDivElement>(`.grid3`);
gridContainerElm.addEventListener('onSelectedRows', this.handleOnClick.bind(this));
this.sgb = new Slicker.GridBundle(gridContainerElm, this.columnDefinitions, { ...ExampleGridOptions, ...this.gridOptions }, this.dataset);
}
initializeGrid() {
// define columns
...
// grid options
this.gridOptions = {
enableAutoResize: true,
enableCellNavigation: true,
enableCheckboxSelector: true,
enableRowSelection: true,
multiSelect: false,
}
}
handleRowSelection(event) {
const args = event?.detail?.args;
console.log(event, args);
}
}
It's preferable to use the with delegate
, but if you really wish, you can also use directly the SlickGrid event that you can subscribe to. However, don't forget to unsubscribe to a SlickGrid event.
this.gridOptions = {
enableAutoResize: true,
enableCellNavigation: true,
enableRowSelection: true
}
gridObjChanged(grid) {
grid.onSelectedRowsChanged.subscribe((e, args) => {
if (Array.isArray(args.rows)) {
this.selectedObjects = args.rows.map(idx => {
const item = grid.getDataItem(idx);
return item.title || '';
});
}
});
}
As for multiple row selections, you need to disable enableCellNavigation
and enable enableCheckboxSelector
and enableRowSelection
. Then as describe earlier, you will subscribe to onSelectedRowsChanged
(for that you need to bind to (gridChanged)
). There are 2 ways to choose for the implementation of a row selection, option 1. is the most common option and is the recommend way of doing it.
You can also do it through a delegate
since all SlickGrid events are exposed as delegate
. For more info see Wiki - OnEvents - 3. delegate
export class Example1 {
attached() {
this.initializeGrid();
this.dataset = this.loadData(500);
const gridContainerElm = document.querySelector<HTMLDivElement>(`.grid3`);
gridContainerElm.addEventListener('onSelectedRows', this.handleOnClick.bind(this));
this.sgb = new Slicker.GridBundle(gridContainerElm, this.columnDefinitions, { ...ExampleGridOptions, ...this.gridOptions }, this.dataset);
}
initializeGrid() {
// define columns
...
// grid options
this.gridOptions = {
enableAutoResize: true,
enableCellNavigation: true,
enableCheckboxSelector: true,
enableRowSelection: true,
rowSelectionOptions: {
// True (Single Selection), False (Multiple Selections)
selectActiveRow: false
},
}
}
handleRowSelection(event) {
const args = event?.detail?.args;
console.log(event, args);
}
}
It's preferable to use the with delegate
, but if you really wish, you can also use directly the SlickGrid event that you can subscribe to. However, don't forget to unsubscribe to a SlickGrid event.
export class Example1 {
defineGrid() {
this.gridOptions = {
enableAutoResize: true,
enableCellNavigation: true,
enableCheckboxSelector: true,
enableRowSelection: true,
rowSelectionOptions: {
// True (Single Selection), False (Multiple Selections)
selectActiveRow: false
},
}
}
gridObjChanged(grid) {
grid.onSelectedRowsChanged.subscribe((e, args) => {
if (Array.isArray(args.rows)) {
this.selectedObjects = args.rows.map(idx => {
const item = grid.getDataItem(idx);
return item.title || '';
});
}
});
}
}
If you want to change from Multiple Selections to Single Selection (and vice-versa), you could toggle the grid options enableCellNavigation
flag (False
when you want Single Selection), however this is not possible when using Inline Editors since this flag is required. Note that there is currently no other ways of toggling dynamically without re-creating the grid.
SlickGrid is so powerful and customizable, you could if you wish mix the multiple row selections (cell column 1) and single row selection (any other cell click). For that though, you will need to use 2 SlickGrid Events (onClick
and onSelectedRowsChanged
). For example with a delegate
we can do it this way:
export class Example1 {
handleMultipleRowSelections(event) {
const args = event?.detail?.args;
console.log('multiple row checkbox selected', event, args);
}
handleSingleRowClick(event) {
const args = event?.detail?.args;
console.log('multiple row checkbox selected', event, args);
// when clicking on any cell, we will make it the new selected row
// however, we don't want to interfere with multiple row selection checkbox which is on 1st column cell
if (args.cell !== 0) {
grid.setSelectedRows([args.row]);
}
}
}
You can use selectableOverride
to provide custom logic to disable certain rows selections, for example the code below will remove the row selection on every second row.
export class Example1 implements OnInit {
prepareGrid() {
this.gridOptions = {
enableRowSelection: true,
enableCheckboxSelector: true,
checkboxSelector: {
// you can override the logic for showing (or not) the expand icon
// for example, display the expand icon only on every 2nd row
selectableOverride: (row: number, dataContext: any, grid: any) => (dataContext.id % 2 === 1)
},
multiSelect: false,
rowSelectionOptions: {
// True (Single Selection), False (Multiple Selections)
selectActiveRow: true,
},
};
}
}
When having an external button that you want to work only when there's row selection, there are 2 ways of doing this.
- use the
onSelectedRowsChanged
event (via your View in HTML or via ViewModel)
<button disabled.bind="isMyButtonDisabled">My Button</button>
<div class="myGrid"
onselectedrowschanged.delegate="handleOnSelectedRowsChanged">
</div>
isMyButtonDisabled = false;
handleOnSelectedRowsChanged(event) {
const args = event?.detail?.args;
this.isMyButtonDisabled = args.rows?.length === 0;
}
- use the
onGridStateChanged
event (see Grid State & Presets Wiki)
<button disabled.bind="isMyButtonDisabled">My Button</button>
<div class="myGrid"
ongridstatechanged.delegate="handleOngridStateChanged">
</div>
isMyButtonDisabled = false;
handleOngridStateChanged(event) {
const gridState = event && event.detail && event.detail.gridState;
if (Array.isArray(gridState?.rowSelection.dataContextIds)) {
this.isMassSelectionDisabled = gridState.rowSelection.dataContextIds.length === 0;
}
}
You can change which row(s) are selected by using the built-in SlickGrid method setSelectedRows(rowIndexes)
(passing an empty array will clear all selection), however please note that it requires an array of row indexes as you see them in the UI and it won't work that great with Pagination (if that is what you are looking for then take a look at this Stack Overflow Q&A)
export class Example1 {
attached() {
this.initializeGrid();
this.dataset = this.loadData(500);
const gridContainerElm = document.querySelector<HTMLDivElement>(`.grid3`);
this.sgb = new Slicker.GridBundle(gridContainerElm, this.columnDefinitions, { ...ExampleGridOptions, ...this.gridOptions }, this.dataset);
}
clearRowSelection() {
this.sgb.slickGrid.setSelectedRows([]); // empty array will clear the row selection
}
changeRowSelections() {
this.sgb.slickGrid.setSelectedRows(rowIndexes);
// OR providing an empty array will clear the row selection
// this.sgb.slickGrid.setSelectedRows([]);
}
}
The reason is because the Row Selection (checkbox) plugin is a special column and Slickgrid-Universal is adding an extra column dynamically for the Row Selection checkbox and that is not reflected in your local copy of columnDefinitions
. To address this issue, you need to get the Slickgrid-Universal internal copy of all columns (including the extra columns), you can get it via getAllColumnDefinitions()
from the Grid Service and then you can use to that array and that will work.
const newColumn = { /*...*/ };
const allColumns = this.sgb.gridService.getAllColumnDefinitions();
allColumns.push(newColumn);
this.columnDefinitions = allColumns.slice(); // or use spread operator [...cols]
// you could also use SlickGrid setColumns() method
// this.sgb.slickGrid.setColumns(cols);
- Slickgrid-Universal Wikis
- Installation
- Styling
- Interfaces/Models
- Column Functionalities
- Events
- Grid Functionalities
- Auto-Resize / Resizer Service
- Resize by Cell Content
- Column Picker
- Composite Editor Modal
- Custom Tooltip
- Context Menu
- Custom Footer
- Export to Excel
- Export to File (csv/txt)
- Grid Menu
- Grid State & Presets
- Grouping & Aggregators
- Header Menu & Header Buttons
- Pinning (frozen) of Columns/Rows
- Row Selection
- Tree Data Grid
- SlickGrid & DataView objects
- Backend Services