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

[Slider] Narrow the type of value in callbacks #1241

Open
wants to merge 4 commits into
base: master
Choose a base branch
from

Conversation

seloner
Copy link

@seloner seloner commented Dec 28, 2024

Trying to fix
image

Closes #1230


/**
* 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[]>(
Copy link
Author

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.

Copy link
Member

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[]>
Copy link
Author

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.

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)

@mui-bot
Copy link

mui-bot commented Dec 28, 2024

Netlify deploy preview

https://deploy-preview-1241--base-ui.netlify.app/

Generated by 🚫 dangerJS against 2bff7c2

@seloner seloner changed the title [Slider]: restrict slider type [Slider] restrict slider type Dec 29, 2024
@seloner seloner marked this pull request as ready for review December 29, 2024 00:04
@oliviertassinari oliviertassinari added the component: slider This is the name of the generic UI component, not the React module! label Dec 29, 2024
@good-jinu

This comment was marked as outdated.

Copy link

@good-jinu good-jinu left a 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:

  1. 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'
  1. Remove Generic Type from SliderRoot: Simplify SliderRoot by removing the generic type.
  2. Replace fixedForwardRef with React.forwardRef: Update the code to use React.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 {
  1. 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!

packages/react/src/slider/root/SliderRoot.tsx Outdated Show resolved Hide resolved
@@ -157,28 +158,25 @@ export namespace SliderRoot {
values: ReadonlyArray<number>;
}

export interface Props
export interface Props<TValue extends number | number[]>

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)

packages/react/src/slider/root/SliderRoot.test.tsx Outdated Show resolved Hide resolved

/**
* 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.

packages/react/src/slider/root/SliderRoot.tsx Outdated Show resolved Hide resolved
@seloner
Copy link
Author

seloner commented Dec 29, 2024

@good-jinu
I have applied your suggestions but I can't make this work 🤔
SlideRoot.test.tsx broke

image

@good-jinu
Copy link

@seloner My bad

An intersection of function types is treated as overloaded signatures in TypeScript.

So It seems like just a field like value doesn't work with it.
How about union type?

For example:

export type Props =  PropsTemplate<number> |
  PropsTemplate<number[]> |
  PropsTemplate<number | number[]>;

@seloner
Copy link
Author

seloner commented Dec 29, 2024

 PropsTemplate<number> |
  PropsTemplate<number[]> |
  PropsTemplate<number | number[]>;

It does not work
I think the generic is needed.
Another approach is to export two SliderRoots one SliderRoot and one RangeSliderRoot to support both types without generic?

@mj12albert
Copy link
Member

Another approach is to export two SliderRoots one SliderRoot and one RangeSliderRoot to support both types without generic?

@seloner Thanks for working on this, I'm afraid this isn't an option though

So It seems like just a field like value doesn't work with it.

@good-jinu Not sure if I understand you correctly, by "field like value" did you mean a single non-array value?

@seloner
Copy link
Author

seloner commented Dec 30, 2024

Another approach is to export two SliderRoots one SliderRoot and one RangeSliderRoot to support both types without generic?

@seloner Thanks for working on this, I'm afraid this isn't an option though

So It seems like just a field like value doesn't work with it.

@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

@mj12albert
Copy link
Member

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!

@seloner
Copy link
Author

seloner commented Dec 30, 2024

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.
It will work for this specific case.
The only issue I see that it tries to bypass the problem and not solve it.
In this case it will be fine but in general for a more complex case it will not work.
For example if there is a case with possible inputs string/number.

I am new to this project so I can't really answer though what is the best solution here.

@good-jinu
Copy link

good-jinu commented Dec 31, 2024

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)

I actually agree with using generics; however, I wanted to avoid changing the public API. My main intention was to eliminate the @ts-expect-error from previous commits due to the TypeScript error.

I also think it's fine as long as it doesn't change the public API.

@seloner seloner force-pushed the fix/slider-types branch 2 times, most recently from 8cc31b2 to 15ae9de Compare January 3, 2025 19:02
@seloner
Copy link
Author

seloner commented Jan 3, 2025

@good-jinu @mj12albert

Updated.
Used the defaults for generics.
The only issue that I can't solve is proptypes.

@github-actions github-actions bot added the PR: out-of-date The pull request has merge conflicts and can't be merged label Jan 7, 2025
@@ -157,16 +158,14 @@ export namespace SliderRoot {
values: ReadonlyArray<number>;
}

export interface Props
export interface Props<TValue extends number[] | number = number | number[]>
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
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

@mj12albert mj12albert changed the title [Slider] restrict slider type [Slider] Narrow the type of value in callbacks Jan 8, 2025

/**
* 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[]>(
Copy link
Member

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

@mj12albert
Copy link
Member

I like your suggestion to remove the number case and always go with an array.

@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 ~

@github-actions github-actions bot removed the PR: out-of-date The pull request has merge conflicts and can't be merged label Jan 8, 2025
type SliderRootType = {
<Value extends number | readonly number[]>(
props: SliderRoot.Props<Value> & {
ref?: React.RefObject<HTMLDivElement>;
Copy link
Author

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 };
Copy link
Author

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

Copy link
Author

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

Copy link

netlify bot commented Jan 8, 2025

Deploy Preview for base-ui ready!

Name Link
🔨 Latest commit 64b61c8
🔍 Latest deploy log https://app.netlify.com/sites/base-ui/deploys/6785bd063c45d10008d1e232
😎 Deploy Preview https://deploy-preview-1241--base-ui.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

@seloner
Copy link
Author

seloner commented Jan 8, 2025

I like your suggestion to remove the number case and always go with an array.

@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 ~

Updated

Copy link

@good-jinu good-jinu left a 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.

@seloner
Copy link
Author

seloner commented Jan 9, 2025

image @good-jinu I don't have access here. What is wrong ? 🤔 Can you help me?

@good-jinu
Copy link

image @good-jinu I don't have access here. What is wrong ? 🤔 Can you help me?

It seems like the error There was an error fetching the storage URL during POST: 503 suggests network connectivity issues between CircleCI and the Codecov server. I think we need help fixing this CI.

@mj12albert can you help us fixing this?

Comment on lines +62 to +64
onValueChange: (onValueChangeProp as useSliderRoot.Parameters['onValueChange']) ?? NOOP,
onValueCommitted:
(onValueCommittedProp as useSliderRoot.Parameters['onValueCommitted']) ?? NOOP,
Copy link
Member

@mj12albert mj12albert Jan 14, 2025

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)

  1. 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
  2. Export a type (from useSliderRoot) to use in SliderRoot as the generic, e.g.export type SliderValue<Value> = Value extends number ? number : number[] (reference)

Let me know if you need any help ~

Copy link
Member

Choose a reason for hiding this comment

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

Copy link
Author

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 👍

Copy link
Author

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 🤔

@mj12albert mj12albert self-assigned this Jan 14, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
component: slider This is the name of the generic UI component, not the React module! typescript
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[Slider] value type should narrow onValueChange value type
5 participants