-
-
Notifications
You must be signed in to change notification settings - Fork 87
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
[Slider] Narrow the type of value
in callbacks
#1241
base: master
Are you sure you want to change the base?
Conversation
|
||
/** | ||
* Groups all parts of the slider. | ||
* Renders a `<div>` element. | ||
* | ||
* Documentation: [Base UI Slider](https://base-ui.com/react/components/slider) | ||
*/ | ||
const SliderRoot = React.forwardRef(function SliderRoot( | ||
props: SliderRoot.Props, | ||
const SliderRoot = fixedForwardRef(function SliderRoot<TValue extends number | number[]>( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ReactForwardRef does not work well with generics.
So we need to create a wrapper function to apply the generic.
This comment was marked as outdated.
This comment was marked as outdated.
Sorry, something went wrong.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Instead of making fixedForwardRef
, we could cast the result of React.forwardRef
, e.g. something like this:
type SliderRootType = {
<Value>(props: SliderRoot.Props<Value>): React.JSX.Element;
propTypes?: any;
}
const SliderRoot = React.forwardRef(function SliderRoot<Value>(
props: SliderRoot.Props<Value>,
forwardedRef: React.ForwardedRef<HTMLDivElement>
) {
//
}) as SliderRootType;
We did something similar in the legacy package: https://github.com/mui/material-ui/blob/master/packages/mui-base/src/Select/Select.tsx#L235
@@ -157,28 +158,25 @@ export namespace SliderRoot { | |||
values: ReadonlyArray<number>; | |||
} | |||
|
|||
export interface Props | |||
export interface Props<TValue extends number | number[]> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My first approach here was to try to carry this generic all the way down in the slider components but it resulted to just pollute the code with generics.
Instead I simple just try to modify the top level component that the consumer will use.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@seloner
I believe we can define Props without using generics. By utilizing intersection types, we can potentially make the necessary changes internally without affecting the external API. This approach could simplify the type definitions and improve readability.
Here’s an example of how we can implement this:
@@ -158,7 +158,10 @@ export namespace SliderRoot {
values: ReadonlyArray<number>;
}
- export interface Props<TValue extends number | number[]>
+ export type Props = PropsTemplate<number> &
+ PropsTemplate<number[]> &
+ PropsTemplate<number | number[]>;
+ export interface PropsTemplate<TValue extends number | number[]>
extends Pick<
useSliderRoot.Parameters,
| 'disabled'
Please refer to mui/material-ui#44777 (comment)
Netlify deploy preview |
7f7d189
to
186397e
Compare
This comment was marked as outdated.
This comment was marked as outdated.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I’d like to suggest the following tasks to address this issue:
- Convert
SliderRoot.Props
to an Intersection Type: Change the Props definition to use an intersection type, eliminating the need for generics.
@@ -158,7 +158,10 @@ export namespace SliderRoot {
values: ReadonlyArray<number>;
}
- export interface Props<TValue extends number | number[]>
+ export type Props = PropsTemplate<number> &
+ PropsTemplate<number[]> &
+ PropsTemplate<number | number[]>;
+ export interface PropsTemplate<TValue extends number | number[]>
extends Pick<
useSliderRoot.Parameters,
| 'disabled'
- Remove Generic Type from SliderRoot: Simplify SliderRoot by removing the generic type.
- Replace
fixedForwardRef
withReact.forwardRef
: Update the code to useReact.forwardRef
directly, which will allow us to remove the// @ts-expect-error
.
@@ -18,8 +18,8 @@ import { fixedForwardRef } from '../utils';
*
* Documentation: [Base UI Slider](https://base-ui.com/react/components/slider)
*/
-const SliderRoot = fixedForwardRef(function SliderRoot<TValue extends number | number[]>(
- props: SliderRoot.Props<TValue>,
+const SliderRoot = React.forwardRef(function SliderRoot(
+ props: SliderRoot.Props,
forwardedRef: React.ForwardedRef<HTMLDivElement>,
) {
const {
- Remove
// @ts-expect-error
: With the changes above, the TypeScript error should be resolved, making the// @ts-expect-error
unnecessary.
These changes should help resolve the type-related issues. Let me know your thoughts!
@@ -157,28 +158,25 @@ export namespace SliderRoot { | |||
values: ReadonlyArray<number>; | |||
} | |||
|
|||
export interface Props | |||
export interface Props<TValue extends number | number[]> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@seloner
I believe we can define Props without using generics. By utilizing intersection types, we can potentially make the necessary changes internally without affecting the external API. This approach could simplify the type definitions and improve readability.
Here’s an example of how we can implement this:
@@ -158,7 +158,10 @@ export namespace SliderRoot {
values: ReadonlyArray<number>;
}
- export interface Props<TValue extends number | number[]>
+ export type Props = PropsTemplate<number> &
+ PropsTemplate<number[]> &
+ PropsTemplate<number | number[]>;
+ export interface PropsTemplate<TValue extends number | number[]>
extends Pick<
useSliderRoot.Parameters,
| 'disabled'
Please refer to mui/material-ui#44777 (comment)
|
||
/** | ||
* Groups all parts of the slider. | ||
* Renders a `<div>` element. | ||
* | ||
* Documentation: [Base UI Slider](https://base-ui.com/react/components/slider) | ||
*/ | ||
const SliderRoot = React.forwardRef(function SliderRoot( | ||
props: SliderRoot.Props, | ||
const SliderRoot = fixedForwardRef(function SliderRoot<TValue extends number | number[]>( |
This comment was marked as outdated.
This comment was marked as outdated.
Sorry, something went wrong.
186397e
to
11792ab
Compare
@good-jinu |
@seloner My bad
So It seems like just a field like value doesn't work with it. For example: export type Props = PropsTemplate<number> |
PropsTemplate<number[]> |
PropsTemplate<number | number[]>; |
It does not work |
@seloner Thanks for working on this, I'm afraid this isn't an option though
@good-jinu Not sure if I understand you correctly, by "field like value" did you mean a single non-array value? |
We can still do the solution with the generics |
I think this is ok as long as it doesn't affect the public API, just wonder why @good-jinu is against generics (I prefer to avoid them too, but I'm just curious here) A while back we did decide that for components with values that could sometimes be an array (accordion, slider, toggle group), we'd make the value in callbacks always be an array so it's predictable, also curious to see what you all think about this! |
We could use a default value so it does no really affect the public API compared to now . type Props<T extends number | number[] = number | number[]> = {
value: T;
onValueChange: (value: T) => void;
};
const Component: <T extends number | number[]>(props: Props<T>) => {}; I like your suggestion to remove the number case and always go with an array. I am new to this project so I can't really answer though what is the best solution here. |
I actually agree with using generics; however, I wanted to avoid changing the public API. My main intention was to eliminate the I also think it's fine as long as it doesn't change the public API. |
8cc31b2
to
15ae9de
Compare
Updated. |
@@ -157,16 +158,14 @@ export namespace SliderRoot { | |||
values: ReadonlyArray<number>; | |||
} | |||
|
|||
export interface Props | |||
export interface Props<TValue extends number[] | number = number | number[]> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
export interface Props<TValue extends number[] | number = number | number[]> | |
export interface Props<Value extends readonly number[] | number = number | readonly number[]> |
Value
is fine ~ we don't use the T*
convention, also the array form of value/defaultValue should be readonly arrays
value
in callbacks
|
||
/** | ||
* Groups all parts of the slider. | ||
* Renders a `<div>` element. | ||
* | ||
* Documentation: [Base UI Slider](https://base-ui.com/react/components/slider) | ||
*/ | ||
const SliderRoot = React.forwardRef(function SliderRoot( | ||
props: SliderRoot.Props, | ||
const SliderRoot = fixedForwardRef(function SliderRoot<TValue extends number | number[]>( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Instead of making fixedForwardRef
, we could cast the result of React.forwardRef
, e.g. something like this:
type SliderRootType = {
<Value>(props: SliderRoot.Props<Value>): React.JSX.Element;
propTypes?: any;
}
const SliderRoot = React.forwardRef(function SliderRoot<Value>(
props: SliderRoot.Props<Value>,
forwardedRef: React.ForwardedRef<HTMLDivElement>
) {
//
}) as SliderRootType;
We did something similar in the legacy package: https://github.com/mui/material-ui/blob/master/packages/mui-base/src/Select/Select.tsx#L235
@seloner Since this would be a breaking change, let's keep going with the solution using generics! I've left some comments, you should also merge/rebase master first btw ~ |
2bff7c2
to
41654ae
Compare
type SliderRootType = { | ||
<Value extends number | readonly number[]>( | ||
props: SliderRoot.Props<Value> & { | ||
ref?: React.RefObject<HTMLDivElement>; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This was needed to include the ref into types @mj12albert
} | ||
} | ||
|
||
export { SliderRoot }; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@mj12albert
This was needed because ts error
Parameter props of call signature from exported interface has or is using private name
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I managed to bypass this with
1b5a14d
✅ Deploy Preview for base-ui ready!
To edit notification comments on pull requests, go to your Netlify site configuration. |
Updated |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Great work on the improvements! Since it has no more ts error, I guess it is ready.
@good-jinu I don't have access here. What is wrong ? 🤔 Can you help me? |
It seems like the error @mj12albert can you help us fixing this? |
onValueChange: (onValueChangeProp as useSliderRoot.Parameters['onValueChange']) ?? NOOP, | ||
onValueCommitted: | ||
(onValueCommittedProp as useSliderRoot.Parameters['onValueCommitted']) ?? NOOP, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@seloner It should be possible without casting these two props (the legacy useSelect
could also be used as a reference)
- Add a type parameter to
useSliderRoot
as well, e.g. https://github.com/mui/material-ui/blob/master/packages/mui-base/src/useSelect/useSelect.types.ts#L22 - Export a type (from
useSliderRoot
) to use inSliderRoot
as the generic, e.g.export type SliderValue<Value> = Value extends number ? number : number[]
(reference)
Let me know if you need any help ~
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@seloner See if this helps: https://github.com/mj12albert/base-ui/commits/slider-value-type/
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I tried yesterday and I was not able to make it work.
Typescript was not correctly infering the type.
I will check again today with your suggestion and update 👍
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@mj12albert
Are you sure your approach is working?
I created a minimal sandbox with the recommended approach but types are not working correctly 🤔
https://stackblitz.com/edit/vitejs-vite-k75jr9vi?file=src%2FApp.tsx
Am I missing something?
I have tried your fork also but it does not work 🤔
Trying to fix
Closes #1230