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 b2a12f0
Show file tree
Hide file tree
Showing 5 changed files with 259 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
131 changes: 92 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,11 +308,16 @@ function SelectMain<T>({
listboxClasses,
containerClasses,
right = false,
multiple = false,
'aria-label': ariaLabel,
'aria-labelledby': ariaLabelledBy,
/* eslint-disable-next-line no-prototype-builtins */
listboxAsPopover = HTMLElement.prototype.hasOwnProperty('popover'),
}: SelectProps<T>) {
if (multiple && !Array.isArray(value)) {
throw new Error('When `multiple` is true, the value must be an array');
}

const wrapperRef = useRef<HTMLDivElement | null>(null);
const listboxRef = useRef<HTMLUListElement | null>(null);
const [listboxOpen, setListboxOpen] = useState(false);
Expand All @@ -295,11 +344,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 Expand Up @@ -387,6 +439,7 @@ function SelectMain<T>({
role="listbox"
ref={listboxRef}
id={listboxId}
aria-multiselectable={multiple}
aria-labelledby={buttonId ?? defaultButtonId}
aria-orientation="vertical"
data-testid="select-listbox"
Expand Down
96 changes: 91 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,71 @@ describe('SelectNext', () => {
});
});

context('when multi-selection is enabled', () => {
it('throws if multiple is true and the value is not an arry', async () => {
assert.throws(
() => createComponent({ multiple: true }),
'When `multiple` is true, the value must be an array',
);
});

it('keeps listbox open when an option is selected if multiple is true', async () => {
const wrapper = createComponent({ multiple: true, value: [] });

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({
multiple: true,
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({
multiple: true,
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({
multiple: true,
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 +474,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,31 @@ 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>
When multi-selection is enabled, the <code>value</code> must
be an array and <code>onChange</code> will receive an array as
an argument.
</p>
</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
Loading

0 comments on commit b2a12f0

Please sign in to comment.