Skip to content

Commit df6e61b

Browse files
authored
[MP-996] Add size property to TagSelector (#4682)
1 parent 73ebf46 commit df6e61b

File tree

7 files changed

+124
-7
lines changed

7 files changed

+124
-7
lines changed

.changeset/lovely-bikes-agree.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
'@toptal/picasso-tagselector': minor
3+
---
4+
5+
### TagSelector
6+
7+
- add `size` property to control input size

packages/base/Tagselector/src/TagSelector/TagSelector.tsx

+5-2
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ import type {
66
FocusEventHandler,
77
} from 'react'
88
import React, { forwardRef, Fragment } from 'react'
9-
import type { BaseProps } from '@toptal/picasso-shared'
9+
import type { BaseProps, SizeType } from '@toptal/picasso-shared'
1010
import type { PopperOptions } from 'popper.js'
1111
import { Autocomplete } from '@toptal/picasso-autocomplete'
1212
import { unsafeErrorLog, noop } from '@toptal/picasso-utils'
@@ -49,7 +49,7 @@ const getItemText = (item: Item | null) =>
4949

5050
export interface Props
5151
extends BaseProps,
52-
Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange' | 'value'> {
52+
Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange' | 'value' | 'size'> {
5353
/** Placeholder for value */
5454
placeholder?: string
5555
/** Disables `TagSelector` */
@@ -101,6 +101,8 @@ export interface Props
101101
onDelete: () => void
102102
disabled?: boolean
103103
}) => ReactNode
104+
/** Component size */
105+
size?: SizeType<'medium' | 'large'>
104106
/** DOM element that wraps the Popper */
105107
popperContainer?: HTMLElement
106108
/** Options provided to the popper.js instance */
@@ -278,6 +280,7 @@ TagSelector.defaultProps = {
278280
placeholder: '',
279281
showOtherOption: false,
280282
status: 'default',
283+
size: 'medium',
281284
}
282285

283286
TagSelector.displayName = 'TagSelector'

packages/base/Tagselector/src/TagSelector/__snapshots__/test.tsx.snap

