Skip to content

Commit

Permalink
Allow multiple selection in SelectNext
Browse files Browse the repository at this point in the history
  • Loading branch information
acelaya committed Jul 3, 2024
1 parent b9b6ce5 commit 4fe5fa3
Show file tree
Hide file tree
Showing 5 changed files with 252 additions and 46 deletions.
4 changes: 2 additions & 2 deletions src/components/input/SelectContext.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
import { createContext } from 'preact';

export type SelectContextType<T = unknown> = {
selectValue: (newValue: T) => void;
value: T;
selectValue: (newValue: T | T[]) => void;
value: T | T[];
};

const SelectContext = createContext<SelectContextType | null>(null);
Expand Down
126 changes: 87 additions & 39 deletions src/components/input/SelectNext.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,35 @@ function SelectOption<T>({
}

const { selectValue, value: currentValue } = selectContext;
const selected = !disabled && currentValue === value;
const valueIsArray = Array.isArray(currentValue);

const selected =
!disabled &&
((valueIsArray && currentValue.includes(value)) || currentValue === value);

const selectOrToggle = useCallback(() => {
// In single-select, just set current value
if (!valueIsArray) {
selectValue(value);
return;
}

// In multi-select, clear selection for nullish values
if (!value) {
selectValue([]);
return;
}

// In multi-select, toggle clicked items
const index = currentValue.indexOf(value);
if (index === -1) {
selectValue([...currentValue, value]);
} else {
const copy = [...currentValue];
copy.splice(index, 1);
selectValue(copy);
}
}, [currentValue, selectValue, value, valueIsArray]);

return (
<li
Expand All @@ -75,13 +103,13 @@ function SelectOption<T>({
)}
onClick={() => {
if (!disabled) {
selectValue(value);
selectOrToggle();
}
}}
onKeyPress={e => {
if (!disabled && ['Enter', 'Space'].includes(e.code)) {
e.preventDefault();
selectValue(value);
selectOrToggle();
}
}}
role="option"
Expand Down Expand Up @@ -215,43 +243,59 @@ function useListboxPositioning(
}, [adjustListboxPositioning, asPopover]);
}

