Skip to content

Commit

Permalink
[EuiBasicTable] Support indeterminate checkboxes in the header select…
Browse files Browse the repository at this point in the history
…ion checkbox

- we have the technology!! I figured, why not?

- make clicking an indeterminate checkbox more like gmail's behavior (i.e., it deselects)

clarify deselect vs select behavior with more titles/aria labels
  • Loading branch information
cee-chen committed Jun 6, 2024
1 parent bcd8f3d commit cff77ae
Show file tree
Hide file tree
Showing 5 changed files with 85 additions and 4 deletions.
1 change: 1 addition & 0 deletions packages/eui/changelogs/upcoming/7814.md
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
- Updated `EuiBasicTable` and `EuiInMemoryTable`s with `selection` - the header row checkbox will now render an indeterminate state if some (but not all) rows are selected
Original file line number Diff line number Diff line change
Expand Up @@ -152,6 +152,7 @@ exports[`EuiBasicTable renders (kitchen sink) with pagination, selection, sortin
class="euiCheckbox__input"
data-test-subj="checkboxSelectAll"
id="_selection_column-checkbox_generated-id_desktop"
title="Select all rows"
type="checkbox"
/>
<div
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ exports[`EuiInMemoryTable behavior mobile header 1`] = `
aria-label="Select all rows"
class="euiCheckbox__input"
id="_selection_column-checkbox_generated-id_mobile"
title="Select all rows"
type="checkbox"
/>
<div
Expand Down
67 changes: 67 additions & 0 deletions packages/eui/src/components/basic_table/basic_table.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
*/

import React from 'react';
import { fireEvent } from '@testing-library/react';
import { render, screen } from '../../test/rtl';
import { requiredProps } from '../../test';
import { shouldRenderCustomStyles } from '../../test/internal';
Expand Down Expand Up @@ -461,6 +462,72 @@ describe('EuiBasicTable', () => {
expect(onSelectionChange).toHaveBeenCalledWith([]);
expect(container.querySelectorAll('[checked]')).toHaveLength(0);
});

describe('header checkbox', () => {
it('selects all rows', () => {
const props: EuiBasicTableProps<BasicItem> = {
items: basicItems,
columns: basicColumns,
itemId: 'id',
selection: {
onSelectionChange: () => {},
initialSelected: [],
},
};
const { getByTestSubject } = render(<EuiBasicTable {...props} />);
expect(getByTestSubject('checkboxSelectAll')).not.toBeChecked();

fireEvent.click(getByTestSubject('checkboxSelectAll'));

expect(getByTestSubject('checkboxSelectAll')).toBeChecked();
expect(getCheckboxAt(1)).toBeChecked();
expect(getCheckboxAt(2)).toBeChecked();
expect(getCheckboxAt(3)).toBeChecked();
});

it('deselects all rows', () => {
const props: EuiBasicTableProps<BasicItem> = {
items: basicItems,
columns: basicColumns,
itemId: 'id',
selection: {
onSelectionChange: () => {},
initialSelected: basicItems,
},
};
const { getByTestSubject } = render(<EuiBasicTable {...props} />);
expect(getByTestSubject('checkboxSelectAll')).toBeChecked();

fireEvent.click(getByTestSubject('checkboxSelectAll'));

expect(getByTestSubject('checkboxSelectAll')).not.toBeChecked();
expect(getCheckboxAt(1)).not.toBeChecked();
expect(getCheckboxAt(2)).not.toBeChecked();
expect(getCheckboxAt(3)).not.toBeChecked();
});

it('renders an indeterminate header checkbox if some but not all rows are selected', () => {
const props: EuiBasicTableProps<BasicItem> = {
items: basicItems,
columns: basicColumns,
itemId: 'id',
selection: {
onSelectionChange: () => {},
initialSelected: [],
},
};
const { getByTestSubject } = render(<EuiBasicTable {...props} />);
expect(getByTestSubject('checkboxSelectAll')).not.toBeChecked();

fireEvent.click(getCheckboxAt(1));
expect(getCheckboxAt(1)).toBeChecked();
expect(getByTestSubject('checkboxSelectAll')).toBePartiallyChecked();

// Should deselect all rows on indeterminate click
fireEvent.click(getByTestSubject('checkboxSelectAll'));
expect(getCheckboxAt(1)).not.toBeChecked();
});
});
});

test('footers', () => {
Expand Down
19 changes: 15 additions & 4 deletions packages/eui/src/components/basic_table/basic_table.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -688,28 +688,39 @@ export class EuiBasicTable<T extends object = any> extends Component<
selectableItems.length > 0 &&
this.state.selection.length === selectableItems.length;

const indeterminate =
!checked &&
this.state.selection &&
selectableItems.length > 0 &&
this.state.selection.length > 0;

const disabled = selectableItems.length === 0;

const onChange = (event: React.ChangeEvent<HTMLInputElement>) => {
if (event.target.checked) {
if (event.target.checked && !indeterminate) {
this.changeSelection(selectableItems);
} else {
this.changeSelection([]);
}
};

return (
<EuiI18n token="euiBasicTable.selectAllRows" default="Select all rows">
{(selectAllRows: string) => (
<EuiI18n
tokens={['euiBasicTable.selectAllRows', 'euiBasicTable.deselectRows']}
defaults={['Select all rows', 'Deselect rows']}
>
{([selectAllRows, deselectRows]: string[]) => (
<EuiCheckbox
id={this.selectAllIdGenerator(isMobile ? 'mobile' : 'desktop')}
type={isMobile ? undefined : 'inList'}
checked={checked}
indeterminate={indeterminate}
disabled={disabled}
onChange={onChange}
// Only add data-test-subj to one of the checkboxes
data-test-subj={isMobile ? undefined : 'checkboxSelectAll'}
aria-label={selectAllRows}
aria-label={checked || indeterminate ? deselectRows : selectAllRows}
title={checked || indeterminate ? deselectRows : selectAllRows}
label={isMobile ? selectAllRows : null}
/>
)}
Expand Down

0 comments on commit cff77ae

Please sign in to comment.