Skip to content

Commit

Permalink
fix: broken types
Browse files Browse the repository at this point in the history
  • Loading branch information
ragafus committed Oct 7, 2024
1 parent 47442d5 commit c3673d2
Show file tree
Hide file tree
Showing 3 changed files with 95 additions and 36 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,21 @@ import SearchSelectInput from '@commercetools-uikit/search-select-input';
import FieldErrors from '@commercetools-uikit/field-errors';
import FieldWarnings from '@commercetools-uikit/field-warnings';

type ReactSelectAsyncProps = AsyncProps<unknown, boolean, GroupBase<unknown>>;
type ReactSelectAsyncProps<
Option,
isMulti extends boolean,
Group extends GroupBase<Option>
> = AsyncProps<Option, isMulti, Group>;

type TCustomEvent = {
type TCustomEvent<
Option,
isMulti extends boolean,
Group extends GroupBase<Option>
> = {
target: {
id?: ReactSelectAsyncProps['inputId'];
name?: ReactSelectAsyncProps['name'];
value?: unknown;
id?: ReactSelectAsyncProps<Option, isMulti, Group>['inputId'];
name?: ReactSelectAsyncProps<Option, isMulti, Group>['name'];
value?: ReactSelectAsyncProps<Option, isMulti, Group>['value'];
};
persist: () => void;
};
Expand All @@ -49,7 +57,11 @@ const sequentialWarningsId = createSequentialId(
'search-select-field-warning-'
)();

export type TSearchSelectFieldProps = {
export type TSearchSelectFieldProps<
Option extends TOptionInnerPropsData = TOptionInnerPropsData,
isMulti extends boolean = boolean,
Group extends GroupBase<Option> = GroupBase<Option>
> = {
/**
*Horizontal size limit of the input fields.
*/
Expand All @@ -75,67 +87,79 @@ export type TSearchSelectFieldProps = {
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
'aria-label'?: ReactSelectAsyncProps['aria-label'];
'aria-label'?: ReactSelectAsyncProps<Option, isMulti, Group>['aria-label'];
/**
* HTML ID of an element that should be used as the label (for assistive tech)
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
'aria-labelledby'?: ReactSelectAsyncProps['aria-labelledby'];
'aria-labelledby'?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['aria-labelledby'];
/**
* The id of the search input. This forwarded as react-select's "inputId"
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
id?: ReactSelectAsyncProps['inputId'];
id?: ReactSelectAsyncProps<Option, isMulti, Group>['inputId'];
/**
* The id to set on the SelectContainer component. This is forwarded as react-select's "id"
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
containerId?: ReactSelectAsyncProps['id'];
containerId?: ReactSelectAsyncProps<Option, isMulti, Group>['id'];
/**
* Name of the HTML Input (optional - without this, no input will be rendered)
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
name?: ReactSelectAsyncProps['name'];
name?: ReactSelectAsyncProps<Option, isMulti, Group>['name'];
/**
* Placeholder text for the select value
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
placeholder?: ReactSelectAsyncProps['placeholder'];
placeholder?: ReactSelectAsyncProps<Option, isMulti, Group>['placeholder'];
/**
* Map of components to overwrite the default ones, see [what components you can override](https://react-select.com/components)
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
components?: ReactSelectAsyncProps['components'];
components?: ReactSelectAsyncProps<Option, isMulti, Group>['components'];
/**
* Control whether the selected values should be rendered in the control
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
controlShouldRenderValue?: ReactSelectAsyncProps['controlShouldRenderValue'];
controlShouldRenderValue?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['controlShouldRenderValue'];
/**
* Sets the tabIndex attribute on the input
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
tabIndex?: ReactSelectAsyncProps['tabIndex'];
tabIndex?: ReactSelectAsyncProps<Option, isMulti, Group>['tabIndex'];
/**
* The value of the select; reflected by the selected option
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
value?: ReactSelectAsyncProps['value'];
value?: ReactSelectAsyncProps<Option, isMulti, Group>['value'];
/**
* Remove the currently focused option when the user presses backspace
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
backspaceRemovesValue?: ReactSelectAsyncProps['backspaceRemovesValue'];
backspaceRemovesValue?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['backspaceRemovesValue'];
/**
* Indicates the input field has an error
*/
Expand Down Expand Up @@ -166,13 +190,17 @@ export type TSearchSelectFieldProps = {
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
isOptionDisabled?: ReactSelectAsyncProps['isOptionDisabled'];
isOptionDisabled?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['isOptionDisabled'];
/**
* Support multiple selected options
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
isMulti?: ReactSelectAsyncProps['isMulti'];
isMulti?: ReactSelectAsyncProps<Option, isMulti, Group>['isMulti'];
/**
* Focus the control when it is mounted. Renamed autoFocus of react-select
*/
Expand All @@ -182,19 +210,31 @@ export type TSearchSelectFieldProps = {
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
noOptionsMessage?: ReactSelectAsyncProps['noOptionsMessage'];
noOptionsMessage?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['noOptionsMessage'];
/**
* Maximum height of the menu before scrolling
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
maxMenuHeight?: ReactSelectAsyncProps['maxMenuHeight'];
maxMenuHeight?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['maxMenuHeight'];
/**
* Dom element to portal the select menu to
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
menuPortalTarget?: ReactSelectAsyncProps['menuPortalTarget'];
menuPortalTarget?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['menuPortalTarget'];
/**
* z-index value for the menu portal
* <br>
Expand All @@ -212,37 +252,48 @@ export type TSearchSelectFieldProps = {
/**
* Handle blur events on the control
*/
onBlur?: (event: TCustomEvent) => void;
onBlur?: (event: TCustomEvent<Option, isMulti, Group>) => void;
/**
* Called with a fake event when value changes.
* <br />
* The event's `target.name` will be the `name` supplied in props. The event's `target.value` will hold the value. The value will be the selected option, or an array of options in case `isMulti` is `true`.
*/
onChange?: (event: TCustomEvent, info: ActionMeta<unknown>) => void;
onChange?: (
event: TCustomEvent<Option, isMulti, Group>,
info: ActionMeta<unknown>
) => void;
/**
* Handle focus events on the control
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
onFocus?: ReactSelectAsyncProps['onFocus'];
onFocus?: ReactSelectAsyncProps<Option, isMulti, Group>['onFocus'];
/**
* Handle change events on the input
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
onInputChange?: ReactSelectAsyncProps['onInputChange'];
onInputChange?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['onInputChange'];
/**
* Select the currently focused option when the user presses tab
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
tabSelectsValue?: ReactSelectAsyncProps['tabSelectsValue'];
tabSelectsValue?: ReactSelectAsyncProps<
Option,
isMulti,
Group
>['tabSelectsValue'];
/**
* Function that returns a promise, which is the set of options to be used once the promise resolves.
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
loadOptions: ReactSelectAsyncProps['loadOptions'];
loadOptions: ReactSelectAsyncProps<Option, isMulti, Group>['loadOptions'];
/**
* The text shown while the options are being loaded
*/
Expand All @@ -252,13 +303,13 @@ export type TSearchSelectFieldProps = {
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
cacheOptions?: ReactSelectAsyncProps['cacheOptions'];
cacheOptions?: ReactSelectAsyncProps<Option, isMulti, Group>['cacheOptions'];
/**
* Custom method to filter whether an option should be displayed in the menu
* <br>
* [Props from React select was used](https://react-select.com/props)
*/
filterOption?: ReactSelectAsyncProps['filterOption'];
filterOption?: ReactSelectAsyncProps<Option, isMulti, Group>['filterOption'];
/**
* The style of the an option in the dropdown menu. It could be single lined option or an option with more and custom info
*/
Expand Down Expand Up @@ -332,6 +383,12 @@ export type TSearchSelectFieldProps = {
iconLeft?: ReactNode;
};

type TOptionInnerPropsData = {
label?: string;
key?: string;
id?: string;
};

const defaultProps: Pick<TSearchSelectFieldProps, 'controlShouldRenderValue'> =
{
controlShouldRenderValue: true,
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import type { Meta, StoryFn } from '@storybook/react';
import SearchSelectInput from './search-select-input';
import SearchSelectInput, {
TOptionInnerPropsData,
} from './search-select-input';
import { iconArgType } from '@/storybook-helpers';
import { useState } from 'react';
import Spacings from '@commercetools-uikit/spacings';
Expand Down Expand Up @@ -62,16 +64,16 @@ const loadOptions = (inputValue: string) =>
delay(500).then(() => filterColors(inputValue));

export const BasicExample: Story = ({ isMulti, ...args }) => {
const [value, onChange] = useState<string | string[] | undefined>(
isMulti ? [] : undefined
);
const [value, onChange] = useState<
TOptionInnerPropsData | readonly TOptionInnerPropsData[] | null | undefined
>(isMulti ? [] : null);

return (
<div style={{ height: 400 }}>
<Spacings.Stack scale="m">
<SearchSelectInput
onChange={(event) => {
onChange(event.target.value as string | string[] | undefined);
onChange(event.target.value);
}}
value={value}
{...args}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -345,7 +345,7 @@ export type TSearchSelectInputProps<
iconLeft?: ReactNode;
};

type TOptionInnerPropsData = {
export type TOptionInnerPropsData = {
label?: string;
key?: string;
id?: string;
Expand Down

0 comments on commit c3673d2

Please sign in to comment.