aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJeremy Davis <jeremy.davis@sonarsource.com>2023-05-08 15:06:33 +0200
committersonartech <sonartech@sonarsource.com>2023-05-10 20:05:28 +0000
commitef4dcc3b8dfe68ae85ae7c146fb1d670f537ee1a (patch)
treee3b71bdab5f2785236960f4f79b3c03a24380cf9
parent2aa46faae4e27b4c1a6bbfb521fed9e5ce584158 (diff)
downloadsonarqube-ef4dcc3b8dfe68ae85ae7c146fb1d670f537ee1a.tar.gz
sonarqube-ef4dcc3b8dfe68ae85ae7c146fb1d670f537ee1a.zip
SONAR-19168 New Tags component
-rw-r--r--server/sonar-web/design-system/src/components/DropdownToggler.tsx5
-rw-r--r--server/sonar-web/design-system/src/components/FocusOutHandler.tsx60
-rw-r--r--server/sonar-web/design-system/src/components/InputSearch.tsx2
-rw-r--r--server/sonar-web/design-system/src/components/MultiSelect.tsx336
-rw-r--r--server/sonar-web/design-system/src/components/MultiSelectOption.tsx64
-rw-r--r--server/sonar-web/design-system/src/components/Tags.tsx113
-rw-r--r--server/sonar-web/design-system/src/components/TagsSelector.tsx70
-rw-r--r--server/sonar-web/design-system/src/components/__tests__/MultiSelect-test.tsx113
-rw-r--r--server/sonar-web/design-system/src/components/__tests__/Tags-test.tsx116
-rw-r--r--server/sonar-web/design-system/src/components/buttons.tsx8
-rw-r--r--server/sonar-web/design-system/src/components/index.ts2
11 files changed, 887 insertions, 2 deletions
diff --git a/server/sonar-web/design-system/src/components/DropdownToggler.tsx b/server/sonar-web/design-system/src/components/DropdownToggler.tsx
index f4217e42ef7..14ed580b209 100644
--- a/server/sonar-web/design-system/src/components/DropdownToggler.tsx
+++ b/server/sonar-web/design-system/src/components/DropdownToggler.tsx
@@ -18,6 +18,7 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
import EscKeydownHandler from './EscKeydownHandler';
+import FocusOutHandler from './FocusOutHandler';
import OutsideClickHandler from './OutsideClickHandler';
import { Popup } from './popups';
@@ -36,7 +37,9 @@ export function DropdownToggler(props: Props) {
overlay={
open ? (
<OutsideClickHandler onClickOutside={onRequestClose}>
- <EscKeydownHandler onKeydown={onRequestClose}>{overlay}</EscKeydownHandler>
+ <FocusOutHandler onFocusOut={onRequestClose}>
+ <EscKeydownHandler onKeydown={onRequestClose}>{overlay}</EscKeydownHandler>
+ </FocusOutHandler>
</OutsideClickHandler>
) : undefined
}
diff --git a/server/sonar-web/design-system/src/components/FocusOutHandler.tsx b/server/sonar-web/design-system/src/components/FocusOutHandler.tsx
new file mode 100644
index 00000000000..dbd700e633a
--- /dev/null
+++ b/server/sonar-web/design-system/src/components/FocusOutHandler.tsx
@@ -0,0 +1,60 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2023 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+import * as React from 'react';
+
+interface Props extends React.BaseHTMLAttributes<HTMLDivElement> {
+ innerRef?: (instance: HTMLDivElement) => void;
+ onFocusOut: () => void;
+}
+
+export default class FocusOutHandler extends React.PureComponent<React.PropsWithChildren<Props>> {
+ ref?: HTMLDivElement;
+
+ componentDidMount() {
+ setTimeout(() => {
+ document.addEventListener('focusin', this.handleFocusOut);
+ }, 0);
+ }
+
+ componentWillUnmount() {
+ document.removeEventListener('focusin', this.handleFocusOut);
+ }
+
+ nodeRef = (node: HTMLDivElement) => {
+ const { innerRef } = this.props;
+ this.ref = node;
+ innerRef?.(node);
+ };
+
+ handleFocusOut = () => {
+ if (this.ref?.querySelector(':focus') === null) {
+ this.props.onFocusOut();
+ }
+ };
+
+ render() {
+ const { onFocusOut, innerRef, children, ...props } = this.props;
+ return (
+ <div ref={this.nodeRef} {...props}>
+ {children}
+ </div>
+ );
+ }
+}
diff --git a/server/sonar-web/design-system/src/components/InputSearch.tsx b/server/sonar-web/design-system/src/components/InputSearch.tsx
index d87c3844468..0bf61f9cae1 100644
--- a/server/sonar-web/design-system/src/components/InputSearch.tsx
+++ b/server/sonar-web/design-system/src/components/InputSearch.tsx
@@ -96,7 +96,7 @@ export function InputSearch({
if (autoFocus && input.current) {
input.current.focus();
}
- });
+ }, [autoFocus]);
const changeValue = (newValue: string) => {
if (newValue.length === 0 || !minLength || minLength <= newValue.length) {
diff --git a/server/sonar-web/design-system/src/components/MultiSelect.tsx b/server/sonar-web/design-system/src/components/MultiSelect.tsx
new file mode 100644
index 00000000000..33b2037afe1
--- /dev/null
+++ b/server/sonar-web/design-system/src/components/MultiSelect.tsx
@@ -0,0 +1,336 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2023 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+import classNames from 'classnames';
+import { difference } from 'lodash';
+import { PureComponent } from 'react';
+import { Key } from '../helpers/keyboard';
+import { ItemDivider, ItemHeader } from './DropdownMenu';
+import { InputSearch } from './InputSearch';
+import { MultiSelectOption } from './MultiSelectOption';
+
+interface Props {
+ allowNewElements?: boolean;
+ allowSelection?: boolean;
+ clearIconAriaLabel: string;
+ createElementLabel: string;
+ elements: string[];
+ footerNode?: React.ReactNode;
+ headerNode?: React.ReactNode;
+ listSize: number;
+ noResultsLabel: string;
+ onSearch: (query: string) => Promise<void>;
+ onSelect: (item: string) => void;
+ onUnselect: (item: string) => void;
+ placeholder: string;
+ searchInputAriaLabel: string;
+ selectedElements: string[];
+ validateSearchInput?: (value: string) => string;
+}
+
+interface State {
+ activeIdx: number;
+ loading: boolean;
+ query: string;
+ selectedElements: string[];
+ unselectedElements: string[];
+}
+
+interface DefaultProps {
+ filterSelected: (query: string, selectedElements: string[]) => string[];
+ renderLabel: (element: string) => React.ReactNode;
+ validateSearchInput: (value: string) => string;
+}
+
+type PropsWithDefault = Props & DefaultProps;
+
+export class MultiSelect extends PureComponent<Props, State> {
+ container?: HTMLDivElement | null;
+ searchInput?: HTMLInputElement | null;
+ mounted = false;
+
+ static defaultProps: DefaultProps = {
+ filterSelected: (query: string, selectedElements: string[]) =>
+ selectedElements.filter((elem) => elem.includes(query)),
+ renderLabel: (element: string) => element,
+ validateSearchInput: (value: string) => value,
+ };
+
+ constructor(props: Props) {
+ super(props);
+ this.state = {
+ activeIdx: 0,
+ loading: true,
+ query: '',
+ selectedElements: [],
+ unselectedElements: [],
+ };
+ }
+
+ componentDidMount() {
+ this.mounted = true;
+ this.onSearchQuery('');
+ this.updateSelectedElements(this.props as PropsWithDefault);
+ this.updateUnselectedElements();
+ if (this.container) {
+ this.container.addEventListener('keydown', this.handleKeyboard, true);
+ }
+ }
+
+ componentDidUpdate(prevProps: Props) {
+ if (this.searchInput) {
+ this.searchInput.focus();
+ }
+
+ if (
+ prevProps.elements !== this.props.elements ||
+ prevProps.selectedElements !== this.props.selectedElements
+ ) {
+ this.updateSelectedElements(this.props as PropsWithDefault);
+ this.updateUnselectedElements();
+
+ const totalElements = this.getAllElements(this.props, this.state).length;
+
+ if (this.state.activeIdx >= totalElements) {
+ this.setState({ activeIdx: totalElements - 1 });
+ }
+ }
+ }
+
+ componentWillUnmount() {
+ this.mounted = false;
+ if (this.container) {
+ this.container.removeEventListener('keydown', this.handleKeyboard);
+ }
+ }
+
+ handleSelectChange = (selected: boolean, item: string) => {
+ if (selected) {
+ this.onSelectItem(item);
+ } else {
+ this.onUnselectItem(item);
+ }
+ };
+
+ handleSearchChange = (value: string) => {
+ this.onSearchQuery((this.props as PropsWithDefault).validateSearchInput(value));
+ };
+
+ handleElementHover = (element: string) => {
+ this.setState((prevState, props) => {
+ return { activeIdx: this.getAllElements(props, prevState).indexOf(element) };
+ });
+ };
+
+ handleKeyboard = (evt: KeyboardEvent) => {
+ switch (evt.key) {
+ case Key.ArrowDown:
+ evt.stopPropagation();
+ evt.preventDefault();
+ this.setState(this.selectNextElement);
+ break;
+ case Key.ArrowUp:
+ evt.stopPropagation();
+ evt.preventDefault();
+ this.setState(this.selectPreviousElement);
+ break;
+ case Key.ArrowLeft:
+ case Key.ArrowRight:
+ evt.stopPropagation();
+ break;
+ case Key.Enter: {
+ const allElements = this.getAllElements(this.props, this.state);
+ if (this.state.activeIdx >= 0 && this.state.activeIdx < allElements.length) {
+ this.toggleSelect(allElements[this.state.activeIdx]);
+ }
+ break;
+ }
+ }
+ };
+
+ onSearchQuery = (query: string) => {
+ this.setState({ activeIdx: 0, loading: true, query });
+ this.props.onSearch(query).then(this.stopLoading, this.stopLoading);
+ };
+
+ onSelectItem = (item: string) => {
+ if (this.isNewElement(item, this.props)) {
+ this.onSearchQuery('');
+ }
+ this.props.onSelect(item);
+ };
+
+ onUnselectItem = (item: string) => this.props.onUnselect(item);
+
+ isNewElement = (elem: string, { selectedElements, elements }: Props) =>
+ elem.length > 0 && selectedElements.indexOf(elem) === -1 && elements.indexOf(elem) === -1;
+
+ updateSelectedElements = (props: PropsWithDefault) => {
+ this.setState((state: State) => {
+ if (state.query) {
+ return {
+ selectedElements: props.filterSelected(state.query, props.selectedElements),
+ };
+ }
+ return { selectedElements: [...props.selectedElements] };
+ });
+ };
+
+ updateUnselectedElements = () => {
+ const { listSize } = this.props;
+ this.setState((state: State) => {
+ if (listSize === 0) {
+ return { unselectedElements: difference(this.props.elements, this.props.selectedElements) };
+ } else if (listSize < state.selectedElements.length) {
+ return { unselectedElements: [] };
+ }
+ return {
+ unselectedElements: difference(this.props.elements, this.props.selectedElements).slice(
+ 0,
+ listSize - state.selectedElements.length
+ ),
+ };
+ });
+ };
+
+ getAllElements = (props: Props, state: State) => {
+ const { allowNewElements = true } = props;
+ if (allowNewElements && this.isNewElement(state.query, props)) {
+ return [...state.selectedElements, ...state.unselectedElements, state.query];
+ }
+ return [...state.selectedElements, ...state.unselectedElements];
+ };
+
+ selectNextElement = (state: State, props: Props) => {
+ const { activeIdx } = state;
+ const allElements = this.getAllElements(props, state);
+ if (activeIdx < 0 || activeIdx >= allElements.length - 1) {
+ return { activeIdx: 0 };
+ }
+ return { activeIdx: activeIdx + 1 };
+ };
+
+ selectPreviousElement = (state: State, props: Props) => {
+ const { activeIdx } = state;
+ const allElements = this.getAllElements(props, state);
+ if (activeIdx <= 0) {
+ const lastIdx = allElements.length - 1;
+ return { activeIdx: lastIdx };
+ }
+ return { activeIdx: activeIdx - 1 };
+ };
+
+ stopLoading = () => {
+ if (this.mounted) {
+ this.setState({ loading: false });
+ }
+ };
+
+ toggleSelect = (item: string) => {
+ if (!this.props.selectedElements.includes(item)) {
+ this.onSelectItem(item);
+ this.setState(this.selectNextElement);
+ } else {
+ this.onUnselectItem(item);
+ }
+ };
+
+ render() {
+ const {
+ allowSelection = true,
+ allowNewElements = true,
+ createElementLabel,
+ headerNode = '',
+ footerNode = '',
+ clearIconAriaLabel,
+ noResultsLabel,
+ searchInputAriaLabel,
+ } = this.props;
+ const { query, activeIdx, selectedElements, unselectedElements } = this.state;
+ const activeElement = this.getAllElements(this.props, this.state)[activeIdx];
+ const showNewElement = allowNewElements && this.isNewElement(query, this.props);
+ const isFixedHeight = this.props.listSize === 0;
+ const hasFooter = Boolean(footerNode);
+
+ return (
+ <div ref={(div) => (this.container = div)}>
+ <div className="sw-px-3">
+ <InputSearch
+ autoFocus={true}
+ className="sw-mt-1"
+ clearIconAriaLabel={clearIconAriaLabel}
+ loading={this.state.loading}
+ onChange={this.handleSearchChange}
+ placeholder={this.props.placeholder}
+ searchInputAriaLabel={searchInputAriaLabel}
+ size="full"
+ value={query}
+ />
+ </div>
+ <ItemHeader>{headerNode}</ItemHeader>
+ <ul
+ className={classNames('sw-mt-2', {
+ 'sw-max-h-abs-200 sw-overflow-y-auto': isFixedHeight,
+ })}
+ >
+ {selectedElements.length > 0 &&
+ selectedElements.map((element) => (
+ <MultiSelectOption
+ active={activeElement === element}
+ createElementLabel={createElementLabel}
+ element={element}
+ key={element}
+ onHover={this.handleElementHover}
+ onSelectChange={this.handleSelectChange}
+ selected={true}
+ />
+ ))}
+ {unselectedElements.length > 0 &&
+ unselectedElements.map((element) => (
+ <MultiSelectOption
+ active={activeElement === element}
+ createElementLabel={createElementLabel}
+ disabled={!allowSelection}
+ element={element}
+ key={element}
+ onHover={this.handleElementHover}
+ onSelectChange={this.handleSelectChange}
+ />
+ ))}
+ {showNewElement && (
+ <MultiSelectOption
+ active={activeElement === query}
+ createElementLabel={createElementLabel}
+ custom={true}
+ element={query}
+ key={query}
+ onHover={this.handleElementHover}
+ onSelectChange={this.handleSelectChange}
+ />
+ )}
+ {!showNewElement && selectedElements.length < 1 && unselectedElements.length < 1 && (
+ <li className="sw-ml-2">{noResultsLabel}</li>
+ )}
+ </ul>
+ {hasFooter && <ItemDivider className="sw-mt-2" />}
+ <div className="sw-px-3">{footerNode}</div>
+ </div>
+ );
+ }
+}
diff --git a/server/sonar-web/design-system/src/components/MultiSelectOption.tsx b/server/sonar-web/design-system/src/components/MultiSelectOption.tsx
new file mode 100644
index 00000000000..6258187395e
--- /dev/null
+++ b/server/sonar-web/design-system/src/components/MultiSelectOption.tsx
@@ -0,0 +1,64 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2023 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+import classNames from 'classnames';
+import { ItemCheckbox } from './DropdownMenu';
+
+export interface MultiSelectOptionProps {
+ active?: boolean;
+ createElementLabel: string;
+ custom?: boolean;
+ disabled?: boolean;
+ element: string;
+ onHover: (element: string) => void;
+ onSelectChange: (selected: boolean, element: string) => void;
+ selected?: boolean;
+}
+
+export function MultiSelectOption(props: MultiSelectOptionProps) {
+ const { active, createElementLabel, custom, disabled, element, onSelectChange, selected } = props;
+ const onHover = () => props.onHover(element);
+
+ return (
+ <ItemCheckbox
+ checked={Boolean(selected)}
+ className={classNames('sw-flex sw-py-2 sw-px-4', { active })}
+ disabled={disabled}
+ id={element}
+ onCheck={onSelectChange}
+ onFocus={onHover}
+ onPointerEnter={onHover}
+ >
+ {custom ? (
+ <span
+ aria-label={`${createElementLabel}: ${element}`}
+ className="sw-ml-3"
+ title={createElementLabel}
+ >
+ <span aria-hidden={true} className="sw-mr-1">
+ +
+ </span>
+ {element}
+ </span>
+ ) : (
+ <span className="sw-ml-3">{element}</span>
+ )}
+ </ItemCheckbox>
+ );
+}
diff --git a/server/sonar-web/design-system/src/components/Tags.tsx b/server/sonar-web/design-system/src/components/Tags.tsx
new file mode 100644
index 00000000000..8d312aea879
--- /dev/null
+++ b/server/sonar-web/design-system/src/components/Tags.tsx
@@ -0,0 +1,113 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2023 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+import styled from '@emotion/styled';
+import classNames from 'classnames';
+import * as React from 'react';
+import tw from 'twin.macro';
+import { PopupPlacement, PopupZLevel } from '../helpers';
+import { themeColor, themeContrast } from '../helpers/theme';
+import { Dropdown } from './Dropdown';
+import { LightLabel } from './Text';
+import { WrapperButton } from './buttons';
+
+interface Props {
+ allowUpdate?: boolean;
+ ariaTagsListLabel: string;
+ className?: string;
+ emptyText: string;
+ menuId?: string;
+ overlay?: React.ReactNode;
+ popupPlacement?: PopupPlacement;
+ tags: string[];
+ tagsToDisplay?: number;
+}
+
+export function Tags({
+ allowUpdate = false,
+ ariaTagsListLabel,
+ className,
+ emptyText,
+ menuId = '',
+ overlay,
+ popupPlacement,
+ tags,
+ tagsToDisplay = 3,
+}: Props) {
+ const displayedTags = tags.slice(0, tagsToDisplay);
+ const extraTags = tags.slice(tagsToDisplay);
+
+ const displayedTagsContent = () => (
+ <span className="sw-inline-flex sw-items-center sw-gap-1" title={tags.join(', ')}>
+ {/* Display first 3 (tagsToDisplay) tags */}
+ {displayedTags.map((tag) => (
+ <TagLabel key={tag}>{tag}</TagLabel>
+ ))}
+
+ {/* Show ellipsis if there are more tags */}
+ {extraTags.length > 0 ? <TagLabel>...</TagLabel> : null}
+
+ {/* Handle no tags with its own styling */}
+ {tags.length === 0 && <LightLabel>{emptyText}</LightLabel>}
+ </span>
+ );
+
+ return (
+ <span
+ aria-label={`${ariaTagsListLabel}: ${tags.join(', ')}`}
+ className={classNames('sw-cursor-default sw-flex sw-items-center', className)}
+ >
+ {allowUpdate ? (
+ <Dropdown
+ allowResizing={true}
+ closeOnClick={false}
+ id={menuId}
+ overlay={overlay}
+ placement={popupPlacement}
+ zLevel={PopupZLevel.Global}
+ >
+ {({ a11yAttrs, onToggleClick }) => (
+ <WrapperButton
+ className="sw-flex sw-items-center sw-gap-1 sw-p-1 sw-h-auto sw-rounded-0"
+ onClick={onToggleClick}
+ {...a11yAttrs}
+ >
+ {displayedTagsContent()}
+ <TagLabel className="sw-cursor-pointer">+</TagLabel>
+ </WrapperButton>
+ )}
+ </Dropdown>
+ ) : (
+ <span>{displayedTagsContent()}</span>
+ )}
+ </span>
+ );
+}
+
+const TagLabel = styled.span`
+ color: ${themeContrast('tag')};
+ background: ${themeColor('tag')};
+
+ ${tw`sw-body-sm`}
+ ${tw`sw-box-border`}
+ ${tw`sw-truncate`}
+ ${tw`sw-rounded-1/2`}
+ ${tw`sw-px-1 sw-py-1/2`}
+ ${tw`sw-max-w-32`}
+`;
diff --git a/server/sonar-web/design-system/src/components/TagsSelector.tsx b/server/sonar-web/design-system/src/components/TagsSelector.tsx
new file mode 100644
index 00000000000..5c4ad8d52b3
--- /dev/null
+++ b/server/sonar-web/design-system/src/components/TagsSelector.tsx
@@ -0,0 +1,70 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2023 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+import { MultiSelect } from './MultiSelect';
+
+interface Props {
+ clearIconAriaLabel: string;
+ createElementLabel: string;
+ headerLabel: string;
+ listSize: number;
+ noResultsLabel: string;
+ onSearch: (query: string) => Promise<void>;
+ onSelect: (item: string) => void;
+ onUnselect: (item: string) => void;
+ searchInputAriaLabel: string;
+ selectedTags: string[];
+ tags: string[];
+}
+
+export function TagsSelector(props: Props) {
+ const {
+ clearIconAriaLabel,
+ createElementLabel,
+ headerLabel,
+ listSize,
+ noResultsLabel,
+ searchInputAriaLabel,
+ selectedTags,
+ tags,
+ } = props;
+
+ return (
+ <MultiSelect
+ clearIconAriaLabel={clearIconAriaLabel}
+ createElementLabel={createElementLabel}
+ elements={tags}
+ headerNode={<div className="sw-mt-4 sw-font-semibold">{headerLabel}</div>}
+ listSize={listSize}
+ noResultsLabel={noResultsLabel}
+ onSearch={props.onSearch}
+ onSelect={props.onSelect}
+ onUnselect={props.onUnselect}
+ placeholder={searchInputAriaLabel}
+ searchInputAriaLabel={searchInputAriaLabel}
+ selectedElements={selectedTags}
+ validateSearchInput={validateTag}
+ />
+ );
+}
+
+export function validateTag(value: string) {
+ // Allow only a-z, 0-9, '+', '-', '#', '.'
+ return value.toLowerCase().replace(/[^-a-z0-9+#.]/gi, '');
+}
diff --git a/server/sonar-web/design-system/src/components/__tests__/MultiSelect-test.tsx b/server/sonar-web/design-system/src/components/__tests__/MultiSelect-test.tsx
new file mode 100644
index 00000000000..2e79eb02cd0
--- /dev/null
+++ b/server/sonar-web/design-system/src/components/__tests__/MultiSelect-test.tsx
@@ -0,0 +1,113 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2023 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+import { render, screen } from '@testing-library/react';
+import userEvent from '@testing-library/user-event';
+import { MultiSelect } from '../MultiSelect';
+
+const elements = ['foo', 'bar', 'baz'];
+
+beforeAll(() => {
+ jest.useFakeTimers();
+});
+
+afterAll(() => {
+ jest.useRealTimers();
+});
+
+it('should allow selecting and deselecting a new option', async () => {
+ const user = userEvent.setup({ delay: null });
+ const onSelect = jest.fn();
+ const onUnselect = jest.fn();
+ renderMultiselect({ elements, onSelect, onUnselect, allowNewElements: true });
+
+ await user.keyboard('new option');
+ jest.runAllTimers(); // skip the debounce
+
+ expect(screen.getByText('new option')).toBeInTheDocument();
+
+ await user.click(screen.getByText('new option'));
+
+ expect(onSelect).toHaveBeenCalledWith('new option');
+
+ renderMultiselect({
+ elements,
+ onUnselect,
+ allowNewElements: true,
+ selectedElements: ['new option'],
+ });
+
+ await user.click(screen.getByText('new option'));
+ expect(onUnselect).toHaveBeenCalledWith('new option');
+});
+
+it('should ignore the left and right arrow keys', async () => {
+ const user = userEvent.setup({ delay: null });
+ const onSelect = jest.fn();
+ renderMultiselect({ elements, onSelect });
+
+ /* eslint-disable testing-library/no-node-access */
+ await user.keyboard('{arrowdown}');
+ expect(screen.getAllByRole('checkbox')[1].parentElement).toHaveClass('active');
+
+ await user.keyboard('{arrowleft}');
+ expect(screen.getAllByRole('checkbox')[1].parentElement).toHaveClass('active');
+
+ await user.keyboard('{arrowright}');
+ expect(screen.getAllByRole('checkbox')[1].parentElement).toHaveClass('active');
+
+ await user.keyboard('{arrowdown}');
+ expect(screen.getAllByRole('checkbox')[1].parentElement).not.toHaveClass('active');
+ expect(screen.getAllByRole('checkbox')[2].parentElement).toHaveClass('active');
+
+ await user.keyboard('{arrowup}');
+ await user.keyboard('{arrowup}');
+ expect(screen.getAllByRole('checkbox')[0].parentElement).toHaveClass('active');
+ await user.keyboard('{arrowup}');
+ expect(screen.getAllByRole('checkbox')[2].parentElement).toHaveClass('active');
+
+ expect(screen.getAllByRole('checkbox')[2]).not.toBeChecked();
+ await user.keyboard('{enter}');
+ expect(onSelect).toHaveBeenCalledWith('baz');
+});
+
+it('should show no results', () => {
+ renderMultiselect();
+ expect(screen.getByText('no results')).toBeInTheDocument();
+});
+
+function renderMultiselect(props: Partial<MultiSelect['props']> = {}) {
+ return render(
+ <MultiSelect
+ clearIconAriaLabel="clear"
+ createElementLabel="create thing"
+ elements={[]}
+ filterSelected={jest.fn()}
+ listSize={10}
+ noResultsLabel="no results"
+ onSearch={jest.fn(() => Promise.resolve())}
+ onSelect={jest.fn()}
+ onUnselect={jest.fn()}
+ placeholder=""
+ searchInputAriaLabel="search"
+ selectedElements={[]}
+ {...props}
+ />
+ );
+}
diff --git a/server/sonar-web/design-system/src/components/__tests__/Tags-test.tsx b/server/sonar-web/design-system/src/components/__tests__/Tags-test.tsx
new file mode 100644
index 00000000000..4749a34cb40
--- /dev/null
+++ b/server/sonar-web/design-system/src/components/__tests__/Tags-test.tsx
@@ -0,0 +1,116 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2023 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+
+/* eslint-disable import/no-extraneous-dependencies */
+
+import { render, screen } from '@testing-library/react';
+import userEvent from '@testing-library/user-event';
+import { useState } from 'react';
+import { FCProps } from '../../types/misc';
+import { Tags } from '../Tags';
+import { TagsSelector } from '../TagsSelector';
+
+it('should display "no tags"', () => {
+ renderTags({ tags: [] });
+
+ expect(screen.getByText('no tags')).toBeInTheDocument();
+});
+
+it('should display tags', () => {
+ const tags = ['tag1', 'tag2'];
+ renderTags({ tags });
+
+ expect(screen.getByText('tag1')).toBeInTheDocument();
+ expect(screen.getByText('tag2')).toBeInTheDocument();
+});
+
+it('should handle more tags than the max to display', () => {
+ const tags = ['tag1', 'tag2', 'tag3'];
+ renderTags({ tags, tagsToDisplay: 2 });
+
+ expect(screen.getByText('tag1')).toBeInTheDocument();
+ expect(screen.getByText('tag2')).toBeInTheDocument();
+ expect(screen.queryByText('tag3')).not.toBeInTheDocument();
+ expect(screen.getByText('...')).toBeInTheDocument();
+});
+
+it('should allow editing tags', async () => {
+ const user = userEvent.setup();
+ renderTags({ allowUpdate: true });
+
+ const plusButton = screen.getByText('+');
+ expect(plusButton).toBeInTheDocument();
+ await user.click(plusButton);
+
+ expect(await screen.findByLabelText('search')).toBeInTheDocument();
+ await user.click(screen.getByText('tag3'));
+ await user.keyboard('{Escape}');
+
+ expect(screen.getByText('tag1')).toBeInTheDocument();
+ expect(screen.getByText('tag3')).toBeInTheDocument();
+
+ await user.click(plusButton);
+ await user.click(screen.getByRole('checkbox', { name: 'tag1' }));
+ await user.keyboard('{Escape}');
+
+ expect(screen.queryByText('tag1')).not.toBeInTheDocument();
+ expect(screen.getByText('tag3')).toBeInTheDocument();
+});
+
+function renderTags(overrides: Partial<FCProps<typeof Tags>> = {}) {
+ render(<Wrapper {...overrides} />);
+}
+
+function Wrapper(overrides: Partial<FCProps<typeof Tags>> = {}) {
+ const [selectedTags, setSelectedTags] = useState<string[]>(overrides.tags ?? ['tag1']);
+
+ const overlay = (
+ <TagsSelector
+ clearIconAriaLabel="clear"
+ createElementLabel="create new tag"
+ headerLabel="edit tags"
+ listSize={4}
+ noResultsLabel="no results"
+ onSearch={jest.fn().mockResolvedValue(undefined)}
+ onSelect={(tag) => {
+ setSelectedTags([...selectedTags, tag]);
+ }}
+ onUnselect={(tag) => {
+ const i = selectedTags.indexOf(tag);
+ if (i > -1) {
+ setSelectedTags([...selectedTags.slice(0, i), ...selectedTags.slice(i + 1)]);
+ }
+ }}
+ searchInputAriaLabel="search"
+ selectedTags={selectedTags}
+ tags={['tag1', 'tag2', 'tag3']}
+ />
+ );
+
+ return (
+ <Tags
+ ariaTagsListLabel="list"
+ emptyText="no tags"
+ overlay={overlay}
+ tags={selectedTags}
+ {...overrides}
+ />
+ );
+}
diff --git a/server/sonar-web/design-system/src/components/buttons.tsx b/server/sonar-web/design-system/src/components/buttons.tsx
index 13304f9215c..18030488b64 100644
--- a/server/sonar-web/design-system/src/components/buttons.tsx
+++ b/server/sonar-web/design-system/src/components/buttons.tsx
@@ -190,6 +190,14 @@ export const DangerButtonSecondary: React.FC<ButtonProps> = styled(Button)`
--border: ${themeBorder('default', 'dangerButtonSecondaryBorder')};
`;
+export const WrapperButton: React.FC<ButtonProps> = styled(Button)`
+ --background: none;
+ --backgroundHover: none;
+ --color: none;
+ --focus: ${themeColor('button', OPACITY_20_PERCENT)};
+ --border: none;
+`;
+
interface ThirdPartyProps extends Omit<ButtonProps, 'Icon'> {
iconPath: string;
name: string;
diff --git a/server/sonar-web/design-system/src/components/index.ts b/server/sonar-web/design-system/src/components/index.ts
index 3a244823b8f..a4af02dcab0 100644
--- a/server/sonar-web/design-system/src/components/index.ts
+++ b/server/sonar-web/design-system/src/components/index.ts
@@ -51,6 +51,8 @@ export * from './SelectionCard';
export * from './Separator';
export * from './SizeIndicator';
export * from './SonarQubeLogo';
+export * from './Tags';
+export * from './TagsSelector';
export * from './Text';
export { ToggleButton } from './ToggleButton';
export { TopBar } from './TopBar';