diff --git a/docs/data/material/pagesApi.js b/docs/data/material/pagesApi.js index 377857c79e5dff..0d8fdd2ebbda25 100644 --- a/docs/data/material/pagesApi.js +++ b/docs/data/material/pagesApi.js @@ -27,6 +27,7 @@ module.exports = [ { pathname: '/material-ui/api/checkbox' }, { pathname: '/material-ui/api/chip' }, { pathname: '/material-ui/api/circular-progress' }, + { pathname: '/material-ui/api/click-away-listener' }, { pathname: '/material-ui/api/collapse' }, { pathname: '/material-ui/api/container' }, { pathname: '/material-ui/api/css-baseline' }, @@ -76,6 +77,7 @@ module.exports = [ { pathname: '/material-ui/api/mobile-stepper' }, { pathname: '/material-ui/api/modal' }, { pathname: '/material-ui/api/native-select' }, + { pathname: '/material-ui/api/no-ssr' }, { pathname: '/material-ui/api/outlined-input' }, { pathname: '/material-ui/api/pagination' }, { pathname: '/material-ui/api/pagination-item' }, @@ -86,6 +88,7 @@ module.exports = [ { pathname: '/material-ui/api/pigment-stack' }, { pathname: '/material-ui/api/popover' }, { pathname: '/material-ui/api/popper' }, + { pathname: '/material-ui/api/portal' }, { pathname: '/material-ui/api/radio' }, { pathname: '/material-ui/api/radio-group' }, { pathname: '/material-ui/api/rating' }, @@ -125,6 +128,7 @@ module.exports = [ { pathname: '/material-ui/api/tab-panel' }, { pathname: '/material-ui/api/tabs' }, { pathname: '/material-ui/api/tab-scroll-button' }, + { pathname: '/material-ui/api/textarea-autosize' }, { pathname: '/material-ui/api/text-field' }, { pathname: '/material-ui/api/timeline' }, { pathname: '/material-ui/api/timeline-connector' }, diff --git a/docs/pages/material-ui/api/click-away-listener.js b/docs/pages/material-ui/api/click-away-listener.js new file mode 100644 index 00000000000000..3776a973527b16 --- /dev/null +++ b/docs/pages/material-ui/api/click-away-listener.js @@ -0,0 +1,23 @@ +import * as React from 'react'; +import ApiPage from 'docs/src/modules/components/ApiPage'; +import mapApiPageTranslations from 'docs/src/modules/utils/mapApiPageTranslations'; +import jsonPageContent from './click-away-listener.json'; + +export default function Page(props) { + const { descriptions, pageContent } = props; + return ; +} + +Page.getInitialProps = () => { + const req = require.context( + 'docs/translations/api-docs/click-away-listener', + false, + /\.\/click-away-listener.*.json$/, + ); + const descriptions = mapApiPageTranslations(req); + + return { + descriptions, + pageContent: jsonPageContent, + }; +}; diff --git a/docs/pages/material-ui/api/click-away-listener.json b/docs/pages/material-ui/api/click-away-listener.json new file mode 100644 index 00000000000000..a33e412e20cdac --- /dev/null +++ b/docs/pages/material-ui/api/click-away-listener.json @@ -0,0 +1,34 @@ +{ + "props": { + "children": { "type": { "name": "custom", "description": "element" }, "required": true }, + "onClickAway": { "type": { "name": "func" }, "required": true }, + "disableReactTree": { "type": { "name": "bool" }, "default": "false" }, + "mouseEvent": { + "type": { + "name": "enum", + "description": "'onClick'
| 'onMouseDown'
| 'onMouseUp'
| 'onPointerDown'
| 'onPointerUp'
| false" + }, + "default": "'onClick'" + }, + "touchEvent": { + "type": { + "name": "enum", + "description": "'onTouchEnd'
| 'onTouchStart'
| false" + }, + "default": "'onTouchEnd'" + } + }, + "name": "ClickAwayListener", + "imports": [ + "import ClickAwayListener from '@mui/material/ClickAwayListener';", + "import { ClickAwayListener } from '@mui/material';" + ], + "classes": [], + "spread": false, + "themeDefaultProps": null, + "muiName": "MuiClickAwayListener", + "filename": "/packages/mui-material/src/ClickAwayListener/ClickAwayListener.tsx", + "inheritance": null, + "demos": "", + "cssComponent": false +} diff --git a/docs/pages/material-ui/api/no-ssr.js b/docs/pages/material-ui/api/no-ssr.js new file mode 100644 index 00000000000000..03c084bbcf6a12 --- /dev/null +++ b/docs/pages/material-ui/api/no-ssr.js @@ -0,0 +1,19 @@ +import * as React from 'react'; +import ApiPage from 'docs/src/modules/components/ApiPage'; +import mapApiPageTranslations from 'docs/src/modules/utils/mapApiPageTranslations'; +import jsonPageContent from './no-ssr.json'; + +export default function Page(props) { + const { descriptions, pageContent } = props; + return ; +} + +Page.getInitialProps = () => { + const req = require.context('docs/translations/api-docs/no-ssr', false, /\.\/no-ssr.*.json$/); + const descriptions = mapApiPageTranslations(req); + + return { + descriptions, + pageContent: jsonPageContent, + }; +}; diff --git a/docs/pages/material-ui/api/no-ssr.json b/docs/pages/material-ui/api/no-ssr.json new file mode 100644 index 00000000000000..464dc7136b18ca --- /dev/null +++ b/docs/pages/material-ui/api/no-ssr.json @@ -0,0 +1,17 @@ +{ + "props": { + "children": { "type": { "name": "node" } }, + "defer": { "type": { "name": "bool" }, "default": "false" }, + "fallback": { "type": { "name": "node" }, "default": "null" } + }, + "name": "NoSsr", + "imports": ["import NoSsr from '@mui/material/NoSsr';", "import { NoSsr } from '@mui/material';"], + "classes": [], + "spread": false, + "themeDefaultProps": null, + "muiName": "MuiNoSsr", + "filename": "/packages/mui-material/src/NoSsr/NoSsr.tsx", + "inheritance": null, + "demos": "", + "cssComponent": false +} diff --git a/docs/pages/material-ui/api/popper.json b/docs/pages/material-ui/api/popper.json index 94c41b76c2bb8f..761d7f7c469311 100644 --- a/docs/pages/material-ui/api/popper.json +++ b/docs/pages/material-ui/api/popper.json @@ -67,7 +67,14 @@ "import Popper from '@mui/material/Popper';", "import { Popper } from '@mui/material';" ], - "classes": [], + "classes": [ + { + "key": "root", + "className": "MuiPopper-root", + "description": "Class name applied to the root element.", + "isGlobal": false + } + ], "spread": true, "themeDefaultProps": false, "muiName": "MuiPopper", diff --git a/docs/pages/material-ui/api/portal.js b/docs/pages/material-ui/api/portal.js new file mode 100644 index 00000000000000..e53de2d6983e86 --- /dev/null +++ b/docs/pages/material-ui/api/portal.js @@ -0,0 +1,19 @@ +import * as React from 'react'; +import ApiPage from 'docs/src/modules/components/ApiPage'; +import mapApiPageTranslations from 'docs/src/modules/utils/mapApiPageTranslations'; +import jsonPageContent from './portal.json'; + +export default function Page(props) { + const { descriptions, pageContent } = props; + return ; +} + +Page.getInitialProps = () => { + const req = require.context('docs/translations/api-docs/portal', false, /\.\/portal.*.json$/); + const descriptions = mapApiPageTranslations(req); + + return { + descriptions, + pageContent: jsonPageContent, + }; +}; diff --git a/docs/pages/material-ui/api/portal.json b/docs/pages/material-ui/api/portal.json new file mode 100644 index 00000000000000..1079011f1f82f4 --- /dev/null +++ b/docs/pages/material-ui/api/portal.json @@ -0,0 +1,20 @@ +{ + "props": { + "children": { "type": { "name": "node" } }, + "container": { "type": { "name": "union", "description": "HTML element
| func" } }, + "disablePortal": { "type": { "name": "bool" }, "default": "false" } + }, + "name": "Portal", + "imports": [ + "import Portal from '@mui/material/Portal';", + "import { Portal } from '@mui/material';" + ], + "classes": [], + "spread": false, + "themeDefaultProps": null, + "muiName": "MuiPortal", + "filename": "/packages/mui-material/src/Portal/Portal.tsx", + "inheritance": null, + "demos": "", + "cssComponent": false +} diff --git a/docs/pages/material-ui/api/textarea-autosize.js b/docs/pages/material-ui/api/textarea-autosize.js new file mode 100644 index 00000000000000..4a8184ae71999b --- /dev/null +++ b/docs/pages/material-ui/api/textarea-autosize.js @@ -0,0 +1,23 @@ +import * as React from 'react'; +import ApiPage from 'docs/src/modules/components/ApiPage'; +import mapApiPageTranslations from 'docs/src/modules/utils/mapApiPageTranslations'; +import jsonPageContent from './textarea-autosize.json'; + +export default function Page(props) { + const { descriptions, pageContent } = props; + return ; +} + +Page.getInitialProps = () => { + const req = require.context( + 'docs/translations/api-docs/textarea-autosize', + false, + /\.\/textarea-autosize.*.json$/, + ); + const descriptions = mapApiPageTranslations(req); + + return { + descriptions, + pageContent: jsonPageContent, + }; +}; diff --git a/docs/pages/material-ui/api/textarea-autosize.json b/docs/pages/material-ui/api/textarea-autosize.json new file mode 100644 index 00000000000000..06a5fbb2bf09b3 --- /dev/null +++ b/docs/pages/material-ui/api/textarea-autosize.json @@ -0,0 +1,22 @@ +{ + "props": { + "maxRows": { "type": { "name": "union", "description": "number
| string" } }, + "minRows": { + "type": { "name": "union", "description": "number
| string" }, + "default": "1" + } + }, + "name": "TextareaAutosize", + "imports": [ + "import TextareaAutosize from '@mui/material/TextareaAutosize';", + "import { TextareaAutosize } from '@mui/material';" + ], + "classes": [], + "spread": true, + "themeDefaultProps": null, + "muiName": "MuiTextareaAutosize", + "filename": "/packages/mui-material/src/TextareaAutosize/TextareaAutosize.tsx", + "inheritance": null, + "demos": "", + "cssComponent": false +} diff --git a/docs/translations/api-docs/click-away-listener/click-away-listener.json b/docs/translations/api-docs/click-away-listener/click-away-listener.json new file mode 100644 index 00000000000000..28136c93c705af --- /dev/null +++ b/docs/translations/api-docs/click-away-listener/click-away-listener.json @@ -0,0 +1,19 @@ +{ + "componentDescription": "Listen for click events that occur somewhere in the document, outside of the element itself.\nFor instance, if you need to hide a menu when people click anywhere else on your page.", + "propDescriptions": { + "children": { "description": "The wrapped element.", "requiresRef": true }, + "disableReactTree": { + "description": "If true, the React tree is ignored and only the DOM tree is considered. This prop changes how portaled elements are handled." + }, + "mouseEvent": { + "description": "The mouse event to listen to. You can disable the listener by providing false." + }, + "onClickAway": { + "description": "Callback fired when a "click away" event is detected." + }, + "touchEvent": { + "description": "The touch event to listen to. You can disable the listener by providing false." + } + }, + "classDescriptions": {} +} diff --git a/docs/translations/api-docs/no-ssr/no-ssr.json b/docs/translations/api-docs/no-ssr/no-ssr.json new file mode 100644 index 00000000000000..ff903d12a20b33 --- /dev/null +++ b/docs/translations/api-docs/no-ssr/no-ssr.json @@ -0,0 +1,11 @@ +{ + "componentDescription": "NoSsr purposely removes components from the subject of Server Side Rendering (SSR).\n\nThis component can be useful in a variety of situations:\n\n* Escape hatch for broken dependencies not supporting SSR.\n* Improve the time-to-first paint on the client by only rendering above the fold.\n* Reduce the rendering time on the server.\n* Under too heavy server load, you can turn on service degradation.", + "propDescriptions": { + "children": { "description": "You can wrap a node." }, + "defer": { + "description": "If true, the component will not only prevent server-side rendering. It will also defer the rendering of the children into a different screen frame." + }, + "fallback": { "description": "The fallback content to display." } + }, + "classDescriptions": {} +} diff --git a/docs/translations/api-docs/popper/popper.json b/docs/translations/api-docs/popper/popper.json index 178879e0bd5352..23b39e61d6fa7a 100644 --- a/docs/translations/api-docs/popper/popper.json +++ b/docs/translations/api-docs/popper/popper.json @@ -41,5 +41,5 @@ "description": "Help supporting a react-transition-group/Transition component." } }, - "classDescriptions": {} + "classDescriptions": { "root": { "description": "Class name applied to the root element." } } } diff --git a/docs/translations/api-docs/portal/portal.json b/docs/translations/api-docs/portal/portal.json new file mode 100644 index 00000000000000..d7fb72bca569c9 --- /dev/null +++ b/docs/translations/api-docs/portal/portal.json @@ -0,0 +1,13 @@ +{ + "componentDescription": "Portals provide a first-class way to render children into a DOM node\nthat exists outside the DOM hierarchy of the parent component.", + "propDescriptions": { + "children": { "description": "The children to render into the container." }, + "container": { + "description": "An HTML element or function that returns one. The container will have the portal children appended to it.
You can also provide a callback, which is called in a React layout effect. This lets you set the container from a ref, and also makes server-side rendering possible.
By default, it uses the body of the top-level document object, so it's simply document.body most of the time." + }, + "disablePortal": { + "description": "The children will be under the DOM hierarchy of the parent component." + } + }, + "classDescriptions": {} +} diff --git a/docs/translations/api-docs/textarea-autosize/textarea-autosize.json b/docs/translations/api-docs/textarea-autosize/textarea-autosize.json new file mode 100644 index 00000000000000..ee91d0dd6d70c6 --- /dev/null +++ b/docs/translations/api-docs/textarea-autosize/textarea-autosize.json @@ -0,0 +1,8 @@ +{ + "componentDescription": "", + "propDescriptions": { + "maxRows": { "description": "Maximum number of rows to display." }, + "minRows": { "description": "Minimum number of rows to display." } + }, + "classDescriptions": {} +} diff --git a/packages/api-docs-builder-core/baseUi/projectSettings.ts b/packages/api-docs-builder-core/baseUi/projectSettings.ts index d032f45b30f755..5c276df37500e9 100644 --- a/packages/api-docs-builder-core/baseUi/projectSettings.ts +++ b/packages/api-docs-builder-core/baseUi/projectSettings.ts @@ -27,6 +27,9 @@ export const projectSettings: ProjectSettings = { getHookInfo: getBaseUiHookInfo, translationLanguages: LANGUAGES, skipComponent: () => false, + skipHook: (filename) => { + return filename.match(/(useSlotProps)/) !== null; + }, onCompleted: async () => { await generateBaseUIApiPages(); }, diff --git a/packages/api-docs-builder/ProjectSettings.ts b/packages/api-docs-builder/ProjectSettings.ts index 4844b59a7cc289..42af79c27432eb 100644 --- a/packages/api-docs-builder/ProjectSettings.ts +++ b/packages/api-docs-builder/ProjectSettings.ts @@ -54,6 +54,10 @@ export interface ProjectSettings { * Fuction called to detemine whether to skip the generation of a particular component's API docs */ skipComponent: (filename: string) => boolean; + /** + * Fuction called to detemine whether to skip the generation of a particular hook's API docs + */ + skipHook?: (filename: string) => boolean; /** * Determine is the component definition should be updated. */ diff --git a/packages/api-docs-builder/buildApi.ts b/packages/api-docs-builder/buildApi.ts index fd8d324599377b..38fe6469cedf29 100644 --- a/packages/api-docs-builder/buildApi.ts +++ b/packages/api-docs-builder/buildApi.ts @@ -134,6 +134,9 @@ async function buildSingleProject( ); const projectHooks = findHooks(path.join(project.rootPath, 'src')).filter((hook) => { + if (projectSettings.skipHook?.(hook.filename)) { + return false; + } if (grep === null) { return true; } diff --git a/packages/mui-base/src/utils/appendOwnerState.ts b/packages/mui-base/src/utils/appendOwnerState.ts index 965867cd8e71f0..9c4b778bef1add 100644 --- a/packages/mui-base/src/utils/appendOwnerState.ts +++ b/packages/mui-base/src/utils/appendOwnerState.ts @@ -1,51 +1,3 @@ -import * as React from 'react'; -import { Simplify } from '@mui/types'; -import { isHostComponent } from './isHostComponent'; +export { default as appendOwnerState } from '@mui/utils/appendOwnerState'; -/** - * Type of the ownerState based on the type of an element it applies to. - * This resolves to the provided OwnerState for React components and `undefined` for host components. - * Falls back to `OwnerState | undefined` when the exact type can't be determined in development time. - */ -type OwnerStateWhenApplicable = - ElementType extends React.ComponentType - ? OwnerState - : ElementType extends keyof React.JSX.IntrinsicElements - ? undefined - : OwnerState | undefined; - -export type AppendOwnerStateReturnType< - ElementType extends React.ElementType, - OtherProps, - OwnerState, -> = Simplify< - OtherProps & { - ownerState: OwnerStateWhenApplicable; - } ->; - -/** - * Appends the ownerState object to the props, merging with the existing one if necessary. - * - * @param elementType Type of the element that owns the `existingProps`. If the element is a DOM node or undefined, `ownerState` is not applied. - * @param otherProps Props of the element. - * @param ownerState - */ -export function appendOwnerState< - ElementType extends React.ElementType, - OtherProps extends Record, - OwnerState, ->( - elementType: ElementType | undefined, - otherProps: OtherProps, - ownerState: OwnerState, -): AppendOwnerStateReturnType { - if (elementType === undefined || isHostComponent(elementType)) { - return otherProps as AppendOwnerStateReturnType; - } - - return { - ...otherProps, - ownerState: { ...otherProps.ownerState, ...ownerState }, - } as AppendOwnerStateReturnType; -} +export type { AppendOwnerStateReturnType } from '@mui/utils/appendOwnerState'; diff --git a/packages/mui-base/src/utils/extractEventHandlers.ts b/packages/mui-base/src/utils/extractEventHandlers.ts index e3998c358e0240..9cead2050dc828 100644 --- a/packages/mui-base/src/utils/extractEventHandlers.ts +++ b/packages/mui-base/src/utils/extractEventHandlers.ts @@ -1,30 +1 @@ -import { EventHandlers } from './types'; - -/** - * Extracts event handlers from a given object. - * A prop is considered an event handler if it is a function and its name starts with `on`. - * - * @param object An object to extract event handlers from. - * @param excludeKeys An array of keys to exclude from the returned object. - */ -export function extractEventHandlers( - object: Record | undefined, - excludeKeys: string[] = [], -): EventHandlers { - if (object === undefined) { - return {}; - } - - const result: EventHandlers = {}; - - Object.keys(object) - .filter( - (prop) => - prop.match(/^on[A-Z]/) && typeof object[prop] === 'function' && !excludeKeys.includes(prop), - ) - .forEach((prop) => { - result[prop] = object[prop]; - }); - - return result; -} +export { default as extractEventHandlers } from '@mui/utils/extractEventHandlers'; diff --git a/packages/mui-base/src/utils/types.ts b/packages/mui-base/src/utils/types.ts index 4d2a96ce8f99e1..3a1883cec220b7 100644 --- a/packages/mui-base/src/utils/types.ts +++ b/packages/mui-base/src/utils/types.ts @@ -1,27 +1,6 @@ -import * as React from 'react'; - -export type EventHandlers = Record>; - -export type WithOptionalOwnerState = Omit< - Props, - 'ownerState' -> & - Partial>; - -export type SlotComponentProps = - | (Partial> & TOverrides) - | (( - ownerState: TOwnerState, - ) => Partial> & TOverrides); - -export type SlotComponentPropsWithSlotState< - TSlotComponent extends React.ElementType, - TOverrides, - TOwnerState, - TSlotState, -> = - | (Partial> & TOverrides) - | (( - ownerState: TOwnerState, - slotState: TSlotState, - ) => Partial> & TOverrides); +export type { + EventHandlers, + WithOptionalOwnerState, + SlotComponentProps, + SlotComponentPropsWithSlotState, +} from '@mui/utils'; diff --git a/packages/mui-base/src/utils/useSlotProps.ts b/packages/mui-base/src/utils/useSlotProps.ts index 10e40630768283..3d11566e23eb35 100644 --- a/packages/mui-base/src/utils/useSlotProps.ts +++ b/packages/mui-base/src/utils/useSlotProps.ts @@ -1,113 +1,4 @@ 'use client'; -import * as React from 'react'; -import { unstable_useForkRef as useForkRef } from '@mui/utils'; -import { appendOwnerState, AppendOwnerStateReturnType } from './appendOwnerState'; -import { - mergeSlotProps, - MergeSlotPropsParameters, - MergeSlotPropsResult, - WithCommonProps, -} from './mergeSlotProps'; -import { resolveComponentProps } from './resolveComponentProps'; +export { default as useSlotProps } from '@mui/utils/useSlotProps'; -export type UseSlotPropsParameters< - ElementType extends React.ElementType, - SlotProps, - ExternalForwardedProps, - ExternalSlotProps, - AdditionalProps, - OwnerState, -> = Omit< - MergeSlotPropsParameters, - 'externalSlotProps' -> & { - /** - * The type of the component used in the slot. - */ - elementType: ElementType | undefined; - /** - * The `slotProps.*` of the Base UI component. - */ - externalSlotProps: - | ExternalSlotProps - | ((ownerState: OwnerState) => ExternalSlotProps) - | undefined; - /** - * The ownerState of the Base UI component. - */ - ownerState: OwnerState; - /** - * Set to true if the slotProps callback should receive more props. - */ - skipResolvingSlotProps?: boolean; -}; - -export type UseSlotPropsResult< - ElementType extends React.ElementType, - SlotProps, - AdditionalProps, - OwnerState, -> = AppendOwnerStateReturnType< - ElementType, - MergeSlotPropsResult['props'] & { - ref: ((instance: any | null) => void) | null; - }, - OwnerState ->; - -/** - * @ignore - do not document. - * Builds the props to be passed into the slot of an unstyled component. - * It merges the internal props of the component with the ones supplied by the user, allowing to customize the behavior. - * If the slot component is not a host component, it also merges in the `ownerState`. - * - * @param parameters.getSlotProps - A function that returns the props to be passed to the slot component. - */ -export function useSlotProps< - ElementType extends React.ElementType, - SlotProps, - AdditionalProps, - OwnerState, ->( - parameters: UseSlotPropsParameters< - ElementType, - SlotProps, - object, - WithCommonProps>, - AdditionalProps, - OwnerState - >, -) { - const { - elementType, - externalSlotProps, - ownerState, - skipResolvingSlotProps = false, - ...rest - } = parameters; - const resolvedComponentsProps = skipResolvingSlotProps - ? {} - : resolveComponentProps(externalSlotProps, ownerState); - const { props: mergedProps, internalRef } = mergeSlotProps({ - ...rest, - externalSlotProps: resolvedComponentsProps, - }); - - const ref = useForkRef( - internalRef, - resolvedComponentsProps?.ref, - parameters.additionalProps?.ref, - ) as ((instance: any | null) => void) | null; - - const props: UseSlotPropsResult = - appendOwnerState( - elementType, - { - ...mergedProps, - ref, - }, - ownerState, - ); - - return props; -} +export type { UseSlotPropsParameters, UseSlotPropsResult } from '@mui/utils/useSlotProps'; diff --git a/packages/mui-material/package.json b/packages/mui-material/package.json index 757ccdd140b493..c7cd20ec85bdcc 100644 --- a/packages/mui-material/package.json +++ b/packages/mui-material/package.json @@ -41,7 +41,6 @@ }, "dependencies": { "@babel/runtime": "^7.24.7", - "@mui/base": "workspace:*", "@mui/core-downloads-tracker": "workspace:^", "@mui/system": "workspace:*", "@mui/types": "workspace:^", diff --git a/packages/mui-material/src/Autocomplete/Autocomplete.d.ts b/packages/mui-material/src/Autocomplete/Autocomplete.d.ts index b3cf3cb08df550..da407dc51dc891 100644 --- a/packages/mui-material/src/Autocomplete/Autocomplete.d.ts +++ b/packages/mui-material/src/Autocomplete/Autocomplete.d.ts @@ -1,8 +1,11 @@ import * as React from 'react'; import { SxProps } from '@mui/system'; import { OverridableStringUnion } from '@mui/types'; -import { - useAutocomplete, +import { IconButtonProps, InternalStandardProps as StandardProps, Theme } from '@mui/material'; +import { ChipProps, ChipTypeMap } from '@mui/material/Chip'; +import { PaperProps } from '@mui/material/Paper'; +import { PopperProps } from '@mui/material/Popper'; +import useAutocomplete, { AutocompleteChangeDetails, AutocompleteChangeReason, AutocompleteCloseReason, @@ -11,11 +14,7 @@ import { createFilterOptions, UseAutocompleteProps, AutocompleteFreeSoloValueMapping, -} from '@mui/base'; -import { IconButtonProps, InternalStandardProps as StandardProps, Theme } from '@mui/material'; -import { ChipProps, ChipTypeMap } from '@mui/material/Chip'; -import { PaperProps } from '@mui/material/Paper'; -import { PopperProps } from '@mui/material/Popper'; +} from '../useAutocomplete'; import { AutocompleteClasses } from './autocompleteClasses'; import { CreateSlotsAndSlotProps, SlotProps } from '../utils/types'; diff --git a/packages/mui-material/src/Autocomplete/Autocomplete.js b/packages/mui-material/src/Autocomplete/Autocomplete.js index 5f3f0014350ae0..b640a95cba538e 100644 --- a/packages/mui-material/src/Autocomplete/Autocomplete.js +++ b/packages/mui-material/src/Autocomplete/Autocomplete.js @@ -4,9 +4,9 @@ import PropTypes from 'prop-types'; import clsx from 'clsx'; import integerPropType from '@mui/utils/integerPropType'; import chainPropTypes from '@mui/utils/chainPropTypes'; -import { useAutocomplete, createFilterOptions } from '@mui/base'; import composeClasses from '@mui/utils/composeClasses'; import { alpha } from '@mui/system/colorManipulator'; +import useAutocomplete, { createFilterOptions } from '../useAutocomplete'; import Popper from '../Popper'; import ListSubheader from '../ListSubheader'; import Paper from '../Paper'; diff --git a/packages/mui-material/src/Badge/Badge.d.ts b/packages/mui-material/src/Badge/Badge.d.ts index a9811fa9a86552..fc4f37a095ad14 100644 --- a/packages/mui-material/src/Badge/Badge.d.ts +++ b/packages/mui-material/src/Badge/Badge.d.ts @@ -1,7 +1,7 @@ import * as React from 'react'; import { SxProps } from '@mui/system'; import { OverridableStringUnion, Simplify } from '@mui/types'; -import { SlotComponentProps } from '@mui/base/utils'; +import { SlotComponentProps } from '../utils/types'; import { Theme } from '../styles'; import { OverridableComponent, OverrideProps } from '../OverridableComponent'; import { BadgeClasses } from './badgeClasses'; diff --git a/packages/mui-material/src/Badge/Badge.js b/packages/mui-material/src/Badge/Badge.js index d3543f3c12715d..dfbcf3bb652643 100644 --- a/packages/mui-material/src/Badge/Badge.js +++ b/packages/mui-material/src/Badge/Badge.js @@ -4,8 +4,8 @@ import PropTypes from 'prop-types'; import clsx from 'clsx'; import usePreviousProps from '@mui/utils/usePreviousProps'; import composeClasses from '@mui/utils/composeClasses'; -import { useBadge } from '@mui/base/useBadge'; -import { useSlotProps } from '@mui/base/utils'; +import useSlotProps from '@mui/utils/useSlotProps'; +import useBadge from './useBadge'; import { styled } from '../zero-styled'; import { useDefaultProps } from '../DefaultPropsProvider'; import capitalize from '../utils/capitalize'; diff --git a/packages/mui-material/src/Badge/useBadge.ts b/packages/mui-material/src/Badge/useBadge.ts new file mode 100644 index 00000000000000..214f281d084c35 --- /dev/null +++ b/packages/mui-material/src/Badge/useBadge.ts @@ -0,0 +1,48 @@ +'use client'; +import * as React from 'react'; +import { usePreviousProps } from '@mui/utils'; +import { UseBadgeParameters, UseBadgeReturnValue } from './useBadge.types'; + +/** + * + * Demos: + * + * - [Badge](https://next.mui.com/base-ui/react-badge/#hook) + * + * API: + * + * - [useBadge API](https://next.mui.com/base-ui/react-badge/hooks-api/#use-badge) + */ +function useBadge(parameters: UseBadgeParameters): UseBadgeReturnValue { + const { + badgeContent: badgeContentProp, + invisible: invisibleProp = false, + max: maxProp = 99, + showZero = false, + } = parameters; + + const prevProps = usePreviousProps({ + badgeContent: badgeContentProp, + max: maxProp, + }); + + let invisible = invisibleProp; + + if (invisibleProp === false && badgeContentProp === 0 && !showZero) { + invisible = true; + } + + const { badgeContent, max = maxProp } = invisible ? prevProps : parameters; + + const displayValue: React.ReactNode = + badgeContent && Number(badgeContent) > max ? `${max}+` : badgeContent; + + return { + badgeContent, + invisible, + max, + displayValue, + }; +} + +export default useBadge; diff --git a/packages/mui-material/src/Badge/useBadge.types.ts b/packages/mui-material/src/Badge/useBadge.types.ts new file mode 100644 index 00000000000000..be6ef34830c5f8 --- /dev/null +++ b/packages/mui-material/src/Badge/useBadge.types.ts @@ -0,0 +1,40 @@ +export interface UseBadgeParameters { + /** + * The content rendered within the badge. + */ + badgeContent?: React.ReactNode; + /** + * If `true`, the badge is invisible. + * @default false + */ + invisible?: boolean; + /** + * Max count to show. + * @default 99 + */ + max?: number; + /** + * Controls whether the badge is hidden when `badgeContent` is zero. + * @default false + */ + showZero?: boolean; +} + +export interface UseBadgeReturnValue { + /** + * Defines the content that's displayed inside the badge. + */ + badgeContent: React.ReactNode; + /** + * If `true`, the component will not be visible. + */ + invisible: boolean; + /** + * Maximum number to be displayed in the badge. + */ + max: number; + /** + * Value to be displayed in the badge. If `badgeContent` is greater than `max`, it will return `max+`. + */ + displayValue: React.ReactNode; +} diff --git a/packages/mui-material/src/Breadcrumbs/Breadcrumbs.d.ts b/packages/mui-material/src/Breadcrumbs/Breadcrumbs.d.ts index 972f2460cc2dbb..4c58c98f7a7225 100644 --- a/packages/mui-material/src/Breadcrumbs/Breadcrumbs.d.ts +++ b/packages/mui-material/src/Breadcrumbs/Breadcrumbs.d.ts @@ -1,6 +1,6 @@ import * as React from 'react'; import { SxProps } from '@mui/system'; -import { SlotComponentProps } from '@mui/base'; +import { SlotComponentProps } from '../utils/types'; import { Theme } from '../styles'; import { OverridableComponent, OverrideProps } from '../OverridableComponent'; import { BreadcrumbsClasses } from './breadcrumbsClasses'; diff --git a/packages/mui-material/src/Breadcrumbs/Breadcrumbs.js b/packages/mui-material/src/Breadcrumbs/Breadcrumbs.js index 45a83fbbe6c62c..a398e73d9fd778 100644 --- a/packages/mui-material/src/Breadcrumbs/Breadcrumbs.js +++ b/packages/mui-material/src/Breadcrumbs/Breadcrumbs.js @@ -4,8 +4,8 @@ import { isFragment } from 'react-is'; import PropTypes from 'prop-types'; import clsx from 'clsx'; import integerPropType from '@mui/utils/integerPropType'; -import { useSlotProps } from '@mui/base/utils'; import composeClasses from '@mui/utils/composeClasses'; +import useSlotProps from '@mui/utils/useSlotProps'; import { styled } from '../zero-styled'; import { useDefaultProps } from '../DefaultPropsProvider'; import Typography from '../Typography'; diff --git a/packages/mui-material/src/ClickAwayListener/ClickAwayListener.test.js b/packages/mui-material/src/ClickAwayListener/ClickAwayListener.test.js new file mode 100644 index 00000000000000..e0c4c3730a4a82 --- /dev/null +++ b/packages/mui-material/src/ClickAwayListener/ClickAwayListener.test.js @@ -0,0 +1,441 @@ +import * as React from 'react'; +import * as ReactDOM from 'react-dom'; +import { expect } from 'chai'; +import { spy } from 'sinon'; +import { + act, + createRenderer, + fireEvent, + fireDiscreteEvent, + screen, +} from '@mui/internal-test-utils'; +import Portal from '@mui/material/Portal'; +import ClickAwayListener from '@mui/material/ClickAwayListener'; + +describe('', () => { + const { render: clientRender, clock } = createRenderer({ clock: 'fake' }); + /** + * @type {typeof plainRender extends (...args: infer T) => any ? T : never} args + * + * @remarks + * This is for all intents and purposes the same as our client render method. + * `plainRender` is already wrapped in act(). + * However, React has a bug that flushes effects in a portal synchronously. + * We have to defer the effect manually like `useEffect` would so we have to flush the effect manually instead of relying on `act()`. + * React bug: https://github.com/facebook/react/issues/20074 + */ + function render(...args) { + const result = clientRender(...args); + clock.tick(0); + return result; + } + + it('should render the children', () => { + const children = ; + const { container } = render( + {}}>{children}, + ); + expect(container.querySelectorAll('span').length).to.equal(1); + }); + + describe('prop: onClickAway', () => { + it('should be called when clicking away', () => { + const handleClickAway = spy(); + render( + + + , + ); + + fireEvent.click(document.body); + expect(handleClickAway.callCount).to.equal(1); + expect(handleClickAway.args[0].length).to.equal(1); + }); + + it('should not be called when clicking inside', () => { + const handleClickAway = spy(); + const { container } = render( + + + , + ); + + fireEvent.click(container.querySelector('span')); + expect(handleClickAway.callCount).to.equal(0); + }); + + it('should be called when preventDefault is `true`', () => { + const handleClickAway = spy(); + render( + + + , + ); + const preventDefault = (event) => event.preventDefault(); + document.body.addEventListener('click', preventDefault); + + fireEvent.click(document.body); + expect(handleClickAway.callCount).to.equal(1); + + document.body.removeEventListener('click', preventDefault); + }); + + it('should not be called when clicking inside a portaled element', () => { + const handleClickAway = spy(); + const { getByText } = render( + +
+ + Inside a portal + +
+
, + ); + + fireEvent.click(getByText('Inside a portal')); + expect(handleClickAway.callCount).to.equal(0); + }); + + it('should be called when clicking inside a portaled element and `disableReactTree` is `true`', () => { + const handleClickAway = spy(); + const { getByText } = render( + +
+ + Inside a portal + +
+
, + ); + + fireEvent.click(getByText('Inside a portal')); + expect(handleClickAway.callCount).to.equal(1); + }); + + it('should not be called even if the event propagation is stopped', () => { + const handleClickAway = spy(); + const { getByText } = render( + +
+
{ + event.stopPropagation(); + }} + > + Outside a portal +
+ + { + event.stopPropagation(); + }} + > + Stop inside a portal + + + + { + event.stopPropagation(); + event.nativeEvent.stopImmediatePropagation(); + }} + > + Stop all inside a portal + + +
+
, + ); + + fireEvent.click(getByText('Outside a portal')); + expect(handleClickAway.callCount).to.equal(0); + + fireEvent.click(getByText('Stop all inside a portal')); + expect(handleClickAway.callCount).to.equal(0); + + fireEvent.click(getByText('Stop inside a portal')); + // undesired behavior in React 16 + expect(handleClickAway.callCount).to.equal(React.version.startsWith('16') ? 1 : 0); + }); + + ['onClick', 'onClickCapture'].forEach((eventListenerName) => { + it(`should not be called when ${eventListenerName} mounted the listener`, () => { + function Test() { + const [open, setOpen] = React.useState(false); + + return ( + + + {toggle ? : } + + ); + } + + render(); + const button = screen.getByRole('button'); + fireEvent.click(button); + await waitFor(() => { + expect(screen.queryByText('LazyRoute')).not.to.equal(null); + }); + }); + + // For https://github.com/mui/material-ui/pull/33253 + it('should update height without an infinite rendering loop', async () => { + function App() { + const [value, setValue] = React.useState('Controlled'); + + const handleChange = (event: React.ChangeEvent) => { + setValue(event.target.value); + }; + + return ; + } + const { container } = render(); + const input = container.querySelector('textarea')!; + act(() => { + input.focus(); + }); + const activeElement = document.activeElement!; + // set the value of the input to be 1 larger than its content width + fireEvent.change(activeElement, { + target: { value: 'Controlled\n' }, + }); + await sleep(0); + fireEvent.change(activeElement, { + target: { value: 'Controlled\n\n' }, + }); + }); + + // For https://github.com/mui/material-ui/pull/37135 + it('should update height without delay', async function test() { + if (/jsdom/.test(window.navigator.userAgent)) { + // It depends on ResizeObserver + this.skip(); + } + + function App() { + const ref = React.useRef(null); + return ( +
+ +
+ +
+
+ ); + } + const { container } = render(); + const input = container.querySelector('textarea')!; + const button = screen.getByRole('button'); + expect(parseInt(input.style.height, 10)).to.be.within(30, 32); + fireEvent.click(button); + await raf(); + await raf(); + expect(parseInt(input.style.height, 10)).to.be.within(15, 17); + }); + + describe('layout', () => { + const getComputedStyleStub = new Map>(); + function setLayout( + input: HTMLTextAreaElement, + shadow: Element, + { + getComputedStyle, + scrollHeight: scrollHeightArg, + lineHeight: lineHeightArg, + }: { + getComputedStyle: Partial; + scrollHeight?: number | (() => number); + lineHeight?: number | (() => number); + }, + ) { + const lineHeight = typeof lineHeightArg === 'function' ? lineHeightArg : () => lineHeightArg; + const scrollHeight = + typeof scrollHeightArg === 'function' ? scrollHeightArg : () => scrollHeightArg; + + getComputedStyleStub.set(input, getComputedStyle); + + let index = 0; + stub(shadow, 'scrollHeight').get(() => { + index += 1; + return index % 2 === 1 ? scrollHeight() : lineHeight(); + }); + } + + before(function beforeHook() { + // Only run the test on node. + if (!/jsdom/.test(window.navigator.userAgent)) { + this.skip(); + } + + stub(window, 'getComputedStyle').value( + (node: Element) => getComputedStyleStub.get(node) || {}, + ); + }); + + after(() => { + sinon.restore(); + }); + + describe('resize', () => { + clock.withFakeTimers(); + + it('should handle the resize event', () => { + const { container } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + + expect(input.style).to.have.property('height', '0px'); + expect(input.style).to.have.property('overflow', 'hidden'); + + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'content-box', + }, + scrollHeight: 30, + lineHeight: 15, + }); + window.dispatchEvent(new window.Event('resize', {})); + + clock.tick(166); + + expect(input.style).to.have.property('height', '30px'); + expect(input.style).to.have.property('overflow', 'hidden'); + }); + }); + + it('should update when uncontrolled', () => { + const handleChange = spy(); + const { container } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + expect(input.style).to.have.property('height', '0px'); + expect(input.style).to.have.property('overflow', 'hidden'); + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'content-box', + }, + scrollHeight: 30, + lineHeight: 15, + }); + act(() => { + input.focus(); + }); + const activeElement = document.activeElement!; + fireEvent.change(activeElement, { target: { value: 'a' } }); + expect(input.style).to.have.property('height', '30px'); + expect(input.style).to.have.property('overflow', 'hidden'); + expect(handleChange.callCount).to.equal(1); + }); + + it('should take the border into account with border-box', () => { + const border = 5; + const { container, forceUpdate } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + expect(input.style).to.have.property('height', '0px'); + expect(input.style).to.have.property('overflow', 'hidden'); + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'border-box', + borderBottomWidth: `${border}px`, + }, + scrollHeight: 30, + lineHeight: 15, + }); + forceUpdate(); + expect(input.style).to.have.property('height', `${30 + border}px`); + expect(input.style).to.have.property('overflow', 'hidden'); + }); + + it('should take the padding into account with content-box', () => { + const padding = 5; + const { container, forceUpdate } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'border-box', + paddingTop: `${padding}px`, + }, + scrollHeight: 30, + lineHeight: 15, + }); + forceUpdate(); + expect(input.style).to.have.property('height', `${30 + padding}px`); + expect(input.style).to.have.property('overflow', 'hidden'); + }); + + it('should have at least height of "minRows"', () => { + const minRows = 3; + const lineHeight = 15; + const { container, forceUpdate } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'content-box', + }, + scrollHeight: 30, + lineHeight, + }); + forceUpdate(); + expect(input.style).to.have.property('height', `${lineHeight * minRows}px`); + expect(input.style).to.have.property('overflow', ''); + }); + + it('should have at max "maxRows" rows', () => { + const maxRows = 3; + const lineHeight = 15; + const { container, forceUpdate } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'content-box', + }, + scrollHeight: 100, + lineHeight, + }); + forceUpdate(); + expect(input.style).to.have.property('height', `${lineHeight * maxRows}px`); + expect(input.style).to.have.property('overflow', ''); + }); + + it('should show scrollbar when having more rows than "maxRows"', () => { + const maxRows = 3; + const lineHeight = 15; + const { container, forceUpdate } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'border-box', + }, + scrollHeight: lineHeight * 2, + lineHeight, + }); + forceUpdate(); + expect(input.style).to.have.property('height', `${lineHeight * 2}px`); + expect(input.style).to.have.property('overflow', 'hidden'); + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'border-box', + }, + scrollHeight: lineHeight * 3, + lineHeight, + }); + forceUpdate(); + expect(input.style).to.have.property('height', `${lineHeight * 3}px`); + expect(input.style).to.have.property('overflow', 'hidden'); + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'border-box', + }, + scrollHeight: lineHeight * 4, + lineHeight, + }); + forceUpdate(); + expect(input.style).to.have.property('height', `${lineHeight * 3}px`); + expect(input.style).to.have.property('overflow', ''); + }); + + it('should update its height when the "maxRows" prop changes', () => { + const lineHeight = 15; + const { container, forceUpdate, setProps } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'content-box', + }, + scrollHeight: 100, + lineHeight, + }); + forceUpdate(); + expect(input.style).to.have.property('height', `${lineHeight * 3}px`); + expect(input.style).to.have.property('overflow', ''); + setProps({ maxRows: 2 }); + expect(input.style).to.have.property('height', `${lineHeight * 2}px`); + expect(input.style).to.have.property('overflow', ''); + }); + + it('should not sync height if container width is 0px', () => { + const lineHeight = 15; + const { container, forceUpdate } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')!; + + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'content-box', + }, + scrollHeight: lineHeight * 2, + lineHeight, + }); + forceUpdate(); + + expect(input.style).to.have.property('height', `${lineHeight * 2}px`); + expect(input.style).to.have.property('overflow', 'hidden'); + + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'content-box', + width: '0px', + }, + scrollHeight: lineHeight * 3, + lineHeight, + }); + + forceUpdate(); + expect(input.style).to.have.property('height', `${lineHeight * 2}px`); + expect(input.style).to.have.property('overflow', 'hidden'); + }); + + it('should compute the correct height if padding-right is greater than 0px', () => { + const paddingRight = 50; + const { container, forceUpdate } = render(); + const input = container.querySelector('textarea[aria-hidden=null]')!; + const shadow = container.querySelector('textarea[aria-hidden=true]')! as HTMLTextAreaElement; + const contentWidth = 100; + const lineHeight = 15; + const width = contentWidth + paddingRight; + setLayout(input, shadow, { + getComputedStyle: { + boxSizing: 'border-box', + width: `${width}px`, + }, + scrollHeight: () => { + // assuming that the width of the word is 1px, and substract the width of the paddingRight + const lineNum = Math.ceil( + input.value.length / (width - getStyleValue(shadow.style.paddingRight)), + ); + return lineNum * lineHeight; + }, + lineHeight, + }); + + act(() => { + input.focus(); + }); + const activeElement = document.activeElement!; + // set the value of the input to be 1 larger than its content width + fireEvent.change(activeElement, { + target: { value: new Array(contentWidth + 1).fill('a').join('') }, + }); + forceUpdate(); + + // the input should be 2 lines + expect(input.style).to.have.property('height', `${lineHeight * 2}px`); + }); + }); + + it('should apply the inline styles using the "style" prop', function test() { + if (/jsdom/.test(window.navigator.userAgent)) { + this.skip(); + } + + const { container } = render(); + const input = container.querySelector('textarea')!; + + expect(input).toHaveComputedStyle({ + backgroundColor: 'rgb(255, 255, 0)', + }); + }); +}); diff --git a/packages/mui-material/src/TextareaAutosize/TextareaAutosize.tsx b/packages/mui-material/src/TextareaAutosize/TextareaAutosize.tsx new file mode 100644 index 00000000000000..c284416fd79f87 --- /dev/null +++ b/packages/mui-material/src/TextareaAutosize/TextareaAutosize.tsx @@ -0,0 +1,264 @@ +'use client'; +import * as React from 'react'; +import PropTypes from 'prop-types'; +import { + unstable_debounce as debounce, + unstable_useForkRef as useForkRef, + unstable_useEnhancedEffect as useEnhancedEffect, + unstable_ownerWindow as ownerWindow, +} from '@mui/utils'; +import { TextareaAutosizeProps } from './TextareaAutosize.types'; + +function getStyleValue(value: string) { + return parseInt(value, 10) || 0; +} + +const styles: { + shadow: React.CSSProperties; +} = { + shadow: { + // Visibility needed to hide the extra text area on iPads + visibility: 'hidden', + // Remove from the content flow + position: 'absolute', + // Ignore the scrollbar width + overflow: 'hidden', + height: 0, + top: 0, + left: 0, + // Create a new layer, increase the isolation of the computed values + transform: 'translateZ(0)', + }, +}; + +type TextareaStyles = { + outerHeightStyle: number; + overflowing: boolean; +}; + +function isEmpty(obj: TextareaStyles) { + return ( + obj === undefined || + obj === null || + Object.keys(obj).length === 0 || + (obj.outerHeightStyle === 0 && !obj.overflowing) + ); +} + +/** + * + * Demos: + * + * - [Textarea Autosize](https://next.mui.com/material-ui/react-textarea-autosize/) + * + * API: + * + * - [TextareaAutosize API](https://next.mui.com/material-ui/api/textarea-autosize/) + */ +const TextareaAutosize = React.forwardRef(function TextareaAutosize( + props: TextareaAutosizeProps, + forwardedRef: React.ForwardedRef, +) { + const { onChange, maxRows, minRows = 1, style, value, ...other } = props; + + const { current: isControlled } = React.useRef(value != null); + const inputRef = React.useRef(null); + const handleRef = useForkRef(forwardedRef, inputRef); + const heightRef = React.useRef(null); + const shadowRef = React.useRef(null); + + const calculateTextareaStyles = React.useCallback(() => { + const input = inputRef.current!; + + const containerWindow = ownerWindow(input); + const computedStyle = containerWindow.getComputedStyle(input); + + // If input's width is shrunk and it's not visible, don't sync height. + if (computedStyle.width === '0px') { + return { + outerHeightStyle: 0, + overflowing: false, + }; + } + + const inputShallow = shadowRef.current!; + + inputShallow.style.width = computedStyle.width; + inputShallow.value = input.value || props.placeholder || 'x'; + if (inputShallow.value.slice(-1) === '\n') { + // Certain fonts which overflow the line height will cause the textarea + // to report a different scrollHeight depending on whether the last line + // is empty. Make it non-empty to avoid this issue. + inputShallow.value += ' '; + } + + const boxSizing = computedStyle.boxSizing; + const padding = + getStyleValue(computedStyle.paddingBottom) + getStyleValue(computedStyle.paddingTop); + const border = + getStyleValue(computedStyle.borderBottomWidth) + getStyleValue(computedStyle.borderTopWidth); + + // The height of the inner content + const innerHeight = inputShallow.scrollHeight; + + // Measure height of a textarea with a single row + inputShallow.value = 'x'; + const singleRowHeight = inputShallow.scrollHeight; + + // The height of the outer content + let outerHeight = innerHeight; + + if (minRows) { + outerHeight = Math.max(Number(minRows) * singleRowHeight, outerHeight); + } + if (maxRows) { + outerHeight = Math.min(Number(maxRows) * singleRowHeight, outerHeight); + } + outerHeight = Math.max(outerHeight, singleRowHeight); + + // Take the box sizing into account for applying this value as a style. + const outerHeightStyle = outerHeight + (boxSizing === 'border-box' ? padding + border : 0); + const overflowing = Math.abs(outerHeight - innerHeight) <= 1; + + return { outerHeightStyle, overflowing }; + }, [maxRows, minRows, props.placeholder]); + + const syncHeight = React.useCallback(() => { + const textareaStyles = calculateTextareaStyles(); + + if (isEmpty(textareaStyles)) { + return; + } + + const outerHeightStyle = textareaStyles.outerHeightStyle; + const input = inputRef.current!; + if (heightRef.current !== outerHeightStyle) { + heightRef.current = outerHeightStyle; + input.style.height = `${outerHeightStyle}px`; + } + input.style.overflow = textareaStyles.overflowing ? 'hidden' : ''; + }, [calculateTextareaStyles]); + + useEnhancedEffect(() => { + const handleResize = () => { + syncHeight(); + }; + // Workaround a "ResizeObserver loop completed with undelivered notifications" error + // in test. + // Note that we might need to use this logic in production per https://github.com/WICG/resize-observer/issues/38 + // Also see https://github.com/mui/mui-x/issues/8733 + let rAF: any; + const rAFHandleResize = () => { + cancelAnimationFrame(rAF); + rAF = requestAnimationFrame(() => { + handleResize(); + }); + }; + const debounceHandleResize = debounce(handleResize); + const input = inputRef.current!; + const containerWindow = ownerWindow(input); + + containerWindow.addEventListener('resize', debounceHandleResize); + + let resizeObserver: ResizeObserver; + + if (typeof ResizeObserver !== 'undefined') { + resizeObserver = new ResizeObserver( + process.env.NODE_ENV === 'test' ? rAFHandleResize : handleResize, + ); + resizeObserver.observe(input); + } + + return () => { + debounceHandleResize.clear(); + cancelAnimationFrame(rAF); + containerWindow.removeEventListener('resize', debounceHandleResize); + if (resizeObserver) { + resizeObserver.disconnect(); + } + }; + }, [calculateTextareaStyles, syncHeight]); + + useEnhancedEffect(() => { + syncHeight(); + }); + + const handleChange = (event: React.ChangeEvent) => { + if (!isControlled) { + syncHeight(); + } + + if (onChange) { + onChange(event); + } + }; + + return ( + +