Merge pull request #858 from acelaya-forks/feature/react-tag-autocomplete-7

Update to react-tag-autocomplete 7
This commit is contained in:
Alejandro Celaya 2023-08-13 17:02:53 +02:00 committed by GitHub
commit 8aa149cac4
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 183 additions and 116 deletions

View file

@ -11,6 +11,5 @@ jobs:
ci: ci:
uses: shlinkio/github-actions/.github/workflows/web-app-ci.yml@main uses: shlinkio/github-actions/.github/workflows/web-app-ci.yml@main
with: with:
node-version: 20.2 node-version: 20.5
publish-coverage: true publish-coverage: true
force-install: true

View file

@ -5,7 +5,7 @@ on:
jobs: jobs:
deploy: deploy:
runs-on: ubuntu-20.04 runs-on: ubuntu-22.04
continue-on-error: true continue-on-error: true
steps: steps:
- name: Checkout code - name: Checkout code
@ -16,10 +16,10 @@ jobs:
- name: Use node.js - name: Use node.js
uses: actions/setup-node@v3 uses: actions/setup-node@v3
with: with:
node-version: 20.2 node-version: 20.5
- name: Build - name: Build
run: | run: |
npm ci --force && \ npm ci && \
node ./scripts/set-homepage.js /shlink-web-client/${GITHUB_HEAD_REF#refs/heads/} && \ node ./scripts/set-homepage.js /shlink-web-client/${GITHUB_HEAD_REF#refs/heads/} && \
npm run build npm run build
- name: Deploy preview - name: Deploy preview

View file

@ -7,16 +7,16 @@ on:
jobs: jobs:
build: build:
runs-on: ubuntu-20.04 runs-on: ubuntu-22.04
steps: steps:
- name: Checkout code - name: Checkout code
uses: actions/checkout@v3 uses: actions/checkout@v3
- name: Use node.js - name: Use node.js
uses: actions/setup-node@v3 uses: actions/setup-node@v3
with: with:
node-version: 20.2 node-version: 20.5
- name: Generate release assets - name: Generate release assets
run: npm ci --force && VERSION=${GITHUB_REF#refs/tags/v} npm run build:dist run: npm ci && VERSION=${GITHUB_REF#refs/tags/v} npm run build:dist
- name: Publish release with assets - name: Publish release with assets
uses: docker://antonyurchenko/git-release:latest uses: docker://antonyurchenko/git-release:latest
env: env:

View file

@ -1,8 +1,8 @@
FROM node:20.2-alpine as node FROM node:20.5-alpine as node
COPY . /shlink-web-client COPY . /shlink-web-client
ARG VERSION="latest" ARG VERSION="latest"
ENV VERSION ${VERSION} ENV VERSION ${VERSION}
RUN cd /shlink-web-client && npm ci --force && npm run build RUN cd /shlink-web-client && npm ci && npm run build
FROM nginx:1.23-alpine FROM nginx:1.23-alpine
LABEL maintainer="Alejandro Celaya <alejandro@alejandrocelaya.com>" LABEL maintainer="Alejandro Celaya <alejandro@alejandrocelaya.com>"

View file

@ -25,3 +25,4 @@ afterEach(() => {
(global as any).scrollTo = () => {}; (global as any).scrollTo = () => {};
(global as any).prompt = () => {}; (global as any).prompt = () => {};
(global as any).matchMedia = (media: string) => ({ matches: false, media }); (global as any).matchMedia = (media: string) => ({ matches: false, media });
(global as any).HTMLElement.prototype.scrollIntoView = () => {};

View file

@ -3,8 +3,8 @@ version: '3'
services: services:
shlink_web_client_node: shlink_web_client_node:
container_name: shlink_web_client_node container_name: shlink_web_client_node
image: node:20.2-alpine image: node:20.5-alpine
command: /bin/sh -c "cd /home/shlink/www && npm install --force && npm run start" command: /bin/sh -c "cd /home/shlink/www && npm install && npm run start"
volumes: volumes:
- ./:/home/shlink/www - ./:/home/shlink/www
ports: ports:

20
package-lock.json generated
View file

@ -39,7 +39,7 @@
"react-redux": "^8.1.2", "react-redux": "^8.1.2",
"react-router-dom": "^6.14.2", "react-router-dom": "^6.14.2",
"react-swipeable": "^7.0.1", "react-swipeable": "^7.0.1",
"react-tag-autocomplete": "^6.3.0", "react-tag-autocomplete": "^7.0.0",
"reactstrap": "^9.2.0", "reactstrap": "^9.2.0",
"redux-localstorage-simple": "^2.5.1", "redux-localstorage-simple": "^2.5.1",
"uuid": "^9.0.0", "uuid": "^9.0.0",
@ -8900,16 +8900,14 @@
} }
}, },
"node_modules/react-tag-autocomplete": { "node_modules/react-tag-autocomplete": {
"version": "6.3.0", "version": "7.0.0",
"resolved": "https://registry.npmjs.org/react-tag-autocomplete/-/react-tag-autocomplete-6.3.0.tgz", "resolved": "https://registry.npmjs.org/react-tag-autocomplete/-/react-tag-autocomplete-7.0.0.tgz",
"integrity": "sha512-MUBVUFh5eOqshUm5NM20qp7zXk8TzSiKO4GoktlFzBLIOLs356npaMKtL02bm0nFV2f1zInUrXn1fq6+i5YX0w==", "integrity": "sha512-PFxT7fpMB8Au+S9cJYAGRVTnacZpeXybc5SkpTCyuJHmUN1Bt8gHb9vZi3f+aX/eDX44x2WIwYiqfRBi2E5AMg==",
"engines": { "engines": {
"node": ">= 10.0.0" "node": ">= 16.12.0"
}, },
"peerDependencies": { "peerDependencies": {
"prop-types": "^15.5.0", "react": "^18.0.0"
"react": "^16.5.0 || ^17.0.0",
"react-dom": "^16.5.0 || ^17.0.0"
} }
}, },
"node_modules/react-transition-group": { "node_modules/react-transition-group": {
@ -17211,9 +17209,9 @@
"requires": {} "requires": {}
}, },
"react-tag-autocomplete": { "react-tag-autocomplete": {
"version": "6.3.0", "version": "7.0.0",
"resolved": "https://registry.npmjs.org/react-tag-autocomplete/-/react-tag-autocomplete-6.3.0.tgz", "resolved": "https://registry.npmjs.org/react-tag-autocomplete/-/react-tag-autocomplete-7.0.0.tgz",
"integrity": "sha512-MUBVUFh5eOqshUm5NM20qp7zXk8TzSiKO4GoktlFzBLIOLs356npaMKtL02bm0nFV2f1zInUrXn1fq6+i5YX0w==", "integrity": "sha512-PFxT7fpMB8Au+S9cJYAGRVTnacZpeXybc5SkpTCyuJHmUN1Bt8gHb9vZi3f+aX/eDX44x2WIwYiqfRBi2E5AMg==",
"requires": {} "requires": {}
}, },
"react-transition-group": { "react-transition-group": {

View file

@ -55,7 +55,7 @@
"react-redux": "^8.1.2", "react-redux": "^8.1.2",
"react-router-dom": "^6.14.2", "react-router-dom": "^6.14.2",
"react-swipeable": "^7.0.1", "react-swipeable": "^7.0.1",
"react-tag-autocomplete": "^6.3.0", "react-tag-autocomplete": "^7.0.0",
"reactstrap": "^9.2.0", "reactstrap": "^9.2.0",
"redux-localstorage-simple": "^2.5.1", "redux-localstorage-simple": "^2.5.1",
"uuid": "^9.0.0", "uuid": "^9.0.0",

View file

@ -4,7 +4,7 @@ import { faDesktop } from '@fortawesome/free-solid-svg-icons';
import { Checkbox, SimpleCard } from '@shlinkio/shlink-frontend-kit'; import { Checkbox, SimpleCard } from '@shlinkio/shlink-frontend-kit';
import classNames from 'classnames'; import classNames from 'classnames';
import { parseISO } from 'date-fns'; import { parseISO } from 'date-fns';
import { isEmpty, pipe, replace, trim } from 'ramda'; import { isEmpty } from 'ramda';
import type { ChangeEvent, FC } from 'react'; import type { ChangeEvent, FC } from 'react';
import { useEffect, useState } from 'react'; import { useEffect, useState } from 'react';
import { Button, FormGroup, Input, Row } from 'reactstrap'; import { Button, FormGroup, Input, Row } from 'reactstrap';
@ -35,7 +35,6 @@ export interface ShortUrlFormProps<T extends ShlinkCreateShortUrlData | ShlinkEd
onSave: (shortUrlData: T) => Promise<unknown>; onSave: (shortUrlData: T) => Promise<unknown>;
} }
const normalizeTag = pipe(trim, replace(/ /g, '-'));
const toDate = (date?: string | Date): Date | undefined => (typeof date === 'string' ? parseISO(date) : date); const toDate = (date?: string | Date): Date | undefined => (typeof date === 'string' ? parseISO(date) : date);
const isCreationData = (data: ShlinkCreateShortUrlData | ShlinkEditShortUrlData): data is ShlinkCreateShortUrlData => const isCreationData = (data: ShlinkCreateShortUrlData | ShlinkEditShortUrlData): data is ShlinkCreateShortUrlData =>
@ -54,7 +53,7 @@ export const ShortUrlForm = (
const isEdit = mode === 'edit'; const isEdit = mode === 'edit';
const isCreation = isCreationData(shortUrlData); const isCreation = isCreationData(shortUrlData);
const isBasicMode = mode === 'create-basic'; const isBasicMode = mode === 'create-basic';
const changeTags = (tags: string[]) => setShortUrlData({ ...shortUrlData, tags: tags.map(normalizeTag) }); const changeTags = (tags: string[]) => setShortUrlData({ ...shortUrlData, tags });
const setResettableValue = (value: string, initialValue?: any) => { const setResettableValue = (value: string, initialValue?: any) => {
if (hasValue(value)) { if (hasValue(value)) {
return value; return value;

View file

@ -1,70 +1,100 @@
import { useElementRef } from '@shlinkio/shlink-frontend-kit';
import classNames from 'classnames';
import { useEffect } from 'react'; import { useEffect } from 'react';
import type { SuggestionComponentProps, TagComponentProps } from 'react-tag-autocomplete'; import type { OptionRendererProps, ReactTagsAPI, TagRendererProps, TagSuggestion } from 'react-tag-autocomplete';
import ReactTags from 'react-tag-autocomplete'; import { ReactTags } from 'react-tag-autocomplete';
import type { ColorGenerator } from '../../utils/services/ColorGenerator'; import type { ColorGenerator } from '../../utils/services/ColorGenerator';
import { useSetting } from '../../utils/settings'; import { useSetting } from '../../utils/settings';
import type { TagsList } from '../reducers/tagsList'; import type { TagsList } from '../reducers/tagsList';
import { normalizeTag } from './index';
import { Tag } from './Tag'; import { Tag } from './Tag';
import { TagBullet } from './TagBullet'; import { TagBullet } from './TagBullet';
export interface TagsSelectorProps { export type TagsSelectorProps = {
selectedTags: string[]; selectedTags: string[];
onChange: (tags: string[]) => void; onChange: (tags: string[]) => void;
placeholder?: string; placeholder?: string;
allowNew?: boolean; allowNew?: boolean;
} };
interface TagsSelectorConnectProps extends TagsSelectorProps { type TagsSelectorConnectProps = TagsSelectorProps & {
listTags: () => void; listTags: () => void;
tagsList: TagsList; tagsList: TagsList;
} };
const toComponentTag = (tag: string) => ({ id: tag, name: tag }); const NOT_FOUND_TAG = 'Tag not found';
const NEW_TAG = 'Add tag';
const isSelectableOption = (tag: string) => tag !== NOT_FOUND_TAG;
const isNewOption = (tag: string) => tag === NEW_TAG;
const toTagObject = (tag: string): TagSuggestion => ({ label: tag, value: tag });
const buildTagRenderer = (colorGenerator: ColorGenerator) => ({ tag, onClick: deleteTag }: TagRendererProps) => (
<Tag colorGenerator={colorGenerator} text={tag.label} clearable className="react-tags__tag" onClose={deleteTag} />
);
const buildOptionRenderer = (colorGenerator: ColorGenerator, api: ReactTagsAPI | null) => (
{ option, classNames: classes, ...rest }: OptionRendererProps,
) => {
const isSelectable = isSelectableOption(option.label);
const isNew = isNewOption(option.label);
return (
<div
className={classNames(classes.option, {
[classes.optionIsActive]: isSelectable && option.active,
'react-tags__listbox-option--not-selectable': !isSelectable,
})}
{...rest}
>
{!isSelectable ? <i>{option.label}</i> : (
<>
{!isNew && <TagBullet tag={`${option.label}`} colorGenerator={colorGenerator} />}
{!isNew ? option.label : <i>Add &quot;{normalizeTag(api?.input.value ?? '')}&quot;</i>}
</>
)}
</div>
);
};
export const TagsSelector = (colorGenerator: ColorGenerator) => ( export const TagsSelector = (colorGenerator: ColorGenerator) => (
{ selectedTags, onChange, placeholder, listTags, tagsList, allowNew = true }: TagsSelectorConnectProps, { selectedTags, onChange, placeholder, listTags, tagsList, allowNew = true }: TagsSelectorConnectProps,
) => { ) => {
const shortUrlCreation = useSetting('shortUrlCreation');
useEffect(() => { useEffect(() => {
listTags(); listTags();
}, []); }, []);
const shortUrlCreation = useSetting('shortUrlCreation');
const searchMode = shortUrlCreation?.tagFilteringMode ?? 'startsWith'; const searchMode = shortUrlCreation?.tagFilteringMode ?? 'startsWith';
const ReactTagsTag = ({ tag, onDelete }: TagComponentProps) => const apiRef = useElementRef<ReactTagsAPI>();
<Tag colorGenerator={colorGenerator} text={tag.name} clearable className="react-tags__tag" onClose={onDelete} />;
const ReactTagsSuggestion = ({ item }: SuggestionComponentProps) => (
<>
<TagBullet tag={`${item.name}`} colorGenerator={colorGenerator} />
{item.name}
</>
);
return ( return (
<ReactTags <ReactTags
tags={selectedTags.map(toComponentTag)} ref={apiRef}
tagComponent={ReactTagsTag} selected={selectedTags.map(toTagObject)}
suggestions={tagsList.tags.filter((tag) => !selectedTags.includes(tag)).map(toComponentTag)} suggestions={tagsList.tags.filter((tag) => !selectedTags.includes(tag)).map(toTagObject)}
suggestionComponent={ReactTagsSuggestion} renderTag={buildTagRenderer(colorGenerator)}
renderOption={buildOptionRenderer(colorGenerator, apiRef.current)}
activateFirstOption
allowNew={allowNew} allowNew={allowNew}
addOnBlur newOptionText={NEW_TAG}
noOptionsText={NOT_FOUND_TAG}
placeholderText={placeholder ?? 'Add tags to the URL'} placeholderText={placeholder ?? 'Add tags to the URL'}
minQueryLength={1} delimiterKeys={['Enter', 'Tab', ',']}
delimiters={['Enter', 'Tab', ',']}
suggestionsTransform={ suggestionsTransform={
searchMode === 'includes' (query, suggestions) => {
? (query, suggestions) => suggestions.filter(({ name }) => name.includes(query)) const searchTerm = query.toLowerCase().trim();
: undefined return searchTerm.length < 1 ? [] : [...suggestions.filter(
({ label }) => (searchMode === 'includes' ? label.includes(searchTerm) : label.startsWith(searchTerm)),
)].slice(0, 5);
}
} }
onDelete={(removedTagIndex) => { onDelete={(removedTagIndex) => {
const tagsCopy = [...selectedTags]; const tagsCopy = [...selectedTags];
tagsCopy.splice(removedTagIndex, 1); tagsCopy.splice(removedTagIndex, 1);
onChange(tagsCopy); onChange(tagsCopy);
}} }}
onAddition={({ name: newTag }) => onChange( onAdd={({ label: newTag }) => onChange(
// * Avoid duplicated tags (thanks to the Set), // Split any of the new tags by comma, allowing to paste multiple comma-separated tags at once.
// * Split any of the new tags by comma, allowing to paste multiple comma-separated tags at once. [...selectedTags, ...newTag.split(',').map(normalizeTag)],
[...new Set([...selectedTags, ...newTag.toLowerCase().split(',')])],
)} )}
/> />
); );

View file

@ -0,0 +1,3 @@
const ONE_OR_MORE_SPACES_REGEX = /\s+/g;
export const normalizeTag = (tag: string) => tag.trim().toLowerCase().replace(ONE_OR_MORE_SPACES_REGEX, '-');

View file

@ -1,5 +1,6 @@
@import '../../../node_modules/@shlinkio/shlink-frontend-kit/dist/base'; @import '../../../node_modules/@shlinkio/shlink-frontend-kit/dist/base';
// Main wrapper
.react-tags { .react-tags {
position: relative; position: relative;
padding: 5px 0 0 6px; padding: 5px 0 0 6px;
@ -26,45 +27,35 @@
background-color: var(--input-color); background-color: var(--input-color);
} }
.react-tags.is-focused { // Mimic bootstrap input focus ring
.react-tags.is-active {
box-shadow: 0 0 0 .2rem rgb(70 150 229 / 25%); box-shadow: 0 0 0 .2rem rgb(70 150 229 / 25%);
} }
.react-tags__label {
display: none;
}
.react-tags__tag { .react-tags__tag {
font-size: 100%; font-size: 100%;
} }
.react-tags__selected { .react-tags__list {
display: inline; display: inline;
vertical-align: 2px; vertical-align: 2px;
padding: 0;
list-style-type: none;
} }
.react-tags__selected-tag { .react-tags__list-item {
display: inline-block; display: inline-block;
box-sizing: border-box; }
margin: 0 6px 6px 0; .react-tags__list-item:not(:last-child) {
padding: 6px 8px; margin-right: 3px;
border: 1px solid var(--input-border-color);
border-radius: .25rem;
background: #f1f1f1;
/* match the font styles */
font-size: inherit;
line-height: inherit;
} }
.react-tags__selected-tag:after { // The block to search
content: '\2715'; .react-tags__combobox {
color: #aaaaaa;
margin-left: 8px;
}
.react-tags__selected-tag:hover,
.react-tags__selected-tag:focus {
border-color: var(--input-border-color);
}
.react-tags__search {
display: inline-block; display: inline-block;
/* match tag layout */ /* match tag layout */
@ -76,13 +67,13 @@
} }
@media screen and (min-width: $smMin) { @media screen and (min-width: $smMin) {
.react-tags__search { .react-tags__combobox {
/* this will become the offsetParent for suggestions */ /* this will become the offsetParent for suggestions */
position: relative; position: relative;
} }
} }
.react-tags__search-input { .react-tags__combobox-input {
font-size: 1.25rem; font-size: 1.25rem;
line-height: inherit; line-height: inherit;
color: var(--input-text-color); color: var(--input-text-color);
@ -98,62 +89,52 @@
outline: none; outline: none;
} }
.react-tags__search-input::placeholder { .react-tags__combobox-input::placeholder {
color: #6c757d; color: #6c757d;
} }
.react-tags__search-input::-ms-clear { .react-tags__combobox-input::-ms-clear {
display: none; display: none;
} }
.react-tags__suggestions { .react-tags__listbox {
position: absolute; position: absolute;
top: 100%; top: 100%;
left: 0; left: 0;
width: 100%; width: 100%;
z-index: 10; z-index: 10;
}
@media screen and (min-width: $smMin) {
.react-tags__suggestions {
width: 240px;
}
}
.react-tags__suggestions ul {
margin: 4px -1px; margin: 4px -1px;
padding: 0; padding: 0;
list-style: none;
background: var(--primary-color); background: var(--primary-color);
border: 1px solid var(--border-color); border: 1px solid var(--border-color);
border-radius: .25rem; border-radius: .25rem;
box-shadow: 0 2px 6px rgb(0 0 0 / .2); box-shadow: 0 2px 6px rgb(0 0 0 / .2);
} }
.react-tags__suggestions li { @media screen and (min-width: $smMin) {
.react-tags__listbox {
width: 240px;
}
}
.react-tags__listbox .react-tags__listbox-option {
padding: 8px 10px; padding: 8px 10px;
} }
.react-tags__suggestions li:not(:last-child) { .react-tags__listbox .react-tags__listbox-option:not(:last-child) {
border-bottom: 1px solid var(--border-color); border-bottom: 1px solid var(--border-color);
} }
.react-tags__suggestions li mark { .react-tags__listbox .react-tags__listbox-option:hover:not(.react-tags__listbox-option--not-selectable) {
text-decoration: underline;
background: none;
font-weight: 600;
}
.react-tags__suggestions li:hover {
cursor: pointer; cursor: pointer;
background-color: var(--active-color); background-color: var(--active-color);
} }
.react-tags__suggestions li.is-active { .react-tags__listbox .react-tags__listbox-option.is-active {
background-color: var(--active-color); background-color: var(--active-color);
} }
.react-tags__suggestions li.is-disabled { .react-tags__listbox .react-tags__listbox-option.is-disabled {
opacity: .5; opacity: .5;
cursor: auto; cursor: auto;
} }

View file

@ -2,22 +2,33 @@ import { screen } from '@testing-library/react';
import { fromPartial } from '@total-typescript/shoehorn'; import { fromPartial } from '@total-typescript/shoehorn';
import { TagsSelector as createTagsSelector } from '../../../src/tags/helpers/TagsSelector'; import { TagsSelector as createTagsSelector } from '../../../src/tags/helpers/TagsSelector';
import type { TagsList } from '../../../src/tags/reducers/tagsList'; import type { TagsList } from '../../../src/tags/reducers/tagsList';
import type { TagFilteringMode } from '../../../src/utils/settings';
import { SettingsProvider } from '../../../src/utils/settings'; import { SettingsProvider } from '../../../src/utils/settings';
import { renderWithEvents } from '../../__helpers__/setUpTest'; import { renderWithEvents } from '../../__helpers__/setUpTest';
import { colorGeneratorMock } from '../../utils/services/__mocks__/ColorGenerator.mock'; import { colorGeneratorMock } from '../../utils/services/__mocks__/ColorGenerator.mock';
type SetUpOptions = {
allowNew?: boolean;
allTags?: string[];
tagFilteringMode?: TagFilteringMode;
};
describe('<TagsSelector />', () => { describe('<TagsSelector />', () => {
const onChange = vi.fn(); const onChange = vi.fn();
const TagsSelector = createTagsSelector(colorGeneratorMock); const TagsSelector = createTagsSelector(colorGeneratorMock);
const tags = ['foo', 'bar']; const tags = ['foo', 'bar'];
const tagsList = fromPartial<TagsList>({ tags: [...tags, 'baz'] }); const setUp = ({ allowNew = true, allTags, tagFilteringMode }: SetUpOptions = {}) => renderWithEvents(
const setUp = () => renderWithEvents( <SettingsProvider
<SettingsProvider value={fromPartial({})}> value={fromPartial({
shortUrlCreation: { tagFilteringMode },
})}
>
<TagsSelector <TagsSelector
selectedTags={tags} selectedTags={tags}
tagsList={tagsList} tagsList={fromPartial<TagsList>({ tags: allTags ?? [...tags, 'baz'] })}
listTags={vi.fn()} listTags={vi.fn()}
onChange={onChange} onChange={onChange}
allowNew={allowNew}
/> />
</SettingsProvider>, </SettingsProvider>,
); );
@ -37,18 +48,36 @@ describe('<TagsSelector />', () => {
it('contains expected suggestions', async () => { it('contains expected suggestions', async () => {
const { container, user } = setUp(); const { container, user } = setUp();
expect(container.querySelector('.react-tags__suggestions')).not.toBeInTheDocument(); expect(container.querySelector('.react-tags__listbox')).not.toBeInTheDocument();
expect(screen.queryByText('baz')).not.toBeInTheDocument(); expect(screen.queryByText('baz')).not.toBeInTheDocument();
await user.type(screen.getByPlaceholderText('Add tags to the URL'), 'ba'); await user.type(screen.getByPlaceholderText('Add tags to the URL'), 'ba');
expect(container.querySelector('.react-tags__suggestions')).toBeInTheDocument(); expect(container.querySelector('.react-tags__listbox')).toBeInTheDocument();
expect(screen.getByText('baz')).toBeInTheDocument(); expect(screen.getByText('baz')).toBeInTheDocument();
}); });
it('limits the amount of suggestions', async () => {
const { user } = setUp({ allTags: ['foo', 'foo1', 'foo2', 'foo3', 'foo4', 'foo5', 'foo6', 'foo7'] });
await user.type(screen.getByPlaceholderText('Add tags to the URL'), 'fo');
// First results are in the document
expect(screen.getByText('foo')).toBeInTheDocument();
expect(screen.getByText('foo1')).toBeInTheDocument();
expect(screen.getByText('foo2')).toBeInTheDocument();
expect(screen.getByText('foo3')).toBeInTheDocument();
expect(screen.getByText('foo4')).toBeInTheDocument();
expect(screen.getByText('foo5')).toBeInTheDocument();
// While the last ones are not
expect(screen.queryByText('foo6')).not.toBeInTheDocument();
expect(screen.queryByText('foo7')).not.toBeInTheDocument();
});
it.each([ it.each([
['The-New-Tag', [...tags, 'the-new-tag']], ['The-New-Tag', [...tags, 'the-new-tag']],
['foo', tags], ['AnOTH er tag ', [...tags, 'anoth-er-tag']],
// ['foo', tags], TODO Test that existing tags are ignored
])('invokes onChange when new tags are added', async (newTag, expectedTags) => { ])('invokes onChange when new tags are added', async (newTag, expectedTags) => {
const { user } = setUp(); const { user } = setUp();
@ -77,4 +106,31 @@ describe('<TagsSelector />', () => {
await user.click(screen.getByLabelText(`Remove ${removedLabel}`)); await user.click(screen.getByLabelText(`Remove ${removedLabel}`));
expect(onChange).toHaveBeenCalledWith([expected]); expect(onChange).toHaveBeenCalledWith([expected]);
}); });
it('displays "Add tag" option for new tags', async () => {
const { user } = setUp();
expect(screen.queryByText(/^Add "/)).not.toBeInTheDocument();
await user.type(screen.getByPlaceholderText('Add tags to the URL'), 'new-tag');
expect(screen.getByText(/^Add "/)).toBeInTheDocument();
});
it('displays "Tag not found" for unknown tags when add is not allowed', async () => {
const { user } = setUp({ allowNew: false });
expect(screen.queryByText('Tag not found')).not.toBeInTheDocument();
await user.type(screen.getByPlaceholderText('Add tags to the URL'), 'not-found-tag');
expect(screen.getByText('Tag not found')).toBeInTheDocument();
});
it.each([
['startsWith' as TagFilteringMode, ['foo', 'foobar']],
['includes' as TagFilteringMode, ['foo', 'barfoo', 'foobar']],
])('filters suggestions with different algorithm based on filtering mode', async (tagFilteringMode, expectedTags) => {
const { user } = setUp({ tagFilteringMode, allTags: ['foo', 'barfoo', 'foobar'] });
await user.type(screen.getByPlaceholderText('Add tags to the URL'), ' Foo');
expectedTags.forEach((tag) => expect(screen.getByText(tag)).toBeInTheDocument());
});
}); });