export type SelectProps<T> = CompositeProps & {
type SingleValueProps<T> = {
value: T;
onChange: (newValue: T) => void;
buttonContent?: ComponentChildren;
disabled?: boolean;
};

/**
* `id` attribute for the toggle button. This is useful to associate a label
* with the control.
*/
buttonId?: string;

/** Additional classes to pass to container */
containerClasses?: string | string[];
/** Additional classes to pass to toggle button */
buttonClasses?: string | string[];
/** Additional classes to pass to listbox */
listboxClasses?: string | string[];

/**
* Align the listbox to the right.
* Useful when the listbox is bigger than the toggle button and this component
* is rendered next to the right side of the page/container.
* Defaults to false.
*/
right?: boolean;

'aria-label'?: string;
'aria-labelledby'?: string;

/**
* Used to determine if the listbox should use the popover API.
* Defaults to true, as long as the browser supports it.
*/
listboxAsPopover?: boolean;
type MultiValueProps<T> = {
value: T[];
onChange: (newValue: T[]) => void;
};

export type SelectProps<T> = CompositeProps &
(SingleValueProps<T> | MultiValueProps<T>) & {
buttonContent?: ComponentChildren;
disabled?: boolean;

/**
* Whether this select should allow multi-selection or not.
* When this is true, the listbox is kept open when an option is selected.
* Defaults to false.
*/
multiple?: boolean;

/**
* `id` attribute for the toggle button. This is useful to associate a label
* with the control.
*/
buttonId?: string;

/** Additional classes to pass to container */
containerClasses?: string | string[];
/** Additional classes to pass to toggle button */
buttonClasses?: string | string[];
/** Additional classes to pass to listbox */
listboxClasses?: string | string[];

/**
* Align the listbox to the right.
* Useful when the listbox is bigger than the toggle button and this component
* is rendered next to the right side of the page/container.
* Defaults to false.
*/
right?: boolean;

'aria-label'?: string;
'aria-labelledby'?: string;

/**
* Used to determine if the listbox should use the popover API.
* Defaults to true, as long as the browser supports it.
*/
listboxAsPopover?: boolean;
};

function SelectMain<T>({
buttonContent,
value,
Expand All @@ -264,6 +308,7 @@ function SelectMain<T>({
listboxClasses,
containerClasses,
right = false,
multiple = false,
'aria-label': ariaLabel,
'aria-labelledby': ariaLabelledBy,
/* eslint-disable-next-line no-prototype-builtins */
Expand Down Expand Up @@ -295,11 +340,14 @@ function SelectMain<T>({
);

const selectValue = useCallback(
(newValue: unknown) => {
onChange(newValue as T);
closeListbox();
(value: unknown) => {
onChange(value as any);
// In multi-select mode, keep list open when selecting values
if (!multiple) {
closeListbox();
}
},
[closeListbox, onChange],
[onChange, multiple, closeListbox],
);

// When clicking away, focusing away or pressing `Esc`, close the listbox
Expand Down
86 changes: 81 additions & 5 deletions src/components/input/test/SelectNext-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,9 @@ describe('SelectNext', () => {

const wrapper = mount(
<SelectNext value={undefined} onChange={sinon.stub()} {...props}>
<SelectNext.Option value={undefined}>
<span data-testid="reset-option">Reset</span>
</SelectNext.Option>
{items.map(item => (
<SelectNext.Option
value={item}
Expand Down Expand Up @@ -87,19 +90,20 @@ describe('SelectNext', () => {
return listboxTop < buttonTop;
};

const clickOption = (wrapper, id) =>
wrapper.find(`[data-testid="option-${id}"]`).simulate('click');

it('changes selected value when an option is clicked', () => {
const onChange = sinon.stub();
const wrapper = createComponent({ onChange });
const clickOption = index =>
wrapper.find(`[data-testid="option-${index}"]`).simulate('click');

clickOption(3);
clickOption(wrapper, 3);
assert.calledWith(onChange.lastCall, items[2]);

clickOption(5);
clickOption(wrapper, 5);
assert.calledWith(onChange.lastCall, items[4]);

clickOption(1);
clickOption(wrapper, 1);
assert.calledWith(onChange.lastCall, items[0]);
});

Expand Down Expand Up @@ -385,6 +389,61 @@ describe('SelectNext', () => {
});
});

context('when multi-selection is enabled', () => {
it('keeps listbox open when an option is selected if multiple is true', async () => {
const wrapper = createComponent({ multiple: true });

toggleListbox(wrapper);
assert.isFalse(isListboxClosed(wrapper));

clickOption(wrapper, 1);

// After clicking an option, the listbox is still open
assert.isFalse(isListboxClosed(wrapper));
});

it('allows multiple items to be selected when the value is an array', () => {
const onChange = sinon.stub();
const wrapper = createComponent({
value: [items[0], items[2]],
onChange,
});

toggleListbox(wrapper);
clickOption(wrapper, 2);

// When a not-yet-selected item is clicked, it will be selected
assert.calledWith(onChange, [items[0], items[2], items[1]]);
});

it('allows deselecting already selected options', () => {
const onChange = sinon.stub();
const wrapper = createComponent({
value: [items[0], items[2]],
onChange,
});

toggleListbox(wrapper);
clickOption(wrapper, 3);

// When an already selected item is clicked, it will be de-selected
assert.calledWith(onChange, [items[0]]);
});

it('resets selection when option value is nullish and select value is an array', () => {
const onChange = sinon.stub();
const wrapper = createComponent({
value: [items[0], items[2]],
onChange,
});

toggleListbox(wrapper);
wrapper.find(`[data-testid="reset-option"]`).simulate('click');

assert.calledWith(onChange, []);
});
});

it(
'should pass a11y checks',
checkAccessibility([
Expand All @@ -405,6 +464,23 @@ describe('SelectNext', () => {
);
toggleListbox(wrapper);

return wrapper;
},
},
{
name: 'Open Multi-Select listbox',
content: () => {
const wrapper = createComponent(
{
buttonContent: 'Select',
'aria-label': 'Select',
value: [items[1], items[3]],
multiple: true,
},
{ optionsChildrenAsCallback: false },
);
toggleListbox(wrapper);

return wrapper;
},
},
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -425,6 +425,39 @@ export default function SelectNextPage() {
withSource
/>
</Library.Example>
<Library.Example title="multiple">
<Library.Info>
<Library.InfoItem label="description">
Determines if more than one item can be selected at once,
causing the listbox to stay open when an option is selected on
it.
<p>Considerations:</p>
<ul className="list-disc">
<li>
When multi-selection is enabled, the <code>value</code>{' '}
should be an array instead of a single item. This will cause
every option to be {'"toggable"'} instead of having single
selection.
</li>
<li>
Managing the state and representing the list of selected
items is up to consumers, like with single selection.
</li>
</ul>
</Library.InfoItem>
<Library.InfoItem label="type">
<code>boolean</code>
</Library.InfoItem>
<Library.InfoItem label="default">
<code>false</code>
</Library.InfoItem>
</Library.Info>
<Library.Demo
title="Multi-select listbox"
exampleFile="select-next-multiple"
withSource
/>
</Library.Example>
</Library.Pattern>

<Library.Pattern title="SelectNext.Option component API">
Expand Down
49 changes: 49 additions & 0 deletions src/pattern-library/examples/select-next-multiple.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import { useId, useState } from 'preact/hooks';

import { SelectNext } from '../..';

type ItemType = {
id: string;
name: string;
};

const items: ItemType[] = [
{ id: '1', name: 'John Doe' },
{ id: '2', name: 'Albert Banana' },
{ id: '3', name: 'Bernard California' },
{ id: '4', name: 'Cecelia Davenport' },
{ id: '5', name: 'Doris Evanescence' },
];

export default function App() {
const [values, setSelected] = useState<ItemType[]>([items[0], items[3]]);
const selectId = useId();

return (
<div className="w-96 mx-auto">
<label htmlFor={selectId}>Select students</label>
<SelectNext
multiple
value={values}
onChange={setSelected}
buttonId={selectId}
buttonContent={
values.length === 0 ? (
<>All students</>
) : values.length === 1 ? (
values[0].name
) : (
<>{values.length} students selected</>
)
}
>
<SelectNext.Option value={undefined}>All students</SelectNext.Option>
{items.map(item => (
<SelectNext.Option value={item} key={item.id}>
{item.name}
</SelectNext.Option>
))}
</SelectNext>
</div>
);
}

0 comments on commit 4fe5fa3

Please sign in to comment.