Skip to content

Commit

Permalink
feat(select): removed dual placeholder logic from select
Browse files Browse the repository at this point in the history
Select.Value placeholder prop has been removed

BREAKING CHANGE: prop removal - placeholder of Select.Value
  • Loading branch information
Powerplex committed Jul 29, 2024
1 parent 95ce1a5 commit eaf503d
Show file tree
Hide file tree
Showing 6 changed files with 123 additions and 46 deletions.
2 changes: 1 addition & 1 deletion packages/components/dropdown/src/DropdownValue.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import { useDropdownContext } from './DropdownContext'
export interface ValueProps {
children?: ReactNode
className?: string
placeholder: string
placeholder?: string
}

export const Value = forwardRef(
Expand Down
14 changes: 14 additions & 0 deletions packages/components/select/src/Select.doc.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,10 @@ import { Select } from '@spark-ui/select'
of: Select.Items,
description: 'The wrapper which contains all the options.',
},
'Select.Placeholder': {
of: Select.Placeholder,
description: 'Option selected when no value is set',
},
'Select.Item': { of: Select.Item, description: 'Each option of the element field' },
'Select.Group': {
of: Select.Group,
Expand Down Expand Up @@ -97,6 +101,16 @@ It is important to use `Select.Label` inside each `Select.Group` to give it an a

<Canvas of={stories.Grouped} />

### Placeholder

By default, the `Select.Placeholder` option can be selected.

You can apply `disabled` if the user should not be able to go to unselected state.

You can apply `disabled` and `hidden` to hide the placeholder entirely (only it's value will be shown while in closed state).

<Canvas of={stories.Placeholder} />

### Read only

Use `readOnly` prop to indicate the Select is only readable.
Expand Down
108 changes: 87 additions & 21 deletions packages/components/select/src/Select.stories.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
/* eslint-disable max-lines */
import { StoryLabel } from '@docs/helpers/StoryLabel'
import { FormField } from '@spark-ui/form-field'
import { BookmarkFill } from '@spark-ui/icons/dist/icons/BookmarkFill'
import { VisuallyHidden } from '@spark-ui/visually-hidden'
Expand All @@ -22,11 +23,11 @@ export const Default: StoryFn = _args => {
<Select.LeadingIcon>
<BookmarkFill />
</Select.LeadingIcon>
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder>--Pick a book--</Select.Placeholder>
<Select.Placeholder>Pick a book</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3">The Idiot</Select.Item>
Expand All @@ -46,11 +47,11 @@ export const Controlled: StoryFn = () => {
<div>
<Select name="book" value={value} onValueChange={setValue}>
<Select.Trigger aria-label="Book">
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder>--Pick a book--</Select.Placeholder>
<Select.Placeholder>Pick a book</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3">The Idiot</Select.Item>
Expand All @@ -68,11 +69,11 @@ export const Disabled: StoryFn = _args => {
<div>
<Select name="book" disabled>
<Select.Trigger aria-label="Book">
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder>--Pick a book--</Select.Placeholder>
<Select.Placeholder>Pick a book</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3">The Idiot</Select.Item>
Expand All @@ -85,16 +86,81 @@ export const Disabled: StoryFn = _args => {
)
}

export const Placeholder: StoryFn = _args => {
return (
<div className="flex flex-col gap-lg">
<div>
<StoryLabel>default</StoryLabel>
<Select name="book">
<Select.Trigger aria-label="Book">
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder>Pick a book</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3">The Idiot</Select.Item>
<Select.Item value="book-4">A Picture of Dorian Gray</Select.Item>
<Select.Item value="book-5">1984</Select.Item>
<Select.Item value="book-6">Pride and Prejudice</Select.Item>
</Select.Items>
</Select>
</div>

<div>
<StoryLabel>disabled</StoryLabel>
<Select name="book">
<Select.Trigger aria-label="Book">
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder disabled>Pick a book</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3">The Idiot</Select.Item>
<Select.Item value="book-4">A Picture of Dorian Gray</Select.Item>
<Select.Item value="book-5">1984</Select.Item>
<Select.Item value="book-6">Pride and Prejudice</Select.Item>
</Select.Items>
</Select>
</div>

<div>
<StoryLabel>disabled + hidden</StoryLabel>
<Select name="book">
<Select.Trigger aria-label="Book">
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder disabled hidden>
Pick a book
</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3">The Idiot</Select.Item>
<Select.Item value="book-4">A Picture of Dorian Gray</Select.Item>
<Select.Item value="book-5">1984</Select.Item>
<Select.Item value="book-6">Pride and Prejudice</Select.Item>
</Select.Items>
</Select>
</div>
</div>
)
}

export const ReadOnly: StoryFn = _args => {
return (
<div>
<Select name="book" readOnly>
<Select.Trigger aria-label="Book">
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder>--Pick a book--</Select.Placeholder>
<Select.Placeholder>Pick a book</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3">The Idiot</Select.Item>
Expand All @@ -112,11 +178,11 @@ export const DisabledItem: StoryFn = _args => {
<div>
<Select name="book">
<Select.Trigger aria-label="Book">
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder>--Pick a book--</Select.Placeholder>
<Select.Placeholder>Pick a book</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3" disabled>
Expand All @@ -136,11 +202,11 @@ export const Grouped: StoryFn = _args => {
<div>
<Select name="book">
<Select.Trigger aria-label="Book">
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder>--Pick a book--</Select.Placeholder>
<Select.Placeholder>Pick a book</Select.Placeholder>
<Select.Group>
<Select.Label>Best-sellers</Select.Label>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
Expand Down Expand Up @@ -168,11 +234,11 @@ export const LeadingIcon: StoryFn = _args => {
<Select.LeadingIcon>
<BookmarkFill />
</Select.LeadingIcon>
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
<Select.Placeholder>--Pick a book--</Select.Placeholder>
<Select.Placeholder>Pick a book</Select.Placeholder>
<Select.Item value="book-1">To Kill a Mockingbird</Select.Item>
<Select.Item value="book-2">War and Peace</Select.Item>
<Select.Item value="book-3">The Idiot</Select.Item>
Expand All @@ -196,7 +262,7 @@ export const Statuses: StoryFn = () => {
return (
<Select name={'book-' + status} state={status}>
<Select.Trigger aria-label="Book">
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
Expand All @@ -221,7 +287,7 @@ export const FormFieldLabel: StoryFn = _args => {
<FormField.Label>Book</FormField.Label>
<Select>
<Select.Trigger>
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
Expand All @@ -247,7 +313,7 @@ export const FormFieldHiddenLabel: StoryFn = _args => {
</FormField.Label>
<Select>
<Select.Trigger>
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
Expand All @@ -271,7 +337,7 @@ export const FormFieldReadOnly: StoryFn = _args => {
<FormField.Label>Book</FormField.Label>
<Select>
<Select.Trigger aria-label="Book">
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
Expand All @@ -295,7 +361,7 @@ export const FormFieldDisabled: StoryFn = _args => {
<FormField.Label>Book</FormField.Label>
<Select>
<Select.Trigger>
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
Expand All @@ -319,7 +385,7 @@ export const FormFieldRequired: StoryFn = _args => {
<FormField.Label>Book</FormField.Label>
<Select>
<Select.Trigger>
<Select.Value placeholder="Pick a book" />
<Select.Value />
</Select.Trigger>

<Select.Items>
Expand Down Expand Up @@ -348,7 +414,7 @@ export const FormFieldValidation: StoryFn = () => {
}}
>
<Select.Trigger>
<Select.Value placeholder="Pick an state" />
<Select.Value />
</Select.Trigger>

<Select.Items>
Expand Down
15 changes: 10 additions & 5 deletions packages/components/select/src/SelectItem.tsx
Original file line number Diff line number Diff line change
@@ -1,21 +1,26 @@
import { forwardRef, type Ref } from 'react'
import { ComponentPropsWithoutRef, forwardRef, type Ref } from 'react'

export interface ItemProps {
disabled?: boolean
export interface ItemProps extends ComponentPropsWithoutRef<'option'> {
value: string
children: string
hidden?: boolean
disabled?: boolean
}

export const Item = forwardRef(
({ disabled = false, value, children }: ItemProps, forwardedRef: Ref<HTMLOptionElement>) => {
(
{ disabled = false, hidden = false, value, children, ...props }: ItemProps,
forwardedRef: Ref<HTMLOptionElement>
) => {
return (
<option
data-spark-component="select-item"
ref={forwardedRef}
key={value}
value={value}
disabled={disabled}
// label
hidden={hidden}
{...props}
>
{children}
</option>
Expand Down
15 changes: 8 additions & 7 deletions packages/components/select/src/SelectPlaceholder.tsx
Original file line number Diff line number Diff line change
@@ -1,27 +1,28 @@
import { forwardRef, type Ref, useEffect } from 'react'

import { useSelectContext } from './SelectContext'
import { type ItemProps } from './SelectItem'

export interface PlaceholderProps {
disabled?: boolean
children: string
}
export type PlaceholderProps = Omit<ItemProps, 'value'>

export const Placeholder = forwardRef(
({ disabled = false, children }: PlaceholderProps, forwardedRef: Ref<HTMLOptionElement>) => {
const { setPlaceholder } = useSelectContext()
({ children, ...props }: PlaceholderProps, forwardedRef: Ref<HTMLOptionElement>) => {
const { setPlaceholder, selectedItem } = useSelectContext()

useEffect(() => {
setPlaceholder(children)
}, [children])

const isSelected = selectedItem?.value == null || selectedItem.value === ''

return (
<option
data-spark-component="select-placeholder"
ref={forwardedRef}
key="placeholder"
value=""
disabled={disabled}
selected={isSelected}
{...props}
>
{children}
</option>
Expand Down
15 changes: 3 additions & 12 deletions packages/components/select/src/SelectValue.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,22 +6,13 @@ import { useSelectContext } from './SelectContext'
export interface ValueProps {
children?: ReactNode
className?: string
/**
* Optional placeholder value for the trigger.
* If not specified, the value inside `Select.Placeholder` item will be used.
*/
placeholder?: string
}

export const Value = forwardRef(
(
{ children, className, placeholder: customPlaceholder }: ValueProps,
forwardedRef: Ref<HTMLSpanElement>
) => {
({ children, className }: ValueProps, forwardedRef: Ref<HTMLSpanElement>) => {
const { selectedItem, placeholder, disabled } = useSelectContext()

const isPlaceholderSelected = selectedItem?.value == null
const valuePlaceholder = customPlaceholder || placeholder
const isPlaceholderSelected = selectedItem?.value == null || selectedItem?.value === ''

return (
<span
Expand All @@ -36,7 +27,7 @@ export const Value = forwardRef(
isPlaceholderSelected && !disabled && 'text-on-surface/dim-1'
)}
>
{isPlaceholderSelected ? valuePlaceholder : children || selectedItem?.text}
{isPlaceholderSelected ? placeholder : children || selectedItem?.text}
</span>
</span>
)
Expand Down

0 comments on commit eaf503d

Please sign in to comment.