+3-3
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ exports[`TagSelector disabled render 1`] = `
1515
class="flex"
1616
>
1717
<div
18-
class="base-Input base- base-Input relative gap-y gap-x items-center rounded-sm [font-size:_unset] group text-nowrap p-2 w-[18.75rem] bg-gray after:content-[""] after:inline-block after:absolute after:top-0 after:bottom-0 after:right-0 after:left-0 after:pointer-events after:border-solid after:rounded-sm after:border-gray after:border [&:has(:focus)]:after:shadow-[0_0_0_3px] [&:has(:focus)]:after:shadow-blue text-gray flex flex-wrap h-auto py-1 pl-1 cursor-pointer [&>input]:min-w [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0"
18+
class="base-Input base- base-Input relative gap-y gap-x items-center rounded-sm [font-size:_unset] group text-nowrap p-2 w-[18.75rem] bg-gray after:content-[""] after:inline-block after:absolute after:top-0 after:bottom-0 after:right-0 after:left-0 after:pointer-events after:border-solid after:rounded-sm after:border-gray after:border [&:has(:focus)]:after:shadow-[0_0_0_3px] [&:has(:focus)]:after:shadow-blue text-gray flex flex-wrap h-auto cursor-pointer [&>input]:min-w [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0 py-1 pl-1"
1919
>
2020
<div
2121
aria-disabled="true"
@@ -83,7 +83,7 @@ exports[`TagSelector preselected value 1`] = `
8383
class="flex"
8484
>
8585
<div
86-
class="base-Input base-Input relative gap-y gap-x items-center rounded-sm [font-size:_unset] group text-nowrap p-2 w-[18.75rem] bg-white after:content-[""] after:inline-block after:absolute after:top-0 after:bottom-0 after:right-0 after:left-0 after:pointer-events after:border-solid after:rounded-sm after:border-gray [&:has(:focus)]:after:border-blue after:border [&:has(:focus)]:after:shadow-[0_0_0_3px] [&:has(:focus)]:after:shadow-blue hover:[&:not(:has(:focus))]:after:border-gray text-black flex flex-wrap h-auto py-1 pl-1 cursor-pointer [&>input]:min-w [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0"
86+
class="base-Input base-Input relative gap-y gap-x items-center rounded-sm [font-size:_unset] group text-nowrap p-2 w-[18.75rem] bg-white after:content-[""] after:inline-block after:absolute after:top-0 after:bottom-0 after:right-0 after:left-0 after:pointer-events after:border-solid after:rounded-sm after:border-gray [&:has(:focus)]:after:border-blue after:border [&:has(:focus)]:after:shadow-[0_0_0_3px] [&:has(:focus)]:after:shadow-blue hover:[&:not(:has(:focus))]:after:border-gray text-black flex flex-wrap h-auto cursor-pointer [&>input]:min-w [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0 py-1 pl-1"
8787
>
8888
<div
8989
class="text-lg transition-none border border-solid rounded-[6.25rem] h-6 max-w inline-flex justify-center items-center cursor-default bg-white group align-middle leading-[inherit] text-graphite border-gray"
@@ -149,7 +149,7 @@ exports[`TagSelector renders 1`] = `
149149
class="flex"
150150
>
151151
<div
152-
class="base-Input base-Input relative gap-y gap-x items-center rounded-sm [font-size:_unset] group text-nowrap p-2 w-[18.75rem] bg-white after:content-[""] after:inline-block after:absolute after:top-0 after:bottom-0 after:right-0 after:left-0 after:pointer-events after:border-solid after:rounded-sm after:border-gray [&:has(:focus)]:after:border-blue after:border [&:has(:focus)]:after:shadow-[0_0_0_3px] [&:has(:focus)]:after:shadow-blue hover:[&:not(:has(:focus))]:after:border-gray text-black flex flex-wrap h-auto py-1 pl-1 cursor-pointer [&>input]:min-w [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0"
152+
class="base-Input base-Input relative gap-y gap-x items-center rounded-sm [font-size:_unset] group text-nowrap p-2 w-[18.75rem] bg-white after:content-[""] after:inline-block after:absolute after:top-0 after:bottom-0 after:right-0 after:left-0 after:pointer-events after:border-solid after:rounded-sm after:border-gray [&:has(:focus)]:after:border-blue after:border [&:has(:focus)]:after:shadow-[0_0_0_3px] [&:has(:focus)]:after:shadow-blue hover:[&:not(:has(:focus))]:after:border-gray text-black flex flex-wrap h-auto cursor-pointer [&>input]:min-w [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0 py-1 pl-1"
153153
>
154154
<input
155155
aria-autocomplete="list"
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
1+
import React, { useState } from 'react'
2+
import type { AutocompleteItem, TagSelectorProps } from '@toptal/picasso'
3+
import { Container, TagSelector, Typography } from '@toptal/picasso'
4+
import { isSubstring, SPACING_4 } from '@toptal/picasso-utils'
5+
6+
const allOptions = [
7+
{ value: 'AF', text: 'Afghanistan' },
8+
{ value: 'AI', text: 'Aland Islands' },
9+
{ value: 'ALB', text: 'Albania' },
10+
{ value: 'ALG', text: 'Algeria' },
11+
{ value: 'BY', text: 'Belarus' },
12+
{ value: 'HR', text: 'Croatia' },
13+
{ value: 'LU', text: 'Lithuania' },
14+
{ value: 'SK', text: 'Slovakia' },
15+
{ value: 'SP', text: 'Spain' },
16+
{ value: 'UA', text: 'Ukraine' },
17+
]
18+
19+
const EMPTY_INPUT_VALUE = ''
20+
const getDisplayValue = (item: AutocompleteItem | null) =>
21+
(item && item.text) || EMPTY_INPUT_VALUE
22+
23+
const filterOptions = (str = '') => {
24+
if (str === '') {
25+
return allOptions
26+
}
27+
28+
const result = allOptions.filter(option =>
29+
isSubstring(str, getDisplayValue(option))
30+
)
31+
32+
return result.length > 0 ? result : null
33+
}
34+
35+
const TagSelectWithSize = ({
36+
size,
37+
value,
38+
options,
39+
inputValue,
40+
onChange,
41+
onInputChange,
42+
}: TagSelectorProps) => (
43+
<TagSelector
44+
size={size}
45+
data-testid='component'
46+
options={options}
47+
placeholder='Start typing...'
48+
value={value}
49+
inputValue={inputValue}
50+
getDisplayValue={getDisplayValue}
51+
onChange={onChange}
52+
onInputChange={onInputChange}
53+
/>
54+
)
55+
56+
const Example = () => {
57+
const [options, setOptions] = useState<AutocompleteItem[] | null>(allOptions)
58+
const [value, setValue] = useState<AutocompleteItem[]>([
59+
allOptions[0],
60+
allOptions[1],
61+
allOptions[2],
62+
])
63+
const [inputValue, setInputValue] = useState(EMPTY_INPUT_VALUE)
64+
65+
const onChange = selectedValues => {
66+
window.console.log('onChange values: ', selectedValues)
67+
setValue(selectedValues)
68+
}
69+
70+
const onInputChange = newInputValue => {
71+
window.console.log('onInputChange value: ', newInputValue)
72+
setInputValue(newInputValue)
73+
setOptions(filterOptions(newInputValue))
74+
}
75+
76+
return (
77+
<div>
78+
{(['medium', 'large'] as const).map(size => (
79+
<Container key={size} bottom={SPACING_4}>
80+
<Typography variant='body' titleCase>
81+
{size}
82+
</Typography>
83+
<TagSelectWithSize
84+
options={options}
85+
size={size}
86+
value={value}
87+
inputValue={inputValue}
88+
onChange={onChange}
89+
onInputChange={onInputChange}
90+
/>
91+
</Container>
92+
))}
93+
</div>
94+
)
95+
}
96+
97+
export default Example

packages/base/Tagselector/src/TagSelector/story/index.jsx

+5
Original file line numberDiff line numberDiff line change
@@ -91,3 +91,8 @@ page
9191
'Status',
9292
'base/Tagselector'
9393
)
94+
.addExample(
95+
'TagSelector/story/Sizes.example.tsx',
96+
'Sizes',
97+
'base/Tagselector'
98+
)

packages/base/Tagselector/src/TagSelectorInput/TagSelectorInput.tsx

+6-1
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ export const TagSelectorInput = forwardRef<HTMLInputElement, InputProps>(
3232
inputProps,
3333
testIds,
3434
highlight,
35+
size,
3536
...rest
3637
} = props
3738

@@ -48,10 +49,12 @@ export const TagSelectorInput = forwardRef<HTMLInputElement, InputProps>(
4849
inputRef={ref}
4950
style={style}
5051
className={cx(
51-
`flex flex-wrap h-auto py-1 pl-1 cursor-pointer
52+
`flex flex-wrap h-auto cursor-pointer
5253
[&>input]:min-w-[3em] [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0`,
5354
{
5455
'pr-[2.25em]': Boolean(endAdornment),
56+
'py-1 pl-1': size === 'medium',
57+
'gap-[12px] px-4': size === 'large',
5558
}
5659
)}
5760
highlight={highlight}
@@ -78,6 +81,7 @@ export const TagSelectorInput = forwardRef<HTMLInputElement, InputProps>(
7881
endAdornment={usedEndAdornment}
7982
startAdornment={startAdornment}
8083
onChange={onChange}
84+
size={size}
8185
>
8286
{children}
8387
</OutlinedInput>
@@ -89,6 +93,7 @@ TagSelectorInput.defaultProps = {
8993
multiline: false,
9094
width: 'auto',
9195
status: 'default',
96+
size: 'medium',
9297
}
9398

9499
TagSelectorInput.displayName = 'TagSelectorInput'

packages/base/Tagselector/src/TagSelectorInput/__snapshots__/test.tsx.snap

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ exports[`TagSelectorInput renders 1`] = `
66
class="Picasso-root"
77
>
88
<div
9-
class="base-Input relative gap-y gap-x items-center rounded-sm [font-size:_unset] group text-nowrap p-2 w-[18.75rem] bg-white after:content-[""] after:inline-block after:absolute after:top-0 after:bottom-0 after:right-0 after:left-0 after:pointer-events after:border-solid after:rounded-sm after:border-gray [&:has(:focus)]:after:border-blue after:border [&:has(:focus)]:after:shadow-[0_0_0_3px] [&:has(:focus)]:after:shadow-blue hover:[&:not(:has(:focus))]:after:border-gray text-black flex flex-wrap h-auto py-1 pl-1 cursor-pointer [&>input]:min-w [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0"
9+
class="base-Input relative gap-y gap-x items-center rounded-sm [font-size:_unset] group text-nowrap p-2 w-[18.75rem] bg-white after:content-[""] after:inline-block after:absolute after:top-0 after:bottom-0 after:right-0 after:left-0 after:pointer-events after:border-solid after:rounded-sm after:border-gray [&:has(:focus)]:after:border-blue after:border [&:has(:focus)]:after:shadow-[0_0_0_3px] [&:has(:focus)]:after:shadow-blue hover:[&:not(:has(:focus))]:after:border-gray text-black flex flex-wrap h-auto cursor-pointer [&>input]:min-w [&>input]:flex-grow [&>input]:w-0 [&>input]:h-6 [&>input]:pl-1 [&>input]:pr-0 [&>input]:mb-0 py-1 pl-1"
1010
>
1111
<input
1212
aria-invalid="false"

0 commit comments

Comments
 (0)