Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Try Ariakit Select for new CustomSelectControl component #55790

Merged
merged 36 commits into from
Nov 23, 2023
Merged
Show file tree
Hide file tree
Changes from 20 commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
d3573ae
Create base component using ariakit
brookewp Nov 7, 2023
9d34f51
Simplify story example
brookewp Nov 7, 2023
9000c5a
Add options for children and value with stories
brookewp Nov 7, 2023
0024d32
Update types
brookewp Nov 7, 2023
bc0683f
Add uncontrolled story and adapt value
brookewp Nov 9, 2023
2a3c220
Remove duplicate import
brookewp Nov 9, 2023
2b9174d
Add size prop
brookewp Nov 9, 2023
63267af
Changes from pairing session with ciampo
brookewp Nov 9, 2023
017bd82
Rename and cleanup
brookewp Nov 9, 2023
c6cffa6
Update styles
brookewp Nov 10, 2023
27b9854
Add context
brookewp Nov 10, 2023
add8e6d
Add README
brookewp Nov 10, 2023
d343165
Update manifest
brookewp Nov 10, 2023
3edc792
Update types and move context provider
brookewp Nov 10, 2023
7ef27ea
Add back types for multi-selection
brookewp Nov 15, 2023
d35a044
Remove file for moved component
brookewp Nov 17, 2023
ede0ef9
Add translation for placeholder
brookewp Nov 17, 2023
c3e4d29
Add check and cleanup styles
brookewp Nov 17, 2023
f771fad
Add multiselect example
brookewp Nov 17, 2023
3b5f968
Require type children for CustomSelectItem
brookewp Nov 17, 2023
e2ee037
Add small size and update sizing logic
brookewp Nov 17, 2023
b7b2fff
Update stories with ciampo’s suggested changes
brookewp Nov 17, 2023
e68ea07
Update styles to match legacy CustomSelectControl
brookewp Nov 17, 2023
27fc5fd
Cleanup based on PR feedback
brookewp Nov 18, 2023
24e08ad
Children as optional
brookewp Nov 20, 2023
f866d36
Incorporate ciampo’s suggestions to refine styles
brookewp Nov 20, 2023
ceb6b45
Make value required for CustomSelectItem and fix naming in comment
brookewp Nov 20, 2023
23c0f91
Merge branch 'trunk' into try/customselect-ariakit
brookewp Nov 20, 2023
0b8250a
Update changelog
brookewp Nov 20, 2023
c29569a
Refine styles and add styling to active item
brookewp Nov 21, 2023
39b7650
Only render label if defined
brookewp Nov 21, 2023
ff02e44
Cleanup story
brookewp Nov 21, 2023
b975f51
Add suggestions to WIP README and types
brookewp Nov 21, 2023
04faa3f
Make label required
brookewp Nov 23, 2023
df54fe8
Improve logic for default rendered value
brookewp Nov 23, 2023
45689ab
Update stories
brookewp Nov 23, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions docs/manifest.json
Original file line number Diff line number Diff line change
Expand Up @@ -779,6 +779,12 @@
"markdown_source": "../packages/components/src/confirm-dialog/README.md",
"parent": "components"
},
{
"title": "CustomSelectControlV2",
"slug": "custom-select-control-v2",
"markdown_source": "../packages/components/src/custom-select-control-v2/README.md",
"parent": "components"
},
{
"title": "CustomSelectControl",
"slug": "custom-select-control",
Expand Down
73 changes: 73 additions & 0 deletions packages/components/src/custom-select-control-v2/README.md
brookewp marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
<div class="callout callout-alert">
This feature is still experimental. “Experimental” means this is an early implementation subject to drastic and breaking changes.
</div>

### `CustomSelect`

Used to render a checkbox item.
brookewp marked this conversation as resolved.
Show resolved Hide resolved

#### Props

The component accepts the following props:

##### `children`: `React.ReactNode`

The child elements. This should be composed of CustomSelect.Item components.

- Required: yes

##### `defaultValue`: `string`

An optional default value for the control. If left `undefined`, the first non-disabled item will be used.

- Required: no

##### `label`: `string`

Label for the control.

- Required: no

##### `onChange`: `( newValue: string ) => void`

A function that receives the new value of the input.

- Required: no

##### `renderSelectedValue`: `( selectValue: string ) => React.ReactNode`

Can be used to render select UI with custom styled values.

- Required: no

##### `size`: `'default' | 'large'`

The size of the control.

- Required: no
ciampo marked this conversation as resolved.
Show resolved Hide resolved

##### `value`: `string`

Can be used to externally control the value of the control.

- Required: no

### `CustomSelectItemProps`
brookewp marked this conversation as resolved.
Show resolved Hide resolved

Used to render a select item.

#### Props

The component accepts the following props:

##### `children`: `React.ReactNode`

The children to display for each select item.
brookewp marked this conversation as resolved.
Show resolved Hide resolved

- Required: no

##### `value`: `string`
ciampo marked this conversation as resolved.
Show resolved Hide resolved

The value of the select item. This will be used as the children if children are left `undefined`.

- Required: yes
86 changes: 86 additions & 0 deletions packages/components/src/custom-select-control-v2/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
/**
* External dependencies
*/
// eslint-disable-next-line no-restricted-imports
import * as Ariakit from '@ariakit/react';
/**
* WordPress dependencies
*/
import { createContext, useContext, useMemo } from '@wordpress/element';
import { __ } from '@wordpress/i18n';

/**
* Internal dependencies
*/
import { useCx } from '../utils/hooks/use-cx';
import * as Styled from './styles';
import type {
CustomSelectProps,
CustomSelectItemProps,
CustomSelectContext as CustomSelectContextType,
} from './types';

export const CustomSelectContext =
createContext< CustomSelectContextType >( undefined );

export function CustomSelect( props: CustomSelectProps ) {
brookewp marked this conversation as resolved.
Show resolved Hide resolved
const {
children,
defaultValue,
label,
onChange,
size = 'default',
value,
renderSelectedValue,
} = props;

const store = Ariakit.useSelectStore( {
setValue: ( nextValue ) => onChange?.( nextValue ),
defaultValue,
value,
} );

const { value: currentValue } = store.useState();

const cx = useCx();

const classes = useMemo(
() => cx( Styled.inputSize[ size ] ),
brookewp marked this conversation as resolved.
Show resolved Hide resolved
[ cx, size ]
);

return (
<>
<Styled.CustomSelectLabel store={ store }>
brookewp marked this conversation as resolved.
Show resolved Hide resolved
{ label }
</Styled.CustomSelectLabel>
<Styled.CustomSelectButton className={ classes } store={ store }>
{ renderSelectedValue
? renderSelectedValue( currentValue )
: currentValue ?? __( 'Select an item' ) }
brookewp marked this conversation as resolved.
Show resolved Hide resolved
<Ariakit.SelectArrow />
</Styled.CustomSelectButton>
<Styled.CustomSelectPopover store={ store } sameWidth>
<CustomSelectContext.Provider value={ { store } }>
{ children }
</CustomSelectContext.Provider>
</Styled.CustomSelectPopover>
</>
);
}

export function CustomSelectItem( {
children,
...props
}: CustomSelectItemProps ) {
const customSelectContext = useContext( CustomSelectContext );
return (
<Styled.CustomSelectItem
brookewp marked this conversation as resolved.
Show resolved Hide resolved
brookewp marked this conversation as resolved.
Show resolved Hide resolved
store={ customSelectContext?.store }
{ ...props }
>
{ children }
<Ariakit.SelectItemCheck />
ciampo marked this conversation as resolved.
Show resolved Hide resolved
</Styled.CustomSelectItem>
);
}
brookewp marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,149 @@
/**
* External dependencies
*/
import type { Meta, StoryFn } from '@storybook/react';

/**
* WordPress dependencies
*/
import { useState } from '@wordpress/element';

/**
* Internal dependencies
*/
import { CustomSelect, CustomSelectItem } from '..';

const meta: Meta< typeof CustomSelect > = {
title: 'Components (Experimental)/CustomSelectControl v2',
component: CustomSelect,
subcomponents: {
// @ts-expect-error - See https://github.com/storybookjs/storybook/issues/23170
CustomSelectItem,
},
argTypes: {
children: { control: { type: null } },
},
parameters: {
actions: { argTypesRegex: '^on.*' },
controls: { expanded: true },
docs: {
canvas: { sourceState: 'shown' },
source: { excludeDecorators: true },
},
},
decorators: [
( Story ) => (
<div
style={ {
minHeight: '250px',
} }
>
<Story />
</div>
),
],
};
export default meta;

const Template: StoryFn< typeof CustomSelect > = () => {
return (
<CustomSelect label="Label" defaultValue="Default">
<CustomSelectItem value="Small">
<span style={ { fontSize: '75%' } }>Small</span>
</CustomSelectItem>
<CustomSelectItem value="Default">Default</CustomSelectItem>
brookewp marked this conversation as resolved.
Show resolved Hide resolved
<CustomSelectItem value="Large">
<span style={ { fontSize: '150%' } }>Large</span>
</CustomSelectItem>
<CustomSelectItem value="Huge">
<span style={ { fontSize: '200%' } }>Huge</span>
</CustomSelectItem>
</CustomSelect>
);
};

export const Default = Template.bind( {} );

const MultiSelectTemplate: StoryFn< typeof CustomSelect > = () => {
function renderValue( value: string | string[] ) {
if ( value.length === 0 ) return '0 colors selected';
return <div>{ value.length } colors selected</div>;
}

const items = [
'amber',
'aquamarine',
'flamingo pink',
'lavendar',
'maroon',
'tangerine',
];

const [ value, setValue ] = useState< string | string[] >( [
'lavendar',
'tangerine',
] );

return (
<CustomSelect
onChange={ ( nextValue ) => setValue( nextValue ) }
defaultValue={ value }
label="Select Colors"
renderSelectedValue={ ( currentValue ) =>
renderValue( currentValue )
}
>
{ items.map( ( item ) => (
<CustomSelectItem key={ item } value={ item }>
{ item }
</CustomSelectItem>
) ) }
</CustomSelect>
);
};

export const MultiSelect = MultiSelectTemplate.bind( {} );

const ControlledTemplate = () => {
function renderValue( gravatar: string | string[] ) {
const avatar = `https://gravatar.com/avatar?d=${ gravatar }`;
return (
<div style={ { display: 'flex', alignItems: 'center' } }>
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This div with flex styles was added after adding the checkmark. The CustomSelectItem needed display: flex so the checkmark is rendered correctly (aligning with our current CustomSelectControl):

export const CustomSelectItem = styled( Ariakit.SelectItem )`
display: flex;
align-items: center;
justify-content: space-between;

It doesn't seem ideal, but the checkmark is rendered as another child. So I can see potential overrides happening unless the consumer wraps the content in a div, like I did above.

<img
style={ { maxHeight: '75px', marginRight: '10px' } }
key={ avatar }
src={ avatar }
alt=""
aria-hidden
/>
<span>{ gravatar }</span>
</div>
);
}

const options = [ 'mystery-person', 'identicon', 'wavatar', 'retro' ];

const [ value, setValue ] = useState< string | string[] >();

return (
<>
<CustomSelect
label="Default Gravatars"
onChange={ ( nextValue ) => setValue( nextValue ) }
size="large"
value={ value }
renderSelectedValue={ ( currentValue ) =>
renderValue( currentValue )
}
>
{ options.map( ( option ) => (
<CustomSelectItem key={ option } value={ option }>
{ renderValue( option ) }
</CustomSelectItem>
) ) }
</CustomSelect>
</>
);
};

export const Controlled = ControlledTemplate.bind( {} );
64 changes: 64 additions & 0 deletions packages/components/src/custom-select-control-v2/styles.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
/**
* External dependencies
*/
import styled from '@emotion/styled';
import { css } from '@emotion/react';
// eslint-disable-next-line no-restricted-imports
import * as Ariakit from '@ariakit/react';

/**
* Internal dependencies
*/
import { COLORS } from '../utils';
import { space } from '../utils/space';

export const CustomSelectLabel = styled( Ariakit.SelectLabel )`
font-size: 11px;
ciampo marked this conversation as resolved.
Show resolved Hide resolved
font-weight: 500;
line-height: 1.4;
text-transform: uppercase;
margin-bottom: ${ space( 2 ) };
`;

export const CustomSelectButton = styled( Ariakit.Select )`
display: flex;
justify-content: space-between;
align-items: center;
background: ${ COLORS.white };
border: 1px solid ${ COLORS.gray[ 600 ] };
border-radius: ${ space( 0.5 ) };
cursor: pointer;
padding: ${ space( 2 ) };
width: 100%;
&[data-focus-visible] {
outline-style: solid;
}
&[aria-expanded='true'] {
border: 1.5px solid ${ COLORS.theme.accent };
}
`;

export const inputSize = {
default: css`
height: 40px;
`,
large: css`
height: auto;
`,
};
brookewp marked this conversation as resolved.
Show resolved Hide resolved

export const CustomSelectPopover = styled( Ariakit.SelectPopover )`
border-radius: ${ space( 0.5 ) };
background: ${ COLORS.white };
border: 1px solid ${ COLORS.gray[ 900 ] };
margin: ${ space( 3 ) } 0;
brookewp marked this conversation as resolved.
Show resolved Hide resolved
`;
export const CustomSelectItem = styled( Ariakit.SelectItem )`
display: flex;
align-items: center;
justify-content: space-between;
padding: ${ space( 2 ) };
&:hover {
background-color: ${ COLORS.gray[ 300 ] };
}
brookewp marked this conversation as resolved.
Show resolved Hide resolved
`;
Loading
Loading