From 73a7cde82222f0ff1128b298eca771fa61e191db Mon Sep 17 00:00:00 2001 From: Siriwat K Date: Wed, 4 Oct 2023 16:32:16 +0700 Subject: [PATCH 01/46] [joy-ui] Introduce color inversion utilities (#38916) --- .../templates/framesx-web-blocks/theme.tsx | 15 +- .../ColorInversionAnyParent.js | 88 ++++ .../ColorInversionAnyParent.tsx | 80 ++++ .../ColorInversionNavigation.js | 8 +- .../ColorInversionNavigation.tsx | 8 +- .../color-inversion/ColorInversionOverview.js | 4 +- .../ColorInversionOverview.tsx | 4 +- .../color-inversion/ColorInversionSkip.js | 62 +++ .../color-inversion/ColorInversionSkip.tsx | 62 +++ .../color-inversion/ColorInversionSurface.js | 73 ++++ .../color-inversion/ColorInversionSurface.tsx | 72 ++++ .../ColorInversionSurface.tsx.preview | 2 + .../color-inversion/color-inversion.md | 116 ++++-- .../AccordionGroup/AccordionGroup.test.tsx | 8 +- .../src/AccordionGroup/AccordionGroup.tsx | 7 +- packages/mui-joy/src/Alert/Alert.test.tsx | 8 +- packages/mui-joy/src/Alert/Alert.tsx | 32 +- .../src/AspectRatio/AspectRatio.test.tsx | 11 +- .../mui-joy/src/AspectRatio/AspectRatio.tsx | 5 +- .../src/Autocomplete/Autocomplete.test.tsx | 7 - .../mui-joy/src/Autocomplete/Autocomplete.tsx | 37 +- .../AutocompleteListbox.test.tsx | 11 +- .../AutocompleteListbox.tsx | 7 +- .../AutocompleteOption.test.tsx | 8 +- .../AutocompleteOption/AutocompleteOption.tsx | 5 +- packages/mui-joy/src/Avatar/Avatar.test.tsx | 9 +- packages/mui-joy/src/Avatar/Avatar.tsx | 5 +- packages/mui-joy/src/Badge/Badge.test.tsx | 13 +- packages/mui-joy/src/Badge/Badge.tsx | 16 +- packages/mui-joy/src/Button/Button.test.tsx | 8 +- packages/mui-joy/src/Button/Button.tsx | 9 +- .../mui-joy/src/ButtonGroup/ButtonGroup.tsx | 24 +- packages/mui-joy/src/Card/Card.test.tsx | 8 +- packages/mui-joy/src/Card/Card.tsx | 27 +- .../src/CardOverflow/CardOverflow.test.tsx | 8 +- .../mui-joy/src/CardOverflow/CardOverflow.tsx | 5 +- .../mui-joy/src/Checkbox/Checkbox.test.tsx | 10 +- packages/mui-joy/src/Checkbox/Checkbox.tsx | 7 +- packages/mui-joy/src/Chip/Chip.test.tsx | 9 +- packages/mui-joy/src/Chip/Chip.tsx | 7 +- .../src/ChipDelete/ChipDelete.test.tsx | 10 +- .../mui-joy/src/ChipDelete/ChipDelete.tsx | 4 +- .../CircularProgress.test.tsx | 8 +- .../src/CircularProgress/CircularProgress.tsx | 5 +- .../mui-joy/src/DialogTitle/DialogTitle.tsx | 4 +- packages/mui-joy/src/Drawer/Drawer.tsx | 29 +- .../mui-joy/src/FormControl/FormControl.tsx | 14 +- .../src/IconButton/IconButton.test.tsx | 8 +- .../mui-joy/src/IconButton/IconButton.tsx | 4 +- packages/mui-joy/src/Input/Input.test.tsx | 3 - packages/mui-joy/src/Input/Input.tsx | 27 +- packages/mui-joy/src/Input/InputProps.ts | 4 + .../LinearProgress/LinearProgress.test.tsx | 8 +- .../src/LinearProgress/LinearProgress.tsx | 5 +- packages/mui-joy/src/Link/Link.test.tsx | 10 +- packages/mui-joy/src/Link/Link.tsx | 25 +- packages/mui-joy/src/List/List.test.tsx | 9 +- packages/mui-joy/src/List/List.tsx | 6 +- .../mui-joy/src/ListItem/ListItem.test.tsx | 9 +- packages/mui-joy/src/ListItem/ListItem.tsx | 6 +- .../ListItemButton/ListItemButton.test.tsx | 10 +- .../src/ListItemButton/ListItemButton.tsx | 7 +- .../src/ListSubheader/ListSubheader.test.tsx | 11 +- .../src/ListSubheader/ListSubheader.tsx | 19 +- .../src/ListSubheader/ListSubheaderProps.ts | 7 +- packages/mui-joy/src/Menu/Menu.test.tsx | 10 - packages/mui-joy/src/Menu/Menu.tsx | 54 +-- .../mui-joy/src/MenuButton/MenuButton.tsx | 11 +- .../mui-joy/src/MenuItem/MenuItem.test.tsx | 7 - packages/mui-joy/src/MenuItem/MenuItem.tsx | 5 +- .../mui-joy/src/MenuList/MenuList.test.tsx | 9 +- packages/mui-joy/src/MenuList/MenuList.tsx | 5 +- .../src/ModalClose/ModalClose.test.tsx | 9 +- .../mui-joy/src/ModalClose/ModalClose.tsx | 4 +- .../src/ModalDialog/ModalDialog.test.tsx | 8 +- .../mui-joy/src/ModalDialog/ModalDialog.tsx | 15 +- packages/mui-joy/src/Option/Option.tsx | 6 +- packages/mui-joy/src/Radio/Radio.test.tsx | 10 +- packages/mui-joy/src/Radio/Radio.tsx | 4 +- packages/mui-joy/src/Select/Select.test.tsx | 7 - packages/mui-joy/src/Select/Select.tsx | 90 ++-- packages/mui-joy/src/Select/SelectProps.ts | 4 + packages/mui-joy/src/Sheet/Sheet.test.tsx | 8 +- packages/mui-joy/src/Sheet/Sheet.tsx | 25 +- packages/mui-joy/src/Slider/Slider.test.tsx | 8 +- packages/mui-joy/src/Slider/Slider.tsx | 9 +- packages/mui-joy/src/SvgIcon/SvgIcon.tsx | 1 - packages/mui-joy/src/Switch/Switch.test.tsx | 3 - packages/mui-joy/src/Switch/Switch.tsx | 8 +- packages/mui-joy/src/Tab/Tab.test.tsx | 13 +- packages/mui-joy/src/Tab/Tab.tsx | 5 +- packages/mui-joy/src/TabList/TabList.test.tsx | 13 +- packages/mui-joy/src/TabList/TabList.tsx | 5 +- packages/mui-joy/src/TabPanel/TabPanel.tsx | 6 +- packages/mui-joy/src/Table/Table.test.tsx | 8 +- packages/mui-joy/src/Table/Table.tsx | 6 +- packages/mui-joy/src/Tabs/Tabs.test.tsx | 9 +- packages/mui-joy/src/Tabs/Tabs.tsx | 6 +- .../mui-joy/src/Textarea/Textarea.test.tsx | 3 - packages/mui-joy/src/Textarea/Textarea.tsx | 26 +- .../mui-joy/src/Textarea/TextareaProps.ts | 4 + packages/mui-joy/src/Tooltip/Tooltip.test.tsx | 18 +- packages/mui-joy/src/Tooltip/Tooltip.tsx | 38 +- .../src/Typography/Typography.test.tsx | 8 +- .../mui-joy/src/Typography/Typography.tsx | 13 +- .../colorInversionUtils.spec.tsx | 32 ++ .../colorInversionUtils.test.tsx | 21 + .../src/colorInversion/colorInversionUtils.ts | 386 ++++++++++++++++++ packages/mui-joy/src/colorInversion/index.ts | 1 + .../mui-joy/src/styles/ColorInversion.test.js | 125 ------ .../mui-joy/src/styles/ColorInversion.tsx | 54 --- .../src/styles/CssVarsProvider.test.tsx | 79 +--- .../mui-joy/src/styles/CssVarsProvider.tsx | 19 +- .../mui-joy/src/styles/defaultTheme.test.js | 2 - .../mui-joy/src/styles/extendTheme.spec.ts | 10 - .../mui-joy/src/styles/extendTheme.test.js | 2 - packages/mui-joy/src/styles/extendTheme.ts | 30 +- packages/mui-joy/src/styles/index.ts | 1 - .../src/styles/shouldSkipGeneratingVar.ts | 2 +- .../mui-joy/src/styles/types/colorSystem.ts | 6 +- packages/mui-joy/src/styles/types/theme.ts | 4 +- packages/mui-joy/src/styles/types/variants.ts | 48 +-- .../mui-joy/src/styles/variantUtils.test.js | 9 +- packages/mui-joy/src/styles/variantUtils.ts | 281 +------------ packages/mui-joy/src/utils/useSlot.test.tsx | 107 +---- packages/mui-joy/src/utils/useSlot.ts | 14 +- .../src/describeJoyColorInversion.tsx | 125 ------ packages/test-utils/src/index.ts | 1 - .../JoyColorInversionStyled.js | 75 ++++ 129 files changed, 1409 insertions(+), 1683 deletions(-) create mode 100644 docs/data/joy/main-features/color-inversion/ColorInversionAnyParent.js create mode 100644 docs/data/joy/main-features/color-inversion/ColorInversionAnyParent.tsx create mode 100644 docs/data/joy/main-features/color-inversion/ColorInversionSkip.js create mode 100644 docs/data/joy/main-features/color-inversion/ColorInversionSkip.tsx create mode 100644 docs/data/joy/main-features/color-inversion/ColorInversionSurface.js create mode 100644 docs/data/joy/main-features/color-inversion/ColorInversionSurface.tsx create mode 100644 docs/data/joy/main-features/color-inversion/ColorInversionSurface.tsx.preview create mode 100644 packages/mui-joy/src/colorInversion/colorInversionUtils.spec.tsx create mode 100644 packages/mui-joy/src/colorInversion/colorInversionUtils.test.tsx create mode 100644 packages/mui-joy/src/colorInversion/colorInversionUtils.ts create mode 100644 packages/mui-joy/src/colorInversion/index.ts delete mode 100644 packages/mui-joy/src/styles/ColorInversion.test.js delete mode 100644 packages/mui-joy/src/styles/ColorInversion.tsx delete mode 100644 packages/test-utils/src/describeJoyColorInversion.tsx create mode 100644 test/regressions/fixtures/JoyColorInversion/JoyColorInversionStyled.js diff --git a/docs/data/joy/getting-started/templates/framesx-web-blocks/theme.tsx b/docs/data/joy/getting-started/templates/framesx-web-blocks/theme.tsx index 6c1305214209e3..23200b104b3231 100644 --- a/docs/data/joy/getting-started/templates/framesx-web-blocks/theme.tsx +++ b/docs/data/joy/getting-started/templates/framesx-web-blocks/theme.tsx @@ -55,14 +55,13 @@ export default extendTheme({ JoyInput: { styleOverrides: { root: ({ ownerState, theme }) => ({ - ...(ownerState.variant === 'outlined' && - ownerState.color !== 'context' && { - [`&:not(.${inputClasses.focused}):hover::before`]: { - boxShadow: `inset 0 0 0 2px ${ - theme.vars.palette?.[ownerState.color!]?.outlinedBorder - }`, - }, - }), + ...(ownerState.variant === 'outlined' && { + [`&:not(.${inputClasses.focused}):hover::before`]: { + boxShadow: `inset 0 0 0 2px ${ + theme.vars.palette?.[ownerState.color!]?.outlinedBorder + }`, + }, + }), }), input: { caretColor: 'var(--Input-focusedHighlight)', diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionAnyParent.js b/docs/data/joy/main-features/color-inversion/ColorInversionAnyParent.js new file mode 100644 index 00000000000000..360d2c8d2c581a --- /dev/null +++ b/docs/data/joy/main-features/color-inversion/ColorInversionAnyParent.js @@ -0,0 +1,88 @@ +import * as React from 'react'; +import PropTypes from 'prop-types'; +import { applySolidInversion } from '@mui/joy/colorInversion'; +import Box from '@mui/joy/Box'; +import Button from '@mui/joy/Button'; +import Typography from '@mui/joy/Typography'; + +function Stat({ description, value, sx, ...props }) { + return ( + + + {value} + + + {description} + + + ); +} + +Stat.propTypes = { + description: PropTypes.node, + /** + * The system prop that allows defining system overrides as well as additional CSS styles. + */ + sx: PropTypes.oneOfType([ + PropTypes.arrayOf( + PropTypes.oneOfType([PropTypes.func, PropTypes.object, PropTypes.bool]), + ), + PropTypes.func, + PropTypes.object, + ]), + value: PropTypes.node, +}; + +export default function ColorInversionAnyParent() { + return ( + + `linear-gradient(45deg, ${theme.vars.palette.neutral[800]}, ${theme.vars.palette.neutral[600]})`, + borderRadius: 'sm', + }, + applySolidInversion('neutral'), + ]} + > +
+ + Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod + tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim + veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea + commodo consequat. + + +
+ + + + + + +
+ ); +} diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionAnyParent.tsx b/docs/data/joy/main-features/color-inversion/ColorInversionAnyParent.tsx new file mode 100644 index 00000000000000..7d07e8c37aa1e7 --- /dev/null +++ b/docs/data/joy/main-features/color-inversion/ColorInversionAnyParent.tsx @@ -0,0 +1,80 @@ +import * as React from 'react'; +import { applySolidInversion } from '@mui/joy/colorInversion'; +import Box, { BoxProps } from '@mui/joy/Box'; +import Button from '@mui/joy/Button'; +import Typography from '@mui/joy/Typography'; + +function Stat({ + description, + value, + sx, + ...props +}: { + description: React.ReactNode; + value: React.ReactNode; +} & BoxProps) { + return ( + + + {value} + + + {description} + + + ); +} + +export default function ColorInversionAnyParent() { + return ( + + `linear-gradient(45deg, ${theme.vars.palette.neutral[800]}, ${theme.vars.palette.neutral[600]})`, + borderRadius: 'sm', + }, + applySolidInversion('neutral'), + ]} + > +
+ + Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod + tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim + veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea + commodo consequat. + + +
+ + + + + + +
+ ); +} diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionNavigation.js b/docs/data/joy/main-features/color-inversion/ColorInversionNavigation.js index 135ee4f88b092a..57596bd9a5438d 100644 --- a/docs/data/joy/main-features/color-inversion/ColorInversionNavigation.js +++ b/docs/data/joy/main-features/color-inversion/ColorInversionNavigation.js @@ -86,7 +86,13 @@ export default function ColorInversionNavigation() { Chat - + 5 diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionNavigation.tsx b/docs/data/joy/main-features/color-inversion/ColorInversionNavigation.tsx index a12eb37f9881a6..9bbfc84c38a28d 100644 --- a/docs/data/joy/main-features/color-inversion/ColorInversionNavigation.tsx +++ b/docs/data/joy/main-features/color-inversion/ColorInversionNavigation.tsx @@ -86,7 +86,13 @@ export default function ColorInversionNavigation() { Chat - + 5 diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionOverview.js b/docs/data/joy/main-features/color-inversion/ColorInversionOverview.js index e13b479c70d6ed..bce6170d920826 100644 --- a/docs/data/joy/main-features/color-inversion/ColorInversionOverview.js +++ b/docs/data/joy/main-features/color-inversion/ColorInversionOverview.js @@ -29,9 +29,7 @@ export default function ColorInversionOverview() { > - - Learn how to build super fast websites. - + Learn how to build super fast websites. diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionOverview.tsx b/docs/data/joy/main-features/color-inversion/ColorInversionOverview.tsx index e13b479c70d6ed..bce6170d920826 100644 --- a/docs/data/joy/main-features/color-inversion/ColorInversionOverview.tsx +++ b/docs/data/joy/main-features/color-inversion/ColorInversionOverview.tsx @@ -29,9 +29,7 @@ export default function ColorInversionOverview() { > - - Learn how to build super fast websites. - + Learn how to build super fast websites. diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionSkip.js b/docs/data/joy/main-features/color-inversion/ColorInversionSkip.js new file mode 100644 index 00000000000000..cad9a69a1b9556 --- /dev/null +++ b/docs/data/joy/main-features/color-inversion/ColorInversionSkip.js @@ -0,0 +1,62 @@ +import * as React from 'react'; +import AspectRatio from '@mui/joy/AspectRatio'; +import Card from '@mui/joy/Card'; +import SvgIcon from '@mui/joy/SvgIcon'; +import IconButton from '@mui/joy/IconButton'; +import Typography from '@mui/joy/Typography'; +import ArrowForward from '@mui/icons-material/ArrowForward'; + +export default function ColorInversionSkip() { + return ( + + +
+ + + + + + + + +
+
+ + Design Thinking + + How to apply design thinking to your problem in order to generate innovative + and user-centric solutions. + + + + + +
+ ); +} diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionSkip.tsx b/docs/data/joy/main-features/color-inversion/ColorInversionSkip.tsx new file mode 100644 index 00000000000000..cad9a69a1b9556 --- /dev/null +++ b/docs/data/joy/main-features/color-inversion/ColorInversionSkip.tsx @@ -0,0 +1,62 @@ +import * as React from 'react'; +import AspectRatio from '@mui/joy/AspectRatio'; +import Card from '@mui/joy/Card'; +import SvgIcon from '@mui/joy/SvgIcon'; +import IconButton from '@mui/joy/IconButton'; +import Typography from '@mui/joy/Typography'; +import ArrowForward from '@mui/icons-material/ArrowForward'; + +export default function ColorInversionSkip() { + return ( + + +
+ + + + + + + + +
+
+ + Design Thinking + + How to apply design thinking to your problem in order to generate innovative + and user-centric solutions. + + + + + +
+ ); +} diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionSurface.js b/docs/data/joy/main-features/color-inversion/ColorInversionSurface.js new file mode 100644 index 00000000000000..b7e0b94a246de4 --- /dev/null +++ b/docs/data/joy/main-features/color-inversion/ColorInversionSurface.js @@ -0,0 +1,73 @@ +import * as React from 'react'; +import Box from '@mui/joy/Box'; +import Card from '@mui/joy/Card'; +import CardContent from '@mui/joy/CardContent'; +import Typography from '@mui/joy/Typography'; +import SvgIcon from '@mui/joy/SvgIcon'; + +export default function ColorInversionSurface() { + const creditCard = ( + + +
+ $4,236 + + CREDIT + +
+ + + + + + +
+ + •••• •••• •••• 1212 + + +
+ + CARD NAME + + + JOHN DOE + +
+
+ + EXPIRE + + + 07/25 + +
+
+
+ ); + + return ( + + {creditCard} + {React.cloneElement(creditCard, { variant: 'soft' })} + + ); +} diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionSurface.tsx b/docs/data/joy/main-features/color-inversion/ColorInversionSurface.tsx new file mode 100644 index 00000000000000..3f49510152aaf3 --- /dev/null +++ b/docs/data/joy/main-features/color-inversion/ColorInversionSurface.tsx @@ -0,0 +1,72 @@ +import * as React from 'react'; +import Box from '@mui/joy/Box'; +import Card from '@mui/joy/Card'; +import CardContent from '@mui/joy/CardContent'; +import Typography from '@mui/joy/Typography'; +import SvgIcon from '@mui/joy/SvgIcon'; + +export default function ColorInversionSurface() { + const creditCard = ( + + +
+ $4,236 + + CREDIT + +
+ + + + + + +
+ + •••• •••• •••• 1212 + + +
+ + CARD NAME + + + JOHN DOE + +
+
+ + EXPIRE + + + 07/25 + +
+
+
+ ); + return ( + + {creditCard} + {React.cloneElement(creditCard, { variant: 'soft' })} + + ); +} diff --git a/docs/data/joy/main-features/color-inversion/ColorInversionSurface.tsx.preview b/docs/data/joy/main-features/color-inversion/ColorInversionSurface.tsx.preview new file mode 100644 index 00000000000000..f91cb4f688a1e8 --- /dev/null +++ b/docs/data/joy/main-features/color-inversion/ColorInversionSurface.tsx.preview @@ -0,0 +1,2 @@ +{creditCard} +{React.cloneElement(creditCard, { variant: 'soft' })} \ No newline at end of file diff --git a/docs/data/joy/main-features/color-inversion/color-inversion.md b/docs/data/joy/main-features/color-inversion/color-inversion.md index 7e6bd76d9bb694..835b020e3422a1 100644 --- a/docs/data/joy/main-features/color-inversion/color-inversion.md +++ b/docs/data/joy/main-features/color-inversion/color-inversion.md @@ -21,21 +21,13 @@ The color inversion is implemented to solve this issue, keeping the global varia ## Overview -When color inversion is enabled on the parent component, the children with implicit color will invert their styles to match the parent's background by keeping the same hierarchy of importance based on their variants. +When color inversion is enabled on the parent component, all of Joy UI children will invert their styles regardless of its color prop to match the parent's background by keeping the same hierarchy of importance based on their variants. {{"demo": "ColorInversionOverview.js"}} -**Implicit** color refers to components that don't have the `color` prop specified. - -Color inversion has no effect on children that have an **explicit** `color` prop. - -```js -// implicit color. The styles change when color inversion is enabled. - - -// explicit color. Color inversion has no effect. - -``` +:::info +The color inversion feature is only available for `soft` and `solid` variants because the rest of the global variants don't have background by default. +::: ### Benefits @@ -45,7 +37,7 @@ Color inversion has no effect on children that have an **explicit** `color` prop - It works for both light and dark mode. - It can be disabled at any time without impacting the structure of the components. - It is an opt-in feature. If you don't use it, the extra CSS variables won't be included in the production style sheet. -- It doesn't alter the styles of the children if you explicitly specify the `color` prop on them. +- Some children can be excluded from the color inversion, see ["skip color inversion on a child"](#skip-color-inversion-on-a-child) section. ### Trade-offs @@ -54,13 +46,11 @@ Color inversion has no effect on children that have an **explicit** `color` prop ## Usage -To enable color inversion, use the `invertedColors` prop. Note that this prop only works when the components have the `solid` or `soft` global variant applied. +### Surface components -```js - +Surface components, including the Alert, Card, Drawer, ModalDialog, Menu, and Sheet, have the `invertedColors` prop to enable color inversion for `solid` and `soft` variants. - -``` +{{"demo": "ColorInversionSurface.js"}} ### Portal popup @@ -73,18 +63,71 @@ The popup slot of the Select component has `disablePortal` set to true by defaul {{"demo": "ColorInversionPopup.js"}} -## How it works +### Skip color inversion on a child + +To skip color inversion on a specific child, set `data-skip-inverted-colors` attribute to the component. + +The component with `data-skip-inverted-colors` and its children will be excluded from the color inversion. + +{{"demo": "ColorInversionSkip.js"}} -### Parent component +### Apply color inversion to any parent -When `invertedColors` is set to true on the surface component, a set of CSS variables is applied to it. -The values of those variables come from `theme.colorInversion[variant][color]`, where `variant` and `color` are the component's props. -The surface component also creates a React context to tell the children to update their styles. +Use `applySolidInversion` or `applySoftInversion` utilities to apply color inversion to any parent component. + +They are used internally by the surface components, e.g. [Card](/joy-ui/react-card/#inverted-colors), when `invertedColors` prop is set to true. + +```js +import { applySolidInversion, applySoftInversion } from '@mui/joy/colorInversion'; +``` + +Example usage for `sx` prop and `styled` API: + +- `sx` prop + + ```js + import { applySolidInversion } from '@mui/joy/colorInversion'; + + ({ + display: 'flex', + alignItems: 'center', + background: theme.vars.palette.neutral[900], + }), + applySolidInversion('neutral'), + ]} + > + … + ; + ``` + +- `styled` API + + ```js + import { styled } from '@mui/joy/styles'; + import { applySoftInversion } from '@mui/joy/colorInversion'; + + const StyledBox = styled(Box)( + ({ theme }) => ({ + display: 'flex', + alignItems: 'center', + ...theme.variants.soft.primary, + }), + applySoftInversion('primary'), + ); + ``` + +{{"demo": "ColorInversionAnyParent.js"}} + +## How it works + +When `invertedColors` is set to true or the utility is used on the parent component, a set of CSS variables is applied to it. There is no [React context](https://react.dev/learn/passing-data-deeply-with-context) involved in this feature. ```jsx -// The component style sheet +// The parent's style sheet { // the values of these variables depends on the parent's variant and color. --variant-softColor: …; @@ -93,29 +136,26 @@ The surface component also creates a React context to tell the children to updat --variant-softHoverBg: …; --variant-softActiveBg: …; … // other variants + --joy-palette-text-primary: …; + --joy-palette-text-secondary: …; + --joy-palette-text-tertiary: …; + --joy-palette-background-surface: …; + … // other theme palette tokens } ``` -### Child component - -All Joy UI components that support global variants check the React context that contains the color inversion flag. -If the flag is true and the child has an implicit color, the internal `color` value will switch to `context` and apply the styles from `theme.variants[variant].context`. - -The styles will match the `--variant-*` variables that the parent has. +As a result, the children will use those CSS variables instead of the theme. ```jsx - - -// Component style sheet +// The children style sheet +// The values of these variables are inherited from the parent. { - background-color: var(--variant-softBg); - color: var(--variant-softColor); + color: var(--joy-palette-text-primary); + background: var(--joy-palette-background-surface); + … } ``` -In summary, the parent creates a React context to tell the children that the feature is enabled, and generates CSS variables that will be used by the children. -The children with an implicit color switch their default color value to `context` to get the styles from the theme. - ## Common examples ### Header diff --git a/packages/mui-joy/src/AccordionGroup/AccordionGroup.test.tsx b/packages/mui-joy/src/AccordionGroup/AccordionGroup.test.tsx index 50dbfbd9de058d..4eec48f22eb501 100644 --- a/packages/mui-joy/src/AccordionGroup/AccordionGroup.test.tsx +++ b/packages/mui-joy/src/AccordionGroup/AccordionGroup.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import AccordionGroup, { accordionGroupClasses as classes } from '@mui/joy/AccordionGroup'; @@ -77,8 +73,6 @@ describe('', () => { expect(getByTestId('root')).to.have.class(colorConfig.class); }); }); - - describeJoyColorInversion(, { muiName: 'JoyAccordionGroup', classes }); }); it('should not warn when using custom color, variant, size', () => { diff --git a/packages/mui-joy/src/AccordionGroup/AccordionGroup.tsx b/packages/mui-joy/src/AccordionGroup/AccordionGroup.tsx index 73df8a92bf487f..8ec72d53de2cc9 100644 --- a/packages/mui-joy/src/AccordionGroup/AccordionGroup.tsx +++ b/packages/mui-joy/src/AccordionGroup/AccordionGroup.tsx @@ -4,7 +4,7 @@ import PropTypes from 'prop-types'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { unstable_composeClasses as composeClasses } from '@mui/base'; import { OverridableComponent } from '@mui/types'; -import { useColorInversion, useThemeProps } from '../styles'; +import { useThemeProps } from '../styles'; import styled from '../styles/styled'; import { getAccordionGroupUtilityClass } from './accordionGroupClasses'; import { @@ -84,7 +84,7 @@ const AccordionGroup = React.forwardRef(function AccordionGroup(inProps, ref) { const { component = 'div', - color: colorProp = 'neutral', + color = 'neutral', children, disableDivider = false, variant = 'plain', @@ -97,9 +97,6 @@ const AccordionGroup = React.forwardRef(function AccordionGroup(inProps, ref) { const externalForwardedProps = { ...other, component, slots, slotProps }; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); - const ownerState = { ...props, component, diff --git a/packages/mui-joy/src/Alert/Alert.test.tsx b/packages/mui-joy/src/Alert/Alert.test.tsx index 58003f6ebe9363..c0249786597255 100644 --- a/packages/mui-joy/src/Alert/Alert.test.tsx +++ b/packages/mui-joy/src/Alert/Alert.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import Alert, { AlertClassKey, alertClasses as classes } from '@mui/joy/Alert'; @@ -30,8 +26,6 @@ describe('', () => { skip: ['classesRoot', 'componentsProp'], })); - describeJoyColorInversion(, { muiName: 'JoyAlert', classes }); - describe('prop: variant', () => { it('soft by default', () => { const { getByRole } = render(); diff --git a/packages/mui-joy/src/Alert/Alert.tsx b/packages/mui-joy/src/Alert/Alert.tsx index ae19e140577ce0..b9aef4756b24b5 100644 --- a/packages/mui-joy/src/Alert/Alert.tsx +++ b/packages/mui-joy/src/Alert/Alert.tsx @@ -5,9 +5,9 @@ import clsx from 'clsx'; import { unstable_composeClasses as composeClasses } from '@mui/base'; import { OverridableComponent } from '@mui/types'; import { unstable_capitalize as capitalize } from '@mui/utils'; +import { applySolidInversion, applySoftInversion } from '../colorInversion'; import styled from '../styles/styled'; import useThemeProps from '../styles/useThemeProps'; -import { ColorInversionProvider, useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import { getAlertUtilityClass } from './alertClasses'; import { AlertProps, AlertOwnerState, AlertTypeMap } from './AlertProps'; @@ -76,11 +76,16 @@ const AlertRoot = styled('div', { borderRadius: 'var(--Alert-radius)', ...theme.typography[`body-${({ sm: 'xs', md: 'sm', lg: 'md' } as const)[ownerState.size!]}`], fontWeight: theme.vars.fontWeight.md, + ...(ownerState.variant === 'solid' && + ownerState.color && + ownerState.invertedColors && + applySolidInversion(ownerState.color)(theme)), + ...(ownerState.variant === 'soft' && + ownerState.color && + ownerState.invertedColors && + applySoftInversion(ownerState.color)(theme)), ...theme.variants[ownerState.variant!]?.[ownerState.color!], } as const, - ownerState.color !== 'context' && - ownerState.invertedColors && - theme.colorInversion[ownerState.variant!]?.[ownerState.color!], p !== undefined && { '--Alert-padding': p }, padding !== undefined && { '--Alert-padding': padding }, borderRadius !== undefined && { '--Alert-radius': borderRadius }, @@ -124,7 +129,7 @@ const Alert = React.forwardRef(function Alert(inProps, ref) { const { children, className, - color: colorProp = 'neutral', + color = 'neutral', invertedColors = false, role = 'alert', variant = 'soft', @@ -136,12 +141,11 @@ const Alert = React.forwardRef(function Alert(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, color, + invertedColors, variant, size, }; @@ -174,24 +178,14 @@ const Alert = React.forwardRef(function Alert(inProps, ref) { ownerState, }); - const result = ( - + return ( + {startDecorator && ( {startDecorator} )} {children} {endDecorator && {endDecorator}} - - ); - - return ( - - {invertedColors ? ( - {result} - ) : ( - result - )} ); }) as OverridableComponent; diff --git a/packages/mui-joy/src/AspectRatio/AspectRatio.test.tsx b/packages/mui-joy/src/AspectRatio/AspectRatio.test.tsx index 855482a1a59530..5798d0dc8d6c30 100644 --- a/packages/mui-joy/src/AspectRatio/AspectRatio.test.tsx +++ b/packages/mui-joy/src/AspectRatio/AspectRatio.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import AspectRatio, { @@ -32,11 +28,6 @@ describe('', () => { skip: ['classesRoot', 'componentsProp'], })); - describeJoyColorInversion( - , - { muiName: 'JoyAlert', classes }, - ); - describe('prop: variant', () => { it('plain by default', () => { const { getByTestId } = render(Hello World); diff --git a/packages/mui-joy/src/AspectRatio/AspectRatio.tsx b/packages/mui-joy/src/AspectRatio/AspectRatio.tsx index 7d0323b1529abc..75d4d0e9fe9345 100644 --- a/packages/mui-joy/src/AspectRatio/AspectRatio.tsx +++ b/packages/mui-joy/src/AspectRatio/AspectRatio.tsx @@ -7,7 +7,6 @@ import { unstable_capitalize as capitalize } from '@mui/utils'; import useThemeProps from '../styles/useThemeProps'; import useSlot from '../utils/useSlot'; import styled from '../styles/styled'; -import { useColorInversion } from '../styles/ColorInversion'; import { getAspectRatioUtilityClass } from './aspectRatioClasses'; import { AspectRatioProps, AspectRatioOwnerState, AspectRatioTypeMap } from './AspectRatioProps'; @@ -111,7 +110,7 @@ const AspectRatio = React.forwardRef(function AspectRatio(inProps, ref) { minHeight, maxHeight, objectFit = 'cover', - color: colorProp = 'neutral', + color = 'neutral', variant = 'soft', component, flex = false, @@ -119,8 +118,6 @@ const AspectRatio = React.forwardRef(function AspectRatio(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, diff --git a/packages/mui-joy/src/Autocomplete/Autocomplete.test.tsx b/packages/mui-joy/src/Autocomplete/Autocomplete.test.tsx index 4fbe3908ea0834..8701cac0d7c202 100644 --- a/packages/mui-joy/src/Autocomplete/Autocomplete.test.tsx +++ b/packages/mui-joy/src/Autocomplete/Autocomplete.test.tsx @@ -9,7 +9,6 @@ import { act, fireEvent, strictModeDoubleLoggingSuppressed, - describeJoyColorInversion, } from '@mui-internal/test-utils'; import Autocomplete, { autocompleteClasses as classes, @@ -78,12 +77,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { - muiName: 'JoyAutocomplete', - classes, - portalSlot: 'listbox', - }); - it('should be customizable in the theme', () => { render( , 'onChange' | 'defaultValue'>; @@ -346,12 +345,10 @@ const Autocomplete = React.forwardRef(function Autocomplete( } = props; const other = excludeUseAutocompleteParams(otherProps); - const { getColor } = useColorInversion(variant); const formControl = React.useContext(FormControlContext); const error = inProps.error ?? formControl?.error ?? errorProp; const size = inProps.size ?? formControl?.size ?? sizeProp; - const rootColor = inProps.color ?? (error ? 'danger' : formControl?.color ?? colorProp); - const color = getColor(inProps.color, rootColor); + const color = inProps.color ?? (error ? 'danger' : formControl?.color ?? colorProp); const disabled = disabledProp ?? formControl?.disabled ?? false; const { @@ -386,6 +383,7 @@ const Autocomplete = React.forwardRef(function Autocomplete( // If you modify this, make sure to keep the `AutocompleteOwnerState` type in sync. const ownerState = { + instanceColor: inProps.color, ...props, value, disabled, @@ -426,7 +424,7 @@ const Autocomplete = React.forwardRef(function Autocomplete( key={index} size={size} variant="soft" - color={color === 'context' ? undefined : 'neutral'} + color="neutral" endDecorator={} > {getOptionLabel(option)} @@ -535,9 +533,8 @@ const Autocomplete = React.forwardRef(function Autocomplete( ownerState, getSlotOwnerState: (mergedProps) => ({ size: mergedProps.size || size, - variant: - mergedProps.variant || getChildVariantAndColor(variant, rootColor).variant || 'plain', - color: mergedProps.color || getChildVariantAndColor(variant, rootColor).color || 'neutral', + variant: mergedProps.variant || getChildVariantAndColor(variant, color).variant || 'plain', + color: mergedProps.color || getChildVariantAndColor(variant, color).color || 'neutral', disableColorInversion: !!inProps.color, }), additionalProps: { @@ -554,9 +551,8 @@ const Autocomplete = React.forwardRef(function Autocomplete( ownerState, getSlotOwnerState: (mergedProps) => ({ size: mergedProps.size || size, - variant: - mergedProps.variant || getChildVariantAndColor(variant, rootColor).variant || 'plain', - color: mergedProps.color || getChildVariantAndColor(variant, rootColor).color || 'neutral', + variant: mergedProps.variant || getChildVariantAndColor(variant, color).variant || 'plain', + color: mergedProps.color || getChildVariantAndColor(variant, color).color || 'neutral', disableColorInversion: !!inProps.color, }), additionalProps: { @@ -576,7 +572,7 @@ const Autocomplete = React.forwardRef(function Autocomplete( getSlotOwnerState: (mergedProps) => ({ size: mergedProps.size || size, variant: mergedProps.variant || variant, - color: mergedProps.color || (!mergedProps.disablePortal ? rootColor : color), + color: mergedProps.color || color, disableColorInversion: !mergedProps.disablePortal, }), additionalProps: { @@ -636,9 +632,8 @@ const Autocomplete = React.forwardRef(function Autocomplete( externalForwardedProps, ownerState, getSlotOwnerState: (mergedProps) => ({ - variant: - mergedProps.variant || getChildVariantAndColor(variant, rootColor).variant || 'plain', - color: mergedProps.color || getChildVariantAndColor(variant, rootColor).color || 'neutral', + variant: mergedProps.variant || getChildVariantAndColor(variant, color).variant || 'plain', + color: mergedProps.color || getChildVariantAndColor(variant, color).color || 'neutral', disableColorInversion: !listboxProps.disablePortal, }), additionalProps: { @@ -680,14 +675,11 @@ const Autocomplete = React.forwardRef(function Autocomplete( let popup = null; if (anchorEl) { popup = ( - + ); - - if (!listboxProps.disablePortal) { - // For portal popup, the children should not inherit color inversion from the upper parent. - popup = {popup}; - } } return ( diff --git a/packages/mui-joy/src/AutocompleteListbox/AutocompleteListbox.test.tsx b/packages/mui-joy/src/AutocompleteListbox/AutocompleteListbox.test.tsx index d2aa9408bbc062..2cc52aedc4263a 100644 --- a/packages/mui-joy/src/AutocompleteListbox/AutocompleteListbox.test.tsx +++ b/packages/mui-joy/src/AutocompleteListbox/AutocompleteListbox.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import AutocompleteListbox, { autocompleteListboxClasses as classes, @@ -30,11 +26,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { - muiName: 'JoyAutocompleteListbox', - classes, - }); - it('should have ul tag', () => { const { container } = render(); expect(container.firstChild).to.have.tagName('ul'); diff --git a/packages/mui-joy/src/AutocompleteListbox/AutocompleteListbox.tsx b/packages/mui-joy/src/AutocompleteListbox/AutocompleteListbox.tsx index 4170a446629a1a..ca5381f029b203 100644 --- a/packages/mui-joy/src/AutocompleteListbox/AutocompleteListbox.tsx +++ b/packages/mui-joy/src/AutocompleteListbox/AutocompleteListbox.tsx @@ -16,7 +16,6 @@ import { import listItemClasses from '../ListItem/listItemClasses'; import listClasses from '../List/listClasses'; import { scopedVariables } from '../List/ListProvider'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; const useUtilityClasses = (ownerState: AutocompleteListboxOwnerState) => { @@ -107,15 +106,13 @@ const AutocompleteListbox = React.forwardRef(function AutocompleteListbox(inProp children, className, component, - color: colorProp = 'neutral', + color = 'neutral', variant = 'outlined', size = 'md', slots = {}, slotProps = {}, ...otherProps } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, @@ -144,7 +141,7 @@ const AutocompleteListbox = React.forwardRef(function AutocompleteListbox(inProp }); return ( - + {children} ); diff --git a/packages/mui-joy/src/AutocompleteOption/AutocompleteOption.test.tsx b/packages/mui-joy/src/AutocompleteOption/AutocompleteOption.test.tsx index 2e3cbd1519b956..590d7e2cbb3b6d 100644 --- a/packages/mui-joy/src/AutocompleteOption/AutocompleteOption.test.tsx +++ b/packages/mui-joy/src/AutocompleteOption/AutocompleteOption.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import AutocompleteOption, { autocompleteOptionClasses as classes, @@ -30,8 +26,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyAutocompleteOption', classes }); - it('should have li tag', () => { const { getByRole } = render(); expect(getByRole('option')).to.have.tagName('li'); diff --git a/packages/mui-joy/src/AutocompleteOption/AutocompleteOption.tsx b/packages/mui-joy/src/AutocompleteOption/AutocompleteOption.tsx index 85e4ceba027655..1d2f6562644a49 100644 --- a/packages/mui-joy/src/AutocompleteOption/AutocompleteOption.tsx +++ b/packages/mui-joy/src/AutocompleteOption/AutocompleteOption.tsx @@ -10,7 +10,6 @@ import { styled, useThemeProps } from '../styles'; import { useVariantColor } from '../styles/variantColorInheritance'; import { getAutocompleteOptionUtilityClass } from './autocompleteOptionClasses'; import { AutocompleteOptionOwnerState, AutocompleteOptionTypeMap } from './AutocompleteOptionProps'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; const useUtilityClasses = (ownerState: AutocompleteOptionOwnerState) => { @@ -68,12 +67,10 @@ const AutocompleteOption = React.forwardRef(function AutocompleteOption(inProps, slotProps = {}, ...other } = props; - const { variant = variantProp, color: inheritedColor = colorProp } = useVariantColor( + const { variant = variantProp, color = colorProp } = useVariantColor( inProps.variant, inProps.color, ); - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, inheritedColor); const ownerState = { ...props, diff --git a/packages/mui-joy/src/Avatar/Avatar.test.tsx b/packages/mui-joy/src/Avatar/Avatar.test.tsx index c1d9bf1be2aabf..48ca2a24a8f36a 100644 --- a/packages/mui-joy/src/Avatar/Avatar.test.tsx +++ b/packages/mui-joy/src/Avatar/Avatar.test.tsx @@ -1,12 +1,7 @@ import * as React from 'react'; import { expect } from 'chai'; import { spy } from 'sinon'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, - fireEvent, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance, fireEvent } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import Avatar, { AvatarClassKey, avatarClasses as classes } from '@mui/joy/Avatar'; @@ -33,8 +28,6 @@ describe('', () => { skip: ['classesRoot', 'componentsProp'], })); - describeJoyColorInversion(, { muiName: 'JoyAvatar', classes }); - describe('prop: variant', () => { it('soft by default', () => { const { getByTestId } = render(); diff --git a/packages/mui-joy/src/Avatar/Avatar.tsx b/packages/mui-joy/src/Avatar/Avatar.tsx index fd2eb21fa1e240..aa8f54fe1c9667 100644 --- a/packages/mui-joy/src/Avatar/Avatar.tsx +++ b/packages/mui-joy/src/Avatar/Avatar.tsx @@ -7,7 +7,6 @@ import { unstable_capitalize as capitalize } from '@mui/utils'; import { useThemeProps } from '../styles'; import useSlot from '../utils/useSlot'; import styled from '../styles/styled'; -import { useColorInversion } from '../styles/ColorInversion'; import Person from '../internal/svg-icons/Person'; import { getAvatarUtilityClass } from './avatarClasses'; import { AvatarProps, AvatarOwnerState, AvatarTypeMap } from './AvatarProps'; @@ -169,9 +168,7 @@ const Avatar = React.forwardRef(function Avatar(inProps, ref) { ...other } = props; const variant = inProps.variant || groupContext?.variant || variantProp; - const { getColor } = useColorInversion(variant); - const colorFromContext = inProps.color || groupContext?.color; - const color = colorFromContext !== 'context' ? getColor(colorFromContext, colorProp) : colorProp; + const color = inProps.color || groupContext?.color || colorProp; const size = inProps.size || groupContext?.size || sizeProp; let children = null; diff --git a/packages/mui-joy/src/Badge/Badge.test.tsx b/packages/mui-joy/src/Badge/Badge.test.tsx index 672b81c87d3ae2..51d2693b54cb22 100644 --- a/packages/mui-joy/src/Badge/Badge.test.tsx +++ b/packages/mui-joy/src/Badge/Badge.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import Badge, { BadgeClassKey, BadgeOrigin, badgeClasses as classes } from '@mui/joy/Badge'; @@ -46,13 +42,6 @@ describe('', () => { }), ); - describeJoyColorInversion( - - , { muiName: 'JoyButton', classes }); - it('by default, should render with the root, variantSolid, sizeMd and colorPrimary classes', () => { const { getByRole } = render(); const button = getByRole('button'); diff --git a/packages/mui-joy/src/Button/Button.tsx b/packages/mui-joy/src/Button/Button.tsx index b1077f1186525b..2735eb215ac9e4 100644 --- a/packages/mui-joy/src/Button/Button.tsx +++ b/packages/mui-joy/src/Button/Button.tsx @@ -6,7 +6,6 @@ import { unstable_composeClasses as composeClasses } from '@mui/base/composeClas import { Interpolation } from '@mui/system'; import { unstable_capitalize as capitalize, unstable_useForkRef as useForkRef } from '@mui/utils'; import { styled, Theme, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import CircularProgress from '../CircularProgress'; import buttonClasses, { getButtonUtilityClass } from './buttonClasses'; @@ -214,8 +213,7 @@ const Button = React.forwardRef(function Button(inProps, ref) { const variant = inProps.variant || buttonGroup.variant || variantProp; const size = inProps.size || buttonGroup.size || sizeProp; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, buttonGroup.color || colorProp); + const color = inProps.color || buttonGroup.color || colorProp; const disabled = (inProps.disabled || inProps.loading) ?? (buttonGroup.disabled || disabledProp || loading); @@ -229,10 +227,7 @@ const Button = React.forwardRef(function Button(inProps, ref) { }); const loadingIndicator = loadingIndicatorProp ?? ( - + ); React.useImperativeHandle( diff --git a/packages/mui-joy/src/ButtonGroup/ButtonGroup.tsx b/packages/mui-joy/src/ButtonGroup/ButtonGroup.tsx index 110e772e5089ed..8dfda1b3ae7c8f 100644 --- a/packages/mui-joy/src/ButtonGroup/ButtonGroup.tsx +++ b/packages/mui-joy/src/ButtonGroup/ButtonGroup.tsx @@ -64,14 +64,15 @@ export const StyledButtonGroup = styled('div')<{ ownerState: ButtonGroupOwnerSta } }, ); + const outlinedStyle = theme.variants.outlined?.[ownerState.color!]; + const outlinedDisabledStyle = theme.variants.outlinedDisabled?.[ownerState.color!]; + const outlinedHoverStyle = theme.variants.outlinedHover?.[ownerState.color!]; return [ { '--ButtonGroup-separatorSize': ownerState.variant === 'outlined' ? '1px' : 'calc(var(--ButtonGroup-connected) * 1px)', - ...(ownerState.color !== 'context' && { - '--ButtonGroup-separatorColor': theme.vars.palette[ownerState.color!]?.outlinedBorder, - }), + '--ButtonGroup-separatorColor': outlinedStyle?.borderColor, '--ButtonGroup-radius': theme.vars.radius.sm, '--Divider-inset': '0.5rem', '--unstable_childRadius': @@ -131,19 +132,14 @@ export const StyledButtonGroup = styled('div')<{ ownerState: ButtonGroupOwnerSta '&:not(:disabled)': { zIndex: 1, // to make borders appear above disabled buttons. }, - ...(ownerState.color !== 'context' && { - '&:disabled': { - '--ButtonGroup-separatorColor': - theme.vars.palette[ownerState.color!]?.outlinedDisabledBorder, + '&:disabled': { + '--ButtonGroup-separatorColor': outlinedDisabledStyle?.borderColor, + }, + ...(ownerState.variant === 'outlined' && { + '&:hover': { + '--ButtonGroup-separatorColor': outlinedHoverStyle?.borderColor, }, }), - ...(ownerState.variant === 'outlined' && - ownerState.color !== 'context' && { - '&:hover': { - '--ButtonGroup-separatorColor': - theme.vars.palette[ownerState.color!]?.outlinedHoverBorder, - }, - }), [`&:hover, ${theme.focus.selector}`]: { zIndex: 2, // to make borders appear above sibling. }, diff --git a/packages/mui-joy/src/Card/Card.test.tsx b/packages/mui-joy/src/Card/Card.test.tsx index 88272f6688a1c5..a95e7f9a6bf134 100644 --- a/packages/mui-joy/src/Card/Card.test.tsx +++ b/packages/mui-joy/src/Card/Card.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import Card, { cardClasses as classes, CardClassKey } from '@mui/joy/Card'; @@ -30,8 +26,6 @@ describe('', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyCard', classes }); - describe('prop: variant', () => { it('outlined by default', () => { const { getByTestId } = render(Hello World); diff --git a/packages/mui-joy/src/Card/Card.tsx b/packages/mui-joy/src/Card/Card.tsx index 48d3d77fbf904b..c2cf9288631133 100644 --- a/packages/mui-joy/src/Card/Card.tsx +++ b/packages/mui-joy/src/Card/Card.tsx @@ -9,8 +9,8 @@ import { unstable_isMuiElement as isMuiElement, } from '@mui/utils'; import { useThemeProps } from '../styles'; +import { applySolidInversion, applySoftInversion } from '../colorInversion'; import styled from '../styles/styled'; -import { ColorInversionProvider, useColorInversion } from '../styles/ColorInversion'; import { getCardUtilityClass } from './cardClasses'; import { CardProps, CardOwnerState, CardTypeMap } from './CardProps'; import { resolveSxValue } from '../styles/styleUtils'; @@ -83,11 +83,16 @@ export const StyledCardRoot = styled('div')<{ ownerState: CardOwnerState }>( display: 'flex', flexDirection: ownerState.orientation === 'horizontal' ? 'row' : 'column', ...theme.typography[`body-${ownerState.size!}`], + ...(ownerState.variant === 'solid' && + ownerState.color && + ownerState.invertedColors && + applySolidInversion(ownerState.color)(theme)), + ...(ownerState.variant === 'soft' && + ownerState.color && + ownerState.invertedColors && + applySoftInversion(ownerState.color)(theme)), ...theme.variants[ownerState.variant!]?.[ownerState.color!], } as const, - ownerState.color !== 'context' && - ownerState.invertedColors && - theme.colorInversion[ownerState.variant!]?.[ownerState.color!], p !== undefined && { '--Card-padding': p }, padding !== undefined && { '--Card-padding': padding }, borderRadius !== undefined && { '--Card-radius': borderRadius }, @@ -99,7 +104,7 @@ const CardRoot = styled(StyledCardRoot, { name: 'JoyCard', slot: 'Root', overridesResolver: (props, styles) => styles.root, -})({}); +})<{ ownerState: CardOwnerState }>({}); /** * @@ -119,7 +124,7 @@ const Card = React.forwardRef(function Card(inProps, ref) { const { className, - color: colorProp = 'neutral', + color = 'neutral', component = 'div', invertedColors = false, size = 'md', @@ -130,8 +135,6 @@ const Card = React.forwardRef(function Card(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, @@ -140,6 +143,7 @@ const Card = React.forwardRef(function Card(inProps, ref) { orientation, size, variant, + invertedColors, }; const classes = useUtilityClasses(ownerState); @@ -153,7 +157,7 @@ const Card = React.forwardRef(function Card(inProps, ref) { ownerState, }); - const result = ( + return ( {React.Children.map(children, (child, index) => { if (!React.isValidElement(child)) { @@ -185,11 +189,6 @@ const Card = React.forwardRef(function Card(inProps, ref) { })} ); - - if (invertedColors) { - return {result}; - } - return result; }) as OverridableComponent; Card.propTypes /* remove-proptypes */ = { diff --git a/packages/mui-joy/src/CardOverflow/CardOverflow.test.tsx b/packages/mui-joy/src/CardOverflow/CardOverflow.test.tsx index 33619311553c68..ca34945d506f57 100644 --- a/packages/mui-joy/src/CardOverflow/CardOverflow.test.tsx +++ b/packages/mui-joy/src/CardOverflow/CardOverflow.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import CardOverflow, { @@ -33,8 +29,6 @@ describe('', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyCardOverflow', classes }); - describe('prop: variant', () => { it('plain by default', () => { const { getByTestId } = render(Hello World); diff --git a/packages/mui-joy/src/CardOverflow/CardOverflow.tsx b/packages/mui-joy/src/CardOverflow/CardOverflow.tsx index 17ed565de6d0dd..0303c760586743 100644 --- a/packages/mui-joy/src/CardOverflow/CardOverflow.tsx +++ b/packages/mui-joy/src/CardOverflow/CardOverflow.tsx @@ -7,7 +7,6 @@ import { OverridableComponent } from '@mui/types'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { useThemeProps } from '../styles'; import styled from '../styles/styled'; -import { useColorInversion } from '../styles/ColorInversion'; import { getCardOverflowUtilityClass } from './cardOverflowClasses'; import { CardOverflowProps, @@ -128,14 +127,12 @@ const CardOverflow = React.forwardRef(function CardOverflow(inProps, ref) { className, component = 'div', children, - color: colorProp = 'neutral', + color = 'neutral', variant = 'plain', slots = {}, slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, diff --git a/packages/mui-joy/src/Checkbox/Checkbox.test.tsx b/packages/mui-joy/src/Checkbox/Checkbox.test.tsx index a52ee390e075a6..5e615272322fd7 100644 --- a/packages/mui-joy/src/Checkbox/Checkbox.test.tsx +++ b/packages/mui-joy/src/Checkbox/Checkbox.test.tsx @@ -1,12 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - act, - createRenderer, - fireEvent, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, act, createRenderer, fireEvent } from '@mui-internal/test-utils'; import Checkbox, { checkboxClasses as classes } from '@mui/joy/Checkbox'; import { ThemeProvider } from '@mui/joy/styles'; import CloseIcon from '../internal/svg-icons/Close'; @@ -33,8 +27,6 @@ describe('', () => { skip: ['componentProp', 'componentsProp', 'classesRoot', 'propsSpread', 'themeVariants'], })); - describeJoyColorInversion(, { muiName: 'JoyCheckbox', classes }); - it('should have the classes required for Checkbox', () => { expect(classes).to.include.all.keys(['root', 'checked', 'disabled']); }); diff --git a/packages/mui-joy/src/Checkbox/Checkbox.tsx b/packages/mui-joy/src/Checkbox/Checkbox.tsx index 159085a3530a6e..91d0ebad97383b 100644 --- a/packages/mui-joy/src/Checkbox/Checkbox.tsx +++ b/packages/mui-joy/src/Checkbox/Checkbox.tsx @@ -6,7 +6,6 @@ import { unstable_useId as useId, unstable_capitalize as capitalize } from '@mui import { unstable_composeClasses as composeClasses } from '@mui/base'; import { useSwitch } from '@mui/base/useSwitch'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import checkboxClasses, { getCheckboxUtilityClass } from './checkboxClasses'; import { CheckboxOwnerState, CheckboxTypeMap } from './CheckboxProps'; @@ -270,11 +269,7 @@ const Checkbox = React.forwardRef(function Checkbox(inProps, ref) { const activeVariant = variantProp || 'solid'; const inactiveVariant = variantProp || 'outlined'; const variant = isCheckboxActive ? activeVariant : inactiveVariant; - const { getColor } = useColorInversion(variant); - const color = getColor( - inProps.color, - formControl?.error ? 'danger' : formControl?.color ?? colorProp, - ); + const color = inProps.color || (formControl?.error ? 'danger' : formControl?.color ?? colorProp); const activeColor = color || 'primary'; const inactiveColor = color || 'neutral'; diff --git a/packages/mui-joy/src/Chip/Chip.test.tsx b/packages/mui-joy/src/Chip/Chip.test.tsx index 952079d7dd80a6..35b1d18dd5147f 100644 --- a/packages/mui-joy/src/Chip/Chip.test.tsx +++ b/packages/mui-joy/src/Chip/Chip.test.tsx @@ -1,12 +1,7 @@ import * as React from 'react'; import { expect } from 'chai'; import { spy } from 'sinon'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, - fireEvent, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance, fireEvent } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import Chip, { ChipClassKey, chipClasses as classes } from '@mui/joy/Chip'; @@ -39,8 +34,6 @@ describe('', () => { }), ); - describeJoyColorInversion(, { muiName: 'JoyChip', classes }); - it('renders children', () => { const { getByText } = render( diff --git a/packages/mui-joy/src/Chip/Chip.tsx b/packages/mui-joy/src/Chip/Chip.tsx index 19793643c40d7c..557126e0b5c80f 100644 --- a/packages/mui-joy/src/Chip/Chip.tsx +++ b/packages/mui-joy/src/Chip/Chip.tsx @@ -9,7 +9,6 @@ import { unstable_capitalize as capitalize, unstable_useId as useId } from '@mui import { useThemeProps } from '../styles'; import styled from '../styles/styled'; import { VariantColorProvider } from '../styles/variantColorInheritance'; -import { useColorInversion } from '../styles/ColorInversion'; import { resolveSxValue } from '../styles/styleUtils'; import chipClasses, { getChipUtilityClass } from './chipClasses'; import { ChipProps, ChipOwnerState, ChipTypeMap } from './ChipProps'; @@ -224,7 +223,7 @@ const Chip = React.forwardRef(function Chip(inProps, ref) { const { children, className, - color: colorProp = 'neutral', + color = 'neutral', onClick, disabled = false, size = 'md', @@ -236,8 +235,6 @@ const Chip = React.forwardRef(function Chip(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const clickable = !!onClick || !!slotProps.action; const ownerState: ChipOwnerState = { @@ -313,7 +310,7 @@ const Chip = React.forwardRef(function Chip(inProps, ref) { return ( - + {clickable && } diff --git a/packages/mui-joy/src/ChipDelete/ChipDelete.test.tsx b/packages/mui-joy/src/ChipDelete/ChipDelete.test.tsx index b6ff05e0483838..d10ee7b52b5a70 100644 --- a/packages/mui-joy/src/ChipDelete/ChipDelete.test.tsx +++ b/packages/mui-joy/src/ChipDelete/ChipDelete.test.tsx @@ -1,13 +1,7 @@ import * as React from 'react'; import { expect } from 'chai'; import { spy } from 'sinon'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, - act, - fireEvent, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance, act, fireEvent } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import Chip from '@mui/joy/Chip'; import ChipDelete, { chipDeleteClasses as classes } from '@mui/joy/ChipDelete'; @@ -33,8 +27,6 @@ describe('', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyChipDelete', classes }); - describe('Chip context', () => { it('disabled', () => { const { getByRole } = render( diff --git a/packages/mui-joy/src/ChipDelete/ChipDelete.tsx b/packages/mui-joy/src/ChipDelete/ChipDelete.tsx index 162e729b5e4b7c..e08d729e3e16f2 100644 --- a/packages/mui-joy/src/ChipDelete/ChipDelete.tsx +++ b/packages/mui-joy/src/ChipDelete/ChipDelete.tsx @@ -8,7 +8,6 @@ import { useButton } from '@mui/base/useButton'; import { useThemeProps } from '../styles'; import styled from '../styles/styled'; import { useVariantColor } from '../styles/variantColorInheritance'; -import { useColorInversion } from '../styles/ColorInversion'; import Cancel from '../internal/svg-icons/Cancel'; import { getChipDeleteUtilityClass } from './chipDeleteClasses'; import { ChipDeleteProps, ChipDeleteOwnerState, ChipDeleteTypeMap } from './ChipDeleteProps'; @@ -83,8 +82,7 @@ const ChipDelete = React.forwardRef(function ChipDelete(inProps, ref) { inProps.color, true, ); - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, inheritedColor); + const color = inProps.color || inheritedColor; const disabled = disabledProp ?? chipContext.disabled; const buttonRef = React.useRef(null); diff --git a/packages/mui-joy/src/CircularProgress/CircularProgress.test.tsx b/packages/mui-joy/src/CircularProgress/CircularProgress.test.tsx index 8316f1454bb014..bc2b03624d0c32 100644 --- a/packages/mui-joy/src/CircularProgress/CircularProgress.test.tsx +++ b/packages/mui-joy/src/CircularProgress/CircularProgress.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import CircularProgress, { circularProgressClasses as classes } from '@mui/joy/CircularProgress'; @@ -38,8 +34,6 @@ describe('', () => { skip: ['classesRoot', 'componentsProp'], })); - describeJoyColorInversion(, { muiName: 'JoyCircularProgress', classes }); - describe('prop: determinate', () => { it('should render a determinate circular progress', () => { const { getByRole } = render(); diff --git a/packages/mui-joy/src/CircularProgress/CircularProgress.tsx b/packages/mui-joy/src/CircularProgress/CircularProgress.tsx index d53022c549a2a8..7b71e59c98ab72 100644 --- a/packages/mui-joy/src/CircularProgress/CircularProgress.tsx +++ b/packages/mui-joy/src/CircularProgress/CircularProgress.tsx @@ -8,7 +8,6 @@ import { unstable_composeClasses as composeClasses } from '@mui/base'; import { css, keyframes } from '@mui/system'; import styled from '../styles/styled'; import useThemeProps from '../styles/useThemeProps'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import { getCircularProgressUtilityClass } from './circularProgressClasses'; import { @@ -213,7 +212,7 @@ const CircularProgress = React.forwardRef(function CircularProgress(inProps, ref const { children, className, - color: colorProp = 'primary', + color = 'primary', size = 'md', variant = 'soft', thickness, @@ -224,8 +223,6 @@ const CircularProgress = React.forwardRef(function CircularProgress(inProps, ref slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, diff --git a/packages/mui-joy/src/DialogTitle/DialogTitle.tsx b/packages/mui-joy/src/DialogTitle/DialogTitle.tsx index 45c3a04bba3a40..8fb74285138d3b 100644 --- a/packages/mui-joy/src/DialogTitle/DialogTitle.tsx +++ b/packages/mui-joy/src/DialogTitle/DialogTitle.tsx @@ -6,7 +6,6 @@ import { unstable_composeClasses as composeClasses } from '@mui/base'; import { OverridableComponent } from '@mui/types'; import { useThemeProps } from '../styles'; import styled from '../styles/styled'; -import { useColorInversion } from '../styles/ColorInversion'; import { getDialogTitleUtilityClass } from './dialogTitleClasses'; import { DialogTitleProps, DialogTitleOwnerState, DialogTitleTypeMap } from './DialogTitleProps'; import cardOverflowClasses from '../CardOverflow/cardOverflowClasses'; @@ -89,8 +88,7 @@ const DialogTitle = React.forwardRef(function DialogTitle(inProps, ref) { ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, variant ? colorProp ?? 'neutral' : colorProp); + const color = inProps.color || (variant ? colorProp ?? 'neutral' : colorProp); const externalForwardedProps = { ...other, component, slots, slotProps }; diff --git a/packages/mui-joy/src/Drawer/Drawer.tsx b/packages/mui-joy/src/Drawer/Drawer.tsx index 80d5570da4faa2..c5b365418f8172 100644 --- a/packages/mui-joy/src/Drawer/Drawer.tsx +++ b/packages/mui-joy/src/Drawer/Drawer.tsx @@ -11,7 +11,8 @@ import { OverridableComponent } from '@mui/types'; import { unstable_useModal as useModal } from '@mui/base/unstable_useModal'; import { Portal } from '@mui/base/Portal'; import { FocusTrap } from '@mui/base/FocusTrap'; -import { useThemeProps, styled, ColorInversionProvider, useColorInversion } from '../styles'; +import { useThemeProps, styled } from '../styles'; +import { applySoftInversion, applySolidInversion } from '../colorInversion'; import { StyledModalBackdrop, StyledModalRoot } from '../Modal/Modal'; import CloseModalContext from '../Modal/CloseModalContext'; import useSlot from '../utils/useSlot'; @@ -43,7 +44,7 @@ const DrawerRoot = styled(StyledModalRoot as unknown as 'div', { name: 'JoyDrawer', slot: 'Root', overridesResolver: (props, styles) => styles.root, -})<{ ownerState: DrawerOwnerState }>(({ ownerState }) => ({ +})<{ ownerState: DrawerOwnerState }>(({ ownerState, theme }) => ({ '--Drawer-transitionDuration': '0.3s', '--Drawer-transitionFunction': 'ease', '--ModalClose-radius': @@ -71,6 +72,15 @@ const DrawerRoot = styled(StyledModalRoot as unknown as 'div', { ...(!ownerState.open && { visibility: 'hidden', }), + ...(ownerState.variant === 'solid' && + ownerState.color && + ownerState.invertedColors && + applySolidInversion(ownerState.color)(theme)), + ...(ownerState.variant === 'soft' && + ownerState.color && + ownerState.invertedColors && + applySoftInversion(ownerState.color)(theme)), + ...theme.variants[ownerState.variant!]?.[ownerState.color!], })); const DrawerBackdrop = styled(StyledModalBackdrop as unknown as 'div', { @@ -155,7 +165,7 @@ const Drawer = React.forwardRef(function Drawer(inProps, ref) { disableRestoreFocus = false, disableScrollLock = false, hideBackdrop = false, - color: colorProp = 'neutral', + color = 'neutral', variant = 'plain', invertedColors = false, size = 'md', @@ -168,9 +178,6 @@ const Drawer = React.forwardRef(function Drawer(inProps, ref) { ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); - const ownerState = { ...props, anchor, @@ -181,6 +188,7 @@ const Drawer = React.forwardRef(function Drawer(inProps, ref) { disableRestoreFocus, disableScrollLock, hideBackdrop, + invertedColors, color, variant, size, @@ -198,7 +206,7 @@ const Drawer = React.forwardRef(function Drawer(inProps, ref) { const labelledBy = useId(); const describedBy = useId(); const contextValue = React.useMemo( - () => ({ variant, color: color === 'context' ? undefined : color, labelledBy, describedBy }), + () => ({ variant, color, labelledBy, describedBy }), [color, variant, labelledBy, describedBy], ); @@ -233,7 +241,7 @@ const Drawer = React.forwardRef(function Drawer(inProps, ref) { ownerState, }); - const result = ( + return ( @@ -261,11 +269,6 @@ const Drawer = React.forwardRef(function Drawer(inProps, ref) { ); - - if (invertedColors) { - return {result}; - } - return result; }) as OverridableComponent; Drawer.propTypes /* remove-proptypes */ = { diff --git a/packages/mui-joy/src/FormControl/FormControl.tsx b/packages/mui-joy/src/FormControl/FormControl.tsx index 73ac96bb0ff231..e44bfb5e0e0d56 100644 --- a/packages/mui-joy/src/FormControl/FormControl.tsx +++ b/packages/mui-joy/src/FormControl/FormControl.tsx @@ -60,20 +60,16 @@ export const FormControlRoot = styled('div', { '--FormHelperText-fontSize': theme.vars.fontSize.sm, '--FormHelperText-lineHeight': theme.vars.lineHeight.sm, }), - ...(ownerState.color && - ownerState.color !== 'context' && { - '--FormHelperText-color': theme.vars.palette[ownerState.color]?.[500], - }), + ...(ownerState.color && { + '--FormHelperText-color': theme.vars.palette[ownerState.color]?.[500], + }), '--FormHelperText-margin': '0.375rem 0 0 0', [`&.${formControlClasses.error}`]: { '--FormHelperText-color': theme.vars.palette.danger[500], }, [`&.${formControlClasses.disabled}`]: { - ...(ownerState.color !== 'context' && { - '--FormLabel-color': theme.vars.palette[ownerState.color || 'neutral']?.plainDisabledColor, - '--FormHelperText-color': - theme.vars.palette[ownerState.color || 'neutral']?.plainDisabledColor, - }), + '--FormLabel-color': theme.variants.plainDisabled?.[ownerState.color || 'neutral']?.color, + '--FormHelperText-color': theme.variants.plainDisabled?.[ownerState.color || 'neutral']?.color, }, display: 'flex', position: 'relative', // for keeping the control action area, e.g. Switch diff --git a/packages/mui-joy/src/IconButton/IconButton.test.tsx b/packages/mui-joy/src/IconButton/IconButton.test.tsx index 358c1fb4e89a80..b7a76e2ad9e583 100644 --- a/packages/mui-joy/src/IconButton/IconButton.test.tsx +++ b/packages/mui-joy/src/IconButton/IconButton.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer } from '@mui-internal/test-utils'; import IconButton, { iconButtonClasses as classes } from '@mui/joy/IconButton'; import { ThemeProvider } from '@mui/joy/styles'; @@ -25,8 +21,6 @@ describe('Joy ', () => { skip: ['propsSpread', 'componentsProp', 'classesRoot'], })); - describeJoyColorInversion(, { muiName: 'JoyIconButton', classes }); - it('by default, should render with the root, variantPlain, sizeMd and colorNeutral classes', () => { const { getByRole } = render(Hello World); const button = getByRole('button'); diff --git a/packages/mui-joy/src/IconButton/IconButton.tsx b/packages/mui-joy/src/IconButton/IconButton.tsx index 8a169ab2d456c1..3a9513bf260353 100644 --- a/packages/mui-joy/src/IconButton/IconButton.tsx +++ b/packages/mui-joy/src/IconButton/IconButton.tsx @@ -5,7 +5,6 @@ import { unstable_capitalize as capitalize, unstable_useForkRef as useForkRef } import { useButton } from '@mui/base/useButton'; import { unstable_composeClasses as composeClasses } from '@mui/base/composeClasses'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import { getIconButtonUtilityClass } from './iconButtonClasses'; import { IconButtonOwnerState, IconButtonTypeMap, ExtendIconButton } from './IconButtonProps'; @@ -143,8 +142,7 @@ const IconButton = React.forwardRef(function IconButton(inProps, ref) { const buttonGroup = React.useContext(ButtonGroupContext); const variant = inProps.variant || buttonGroup.variant || variantProp; const size = inProps.size || buttonGroup.size || sizeProp; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, buttonGroup.color || colorProp); + const color = inProps.color || buttonGroup.color || colorProp; const disabled = inProps.disabled ?? (buttonGroup.disabled || disabledProp); const buttonRef = React.useRef(null); diff --git a/packages/mui-joy/src/Input/Input.test.tsx b/packages/mui-joy/src/Input/Input.test.tsx index 839ab09b864861..0cf5aaccba0c4f 100644 --- a/packages/mui-joy/src/Input/Input.test.tsx +++ b/packages/mui-joy/src/Input/Input.test.tsx @@ -3,7 +3,6 @@ import { expect } from 'chai'; import { spy } from 'sinon'; import { describeConformance, - describeJoyColorInversion, createRenderer, screen, act, @@ -33,8 +32,6 @@ describe('Joy ', () => { skip: ['propsSpread', 'componentsProp', 'classesRoot'], })); - describeJoyColorInversion(, { muiName: 'JoyInput', classes }); - it('should have placeholder', () => { const { getByPlaceholderText } = render(); expect(getByPlaceholderText('Placeholder')).toBeVisible(); diff --git a/packages/mui-joy/src/Input/Input.tsx b/packages/mui-joy/src/Input/Input.tsx index d925e827bf5496..00789d7dded579 100644 --- a/packages/mui-joy/src/Input/Input.tsx +++ b/packages/mui-joy/src/Input/Input.tsx @@ -5,11 +5,11 @@ import { unstable_capitalize as capitalize } from '@mui/utils'; import { OverridableComponent } from '@mui/types'; import { unstable_composeClasses as composeClasses } from '@mui/base/composeClasses'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import { InputTypeMap, InputProps, InputOwnerState } from './InputProps'; import inputClasses, { getInputUtilityClass } from './inputClasses'; import useForwardedInput from './useForwardedInput'; +import { INVERTED_COLORS_ATTR } from '../colorInversion/colorInversionUtils'; const useUtilityClasses = (ownerState: InputOwnerState) => { const { disabled, fullWidth, variant, color, size } = ownerState; @@ -43,16 +43,17 @@ export const StyledInputRoot = styled('div')<{ ownerState: InputOwnerState }>( '--Input-decoratorColor': theme.vars.palette.text.icon, '--Input-focused': '0', '--Input-focusedThickness': theme.vars.focus.thickness, - ...(ownerState.color === 'context' - ? { - '--Input-focusedHighlight': theme.vars.palette.focusVisible, - } - : { - '--Input-focusedHighlight': - theme.vars.palette[ - ownerState.color === 'neutral' ? 'primary' : ownerState.color! - ]?.[500], - }), + '--Input-focusedHighlight': + theme.vars.palette[ownerState.color === 'neutral' ? 'primary' : ownerState.color!]?.[500], + [`&:not([${INVERTED_COLORS_ATTR}])`]: { + ...(ownerState.instanceColor && { + '--_Input-focusedHighlight': + theme.vars.palette[ + ownerState.instanceColor === 'neutral' ? 'primary' : ownerState.instanceColor + ]?.[500], + }), + '--Input-focusedHighlight': `var(--_Input-focusedHighlight, ${theme.vars.palette.focusVisible})`, + }, ...(ownerState.size === 'sm' && { '--Input-minHeight': '2rem', '--Input-paddingInline': '0.5rem', @@ -280,10 +281,10 @@ const Input = React.forwardRef(function Input(inProps, ref) { const error = inProps.error ?? formControl?.error ?? errorProp; const size = inProps.size ?? formControl?.size ?? sizeProp; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, error ? 'danger' : formControl?.color ?? colorProp); + const color = inProps.color ?? (error ? 'danger' : formControl?.color ?? colorProp); const ownerState = { + instanceColor: error ? 'danger' : inProps.color, ...props, fullWidth, color, diff --git a/packages/mui-joy/src/Input/InputProps.ts b/packages/mui-joy/src/Input/InputProps.ts index e19444406e535f..390029ee5aeea9 100644 --- a/packages/mui-joy/src/Input/InputProps.ts +++ b/packages/mui-joy/src/Input/InputProps.ts @@ -124,4 +124,8 @@ export interface InputOwnerState extends ApplyColorInversion { * If `true`, the input is focused. */ focused: boolean; + /** + * @internal + */ + instanceColor?: OverridableStringUnion; } diff --git a/packages/mui-joy/src/LinearProgress/LinearProgress.test.tsx b/packages/mui-joy/src/LinearProgress/LinearProgress.test.tsx index 1cdac10d1daab2..5a976b191e5386 100644 --- a/packages/mui-joy/src/LinearProgress/LinearProgress.test.tsx +++ b/packages/mui-joy/src/LinearProgress/LinearProgress.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import LinearProgress, { linearProgressClasses as classes } from '@mui/joy/LinearProgress'; @@ -28,8 +24,6 @@ describe('', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyLinearProgress', classes }); - describe('prop: determinate', () => { it('should render a determinate circular progress', () => { const { getByRole } = render(); diff --git a/packages/mui-joy/src/LinearProgress/LinearProgress.tsx b/packages/mui-joy/src/LinearProgress/LinearProgress.tsx index af9fa127fda8a0..ee7d2d03f8ec9b 100644 --- a/packages/mui-joy/src/LinearProgress/LinearProgress.tsx +++ b/packages/mui-joy/src/LinearProgress/LinearProgress.tsx @@ -8,7 +8,6 @@ import { unstable_composeClasses as composeClasses } from '@mui/base'; import { css, keyframes } from '@mui/system'; import styled from '../styles/styled'; import useThemeProps from '../styles/useThemeProps'; -import { useColorInversion } from '../styles/ColorInversion'; import { getLinearProgressUtilityClass } from './linearProgressClasses'; import { LinearProgressOwnerState, @@ -167,7 +166,7 @@ const LinearProgress = React.forwardRef(function LinearProgress(inProps, ref) { children, className, component, - color: colorProp = 'primary', + color = 'primary', size = 'md', variant = 'soft', thickness, @@ -178,8 +177,6 @@ const LinearProgress = React.forwardRef(function LinearProgress(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, diff --git a/packages/mui-joy/src/Link/Link.test.tsx b/packages/mui-joy/src/Link/Link.test.tsx index f5ec486606602e..5271fe9a682422 100644 --- a/packages/mui-joy/src/Link/Link.test.tsx +++ b/packages/mui-joy/src/Link/Link.test.tsx @@ -1,13 +1,7 @@ import * as React from 'react'; import { expect } from 'chai'; import { SinonSpy, spy } from 'sinon'; -import { - act, - createRenderer, - fireEvent, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { act, createRenderer, fireEvent, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import Link, { LinkClassKey, linkClasses as classes } from '@mui/joy/Link'; import Typography from '@mui/joy/Typography'; @@ -46,8 +40,6 @@ describe('', () => { }), ); - describeJoyColorInversion(, { muiName: 'JoyLink', classes }); - it('should render children', () => { const { queryByText } = render(Home); diff --git a/packages/mui-joy/src/Link/Link.tsx b/packages/mui-joy/src/Link/Link.tsx index d2dfc19f4c69db..c3f813d60ddac1 100644 --- a/packages/mui-joy/src/Link/Link.tsx +++ b/packages/mui-joy/src/Link/Link.tsx @@ -12,7 +12,6 @@ import { import { unstable_extendSxProp as extendSxProp } from '@mui/system'; import styled from '../styles/styled'; import useThemeProps from '../styles/useThemeProps'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import linkClasses, { getLinkUtilityClass } from './linkClasses'; import { LinkProps, LinkOwnerState, LinkTypeMap } from './LinkProps'; @@ -109,11 +108,9 @@ const LinkRoot = styled('a', { borderRadius: theme.vars.radius.xs, padding: 0, // Remove the padding in Firefox cursor: 'pointer', - ...(ownerState.color !== 'context' && { - textDecorationColor: `rgba(${ - theme.vars.palette[ownerState.color!]?.mainChannel - } / var(--Link-underlineOpacity, 0.72))`, - }), + textDecorationColor: `var(--variant-outlinedBorder, rgba(${ + theme.vars.palette[ownerState.color!]?.mainChannel + } / var(--Link-underlineOpacity, 0.72)))`, ...(ownerState.variant ? { paddingBlock: 'min(0.1em, 4px)', @@ -123,14 +120,14 @@ const LinkRoot = styled('a', { }), } : { - ...(ownerState.color !== 'context' && { - color: `rgba(${theme.vars.palette[ownerState.color!]?.mainChannel} / 1)`, - }), + color: `var(--variant-plainColor, rgba(${ + theme.vars.palette[ownerState.color!]?.mainChannel + } / 1))`, [`&.${linkClasses.disabled}`]: { pointerEvents: 'none', - ...(ownerState.color !== 'context' && { - color: `rgba(${theme.vars.palette[ownerState.color!]?.mainChannel} / 0.6)`, - }), + color: `var(--variant-plainDisabledColor, rgba(${ + theme.vars.palette[ownerState.color!]?.mainChannel + } / 0.6))`, }, }), userSelect: 'none', @@ -183,7 +180,7 @@ const LinkRoot = styled('a', { */ const Link = React.forwardRef(function Link(inProps, ref) { const { - color: colorProp = 'primary', + color = 'primary', textColor, variant, ...themeProps @@ -192,8 +189,6 @@ const Link = React.forwardRef(function Link(inProps, ref) { name: 'JoyLink', }); - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const nesting = React.useContext(TypographyNestedContext); const inheriting = React.useContext(TypographyInheritContext); diff --git a/packages/mui-joy/src/List/List.test.tsx b/packages/mui-joy/src/List/List.test.tsx index 5959b877d3619f..8fe2fa12f5b6a2 100644 --- a/packages/mui-joy/src/List/List.test.tsx +++ b/packages/mui-joy/src/List/List.test.tsx @@ -1,11 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - screen, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer, screen } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import List, { listClasses as classes } from '@mui/joy/List'; import ListItem from '@mui/joy/ListItem'; @@ -34,8 +29,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyList', classes }); - it('should have root className', () => { const { container } = render(); expect(container.firstChild).to.have.class(classes.root); diff --git a/packages/mui-joy/src/List/List.tsx b/packages/mui-joy/src/List/List.tsx index 19eb0bec8922d3..7f411c1f56ea49 100644 --- a/packages/mui-joy/src/List/List.tsx +++ b/packages/mui-joy/src/List/List.tsx @@ -7,7 +7,7 @@ import { OverridableComponent } from '@mui/types'; import { unstable_composeClasses as composeClasses } from '@mui/base/composeClasses'; import { styled, useThemeProps } from '../styles'; import { resolveSxValue } from '../styles/styleUtils'; -import { useColorInversion } from '../styles/ColorInversion'; + import { ListProps, ListOwnerState, ListTypeMap } from './ListProps'; import { getListUtilityClass } from './listClasses'; import NestedListContext from './NestedListContext'; @@ -184,14 +184,12 @@ const List = React.forwardRef(function List(inProps, ref) { orientation = 'vertical', wrap = false, variant = 'plain', - color: colorProp = 'neutral', + color = 'neutral', role: roleProp, slots = {}, slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const size = sizeProp || (inProps.size ?? 'md'); let role; diff --git a/packages/mui-joy/src/ListItem/ListItem.test.tsx b/packages/mui-joy/src/ListItem/ListItem.test.tsx index 9275155bcfbd99..e627bc6fa25599 100644 --- a/packages/mui-joy/src/ListItem/ListItem.test.tsx +++ b/packages/mui-joy/src/ListItem/ListItem.test.tsx @@ -1,11 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - screen, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer, screen } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import MenuList from '@mui/joy/MenuList'; import List from '@mui/joy/List'; @@ -32,8 +27,6 @@ describe('Joy ', () => { skip: ['componentsProp', 'classesRoot'], })); - describeJoyColorInversion(, { muiName: 'JoyListItem', classes }); - it('should have root className', () => { const { container } = render(); expect(container.firstChild).to.have.class(classes.root); diff --git a/packages/mui-joy/src/ListItem/ListItem.tsx b/packages/mui-joy/src/ListItem/ListItem.tsx index 69f0a1ed3e8d0d..9b49c821c9c832 100644 --- a/packages/mui-joy/src/ListItem/ListItem.tsx +++ b/packages/mui-joy/src/ListItem/ListItem.tsx @@ -9,7 +9,7 @@ import { import { OverridableComponent } from '@mui/types'; import { unstable_composeClasses as composeClasses } from '@mui/base/composeClasses'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; + import useSlot from '../utils/useSlot'; import { ListItemOwnerState, ListItemTypeMap } from './ListItemProps'; import { getListItemUtilityClass } from './listItemClasses'; @@ -170,7 +170,7 @@ const ListItem = React.forwardRef(function ListItem(inProps, ref) { nested = false, sticky = false, variant = 'plain', - color: colorProp = 'neutral', + color = 'neutral', startAction, endAction, role: roleProp, @@ -178,8 +178,6 @@ const ListItem = React.forwardRef(function ListItem(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const [subheaderId, setSubheaderId] = React.useState(''); diff --git a/packages/mui-joy/src/ListItemButton/ListItemButton.test.tsx b/packages/mui-joy/src/ListItemButton/ListItemButton.test.tsx index 4d84724100881f..22b43ce0f88bef 100644 --- a/packages/mui-joy/src/ListItemButton/ListItemButton.test.tsx +++ b/packages/mui-joy/src/ListItemButton/ListItemButton.test.tsx @@ -1,12 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - describeJoyColorInversion, - createRenderer, - act, - fireEvent, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer, act, fireEvent } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import ListItemButton, { listItemButtonClasses as classes } from '@mui/joy/ListItemButton'; @@ -30,8 +24,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyListItemButton', classes }); - it('should render with the selected class', () => { const { getByRole } = render(); expect(getByRole('button')).to.have.class(classes.selected); diff --git a/packages/mui-joy/src/ListItemButton/ListItemButton.tsx b/packages/mui-joy/src/ListItemButton/ListItemButton.tsx index fce1b49843a663..a9712b5cb8a584 100644 --- a/packages/mui-joy/src/ListItemButton/ListItemButton.tsx +++ b/packages/mui-joy/src/ListItemButton/ListItemButton.tsx @@ -6,7 +6,7 @@ import { unstable_capitalize as capitalize, unstable_useForkRef as useForkRef } import { unstable_composeClasses as composeClasses } from '@mui/base/composeClasses'; import { useButton } from '@mui/base/useButton'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; + import { ListItemButtonOwnerState, ExtendListItemButton, @@ -146,16 +146,13 @@ const ListItemButton = React.forwardRef(function ListItemButton(inProps, ref) { orientation = 'horizontal', role, selected = false, - color: colorProp = 'neutral', + color = 'neutral', variant = 'plain', slots = {}, slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); - const buttonRef = React.useRef(null); const handleRef = useForkRef(buttonRef, ref); diff --git a/packages/mui-joy/src/ListSubheader/ListSubheader.test.tsx b/packages/mui-joy/src/ListSubheader/ListSubheader.test.tsx index 0e406e20ed9473..b49ab8229acd86 100644 --- a/packages/mui-joy/src/ListSubheader/ListSubheader.test.tsx +++ b/packages/mui-joy/src/ListSubheader/ListSubheader.test.tsx @@ -1,11 +1,7 @@ import * as React from 'react'; import { expect } from 'chai'; import { spy } from 'sinon'; -import { - describeConformance, - createRenderer, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import ListSubheader, { listSubheaderClasses as classes } from '@mui/joy/ListSubheader'; import ListSubheaderDispatch from './ListSubheaderContext'; @@ -30,11 +26,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { - muiName: 'JoyListSubheader', - classes, - }); - it('should have root className', () => { const { container } = render(); expect(container.firstChild).to.have.class(classes.root); diff --git a/packages/mui-joy/src/ListSubheader/ListSubheader.tsx b/packages/mui-joy/src/ListSubheader/ListSubheader.tsx index 2994fe22581af7..489946b6bbf73b 100644 --- a/packages/mui-joy/src/ListSubheader/ListSubheader.tsx +++ b/packages/mui-joy/src/ListSubheader/ListSubheader.tsx @@ -6,11 +6,11 @@ import { OverridableComponent } from '@mui/types'; import { unstable_useId as useId, unstable_capitalize as capitalize } from '@mui/utils'; import { unstable_composeClasses as composeClasses } from '@mui/base/composeClasses'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import { ListSubheaderOwnerState, ListSubheaderTypeMap } from './ListSubheaderProps'; import { getListSubheaderUtilityClass } from './listSubheaderClasses'; import ListSubheaderDispatch from './ListSubheaderContext'; import useSlot from '../utils/useSlot'; +import { INVERTED_COLORS_ATTR } from '../colorInversion/colorInversionUtils'; const useUtilityClasses = (ownerState: ListSubheaderOwnerState) => { const { variant, color, sticky } = ownerState; @@ -49,10 +49,14 @@ const ListSubheaderRoot = styled('div', { zIndex: 1, background: 'var(--ListItem-stickyBackground)', }), - color: - ownerState.color && ownerState.color !== 'context' - ? `rgba(${theme.vars.palette[ownerState.color!]?.mainChannel} / 1)` - : theme.vars.palette.text.tertiary, + color: ownerState.color + ? `var(--_Link-color, rgba(${theme.vars.palette[ownerState.color!]?.mainChannel} / 1))` + : theme.vars.palette.text.tertiary, + ...(ownerState.instanceColor && { + [`&:not([${INVERTED_COLORS_ATTR}])`]: { + '--_Link-color': theme.vars.palette.text.secondary, + }, + }), ...theme.variants[ownerState.variant!]?.[ownerState.color!], })); /** @@ -78,13 +82,11 @@ const ListSubheader = React.forwardRef(function ListSubheader(inProps, ref) { id: idOverride, sticky = false, variant, - color: colorProp, + color, slots = {}, slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const id = useId(idOverride); const setSubheaderId = React.useContext(ListSubheaderDispatch); @@ -95,6 +97,7 @@ const ListSubheader = React.forwardRef(function ListSubheader(inProps, ref) { }, [setSubheaderId, id]); const ownerState = { + instanceColor: inProps.color, ...props, id, sticky, diff --git a/packages/mui-joy/src/ListSubheader/ListSubheaderProps.ts b/packages/mui-joy/src/ListSubheader/ListSubheaderProps.ts index c372583eb5d884..9bda3152718767 100644 --- a/packages/mui-joy/src/ListSubheader/ListSubheaderProps.ts +++ b/packages/mui-joy/src/ListSubheader/ListSubheaderProps.ts @@ -57,4 +57,9 @@ export type ListSubheaderProps< }, > = OverrideProps, D>; -export interface ListSubheaderOwnerState extends ApplyColorInversion {} +export interface ListSubheaderOwnerState extends ApplyColorInversion { + /** + * @internal + */ + instanceColor?: OverridableStringUnion; +} diff --git a/packages/mui-joy/src/Menu/Menu.test.tsx b/packages/mui-joy/src/Menu/Menu.test.tsx index 143b191faab91e..13221bcafb8d49 100644 --- a/packages/mui-joy/src/Menu/Menu.test.tsx +++ b/packages/mui-joy/src/Menu/Menu.test.tsx @@ -7,7 +7,6 @@ import { describeConformance, screen, fireEvent, - describeJoyColorInversion, } from '@mui-internal/test-utils'; import { Popper as PopperUnstyled } from '@mui/base/Popper'; import { DropdownContext, DropdownContextValue } from '@mui/base/useDropdown'; @@ -62,15 +61,6 @@ describe('Joy ', () => { const anchorEl = document.createElement('div'); anchorEl.setAttribute('aria-controls', 'test'); - describeJoyColorInversion( - anchorEl} data-testid="test-element" />, - { - muiName: 'JoyMenu', - classes, - portalSlot: 'root', - }, - ); - it('should render with `ul` by default', () => { render(); expect(screen.getByTestId('popover')).to.have.tagName('ul'); diff --git a/packages/mui-joy/src/Menu/Menu.tsx b/packages/mui-joy/src/Menu/Menu.tsx index e8f219ed3a4ef3..7cd1f5b16d96dd 100644 --- a/packages/mui-joy/src/Menu/Menu.tsx +++ b/packages/mui-joy/src/Menu/Menu.tsx @@ -12,11 +12,8 @@ import { StyledList } from '../List/List'; import ListProvider, { scopedVariables } from '../List/ListProvider'; import GroupListContext from '../List/GroupListContext'; import { styled, useThemeProps } from '../styles'; +import { applySolidInversion, applySoftInversion } from '../colorInversion'; import { VariantColorProvider } from '../styles/variantColorInheritance'; -import ColorInversion, { - ColorInversionProvider, - useColorInversion, -} from '../styles/ColorInversion'; import { MenuTypeMap, MenuOwnerState } from './MenuProps'; import { getMenuUtilityClass } from './menuClasses'; import { ListOwnerState } from '../List'; @@ -60,10 +57,16 @@ const MenuRoot = styled(StyledList, { ...(!variantStyle?.backgroundColor && { backgroundColor: theme.vars.palette.background.popup, }), + ...(ownerState.variant === 'solid' && + ownerState.color && + ownerState.invertedColors && + applySolidInversion(ownerState.color)(theme)), + ...(ownerState.variant === 'soft' && + ownerState.color && + ownerState.invertedColors && + applySoftInversion(ownerState.color)(theme)), + ...theme.variants[ownerState.variant!]?.[ownerState.color!], }, - ownerState.color !== 'context' && - ownerState.invertedColors && - theme.colorInversion[ownerState.variant!]?.[ownerState.color!], ]; }); @@ -87,7 +90,7 @@ const Menu = React.forwardRef(function Menu(inProps, ref: React.ForwardedRef - {/* If `invertedColors` is true, let the children use their default variant */} - - - {children} - - - - ); - - if (invertedColors) { - result = {result}; - } - - result = ( + return ( - {result} + + {/* If `invertedColors` is true, let the children use their default variant */} + + + {children} + + + ); - - return disablePortal ? ( - result - ) : ( - // For portal popup, the children should not inherit color inversion from the upper parent. - {result} - ); }) as OverridableComponent; Menu.propTypes /* remove-proptypes */ = { diff --git a/packages/mui-joy/src/MenuButton/MenuButton.tsx b/packages/mui-joy/src/MenuButton/MenuButton.tsx index 4b6522d4cba035..b5f5dfae77b7d1 100644 --- a/packages/mui-joy/src/MenuButton/MenuButton.tsx +++ b/packages/mui-joy/src/MenuButton/MenuButton.tsx @@ -15,7 +15,7 @@ import useThemeProps from '../styles/useThemeProps'; import useSlot from '../utils/useSlot'; import CircularProgress from '../CircularProgress'; import { getButtonStyles } from '../Button/Button'; -import { styled, useColorInversion } from '../styles'; +import { styled } from '../styles'; import ButtonGroupContext from '../ButtonGroup/ButtonGroupContext'; const useUtilityClasses = (ownerState: MenuButtonOwnerState) => { @@ -103,7 +103,7 @@ const MenuButton = React.forwardRef(function MenuButton( const { children, - color: colorProp = 'neutral', + color = 'neutral', component, disabled: disabledProp = false, endDecorator, @@ -121,17 +121,12 @@ const MenuButton = React.forwardRef(function MenuButton( const variant = inProps.variant || buttonGroup.variant || variantProp; const size = inProps.size || buttonGroup.size || sizeProp; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, buttonGroup.color || colorProp); const disabled = inProps.disabled ?? (buttonGroup.disabled || disabledProp || loading); const { getRootProps, open, active } = useMenuButton({ rootRef: forwardedRef, disabled }); const loadingIndicator = loadingIndicatorProp ?? ( - + ); const ownerState: MenuButtonOwnerState = { diff --git a/packages/mui-joy/src/MenuItem/MenuItem.test.tsx b/packages/mui-joy/src/MenuItem/MenuItem.test.tsx index 3bce808e700b04..c41f510db2c588 100644 --- a/packages/mui-joy/src/MenuItem/MenuItem.test.tsx +++ b/packages/mui-joy/src/MenuItem/MenuItem.test.tsx @@ -7,7 +7,6 @@ import { createRenderer, fireEvent, screen, - describeJoyColorInversion, } from '@mui-internal/test-utils'; import { MenuProvider, MenuProviderValue } from '@mui/base/useMenu'; import { ThemeProvider } from '@mui/joy/styles'; @@ -64,12 +63,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { - muiName: 'JoyMenuItem', - classes, - wrapper: (node) => {node}, - }); - it('should render with the variant class', () => { const { getByRole } = render(); expect(getByRole('menuitem')).to.have.class(classes.variantOutlined); diff --git a/packages/mui-joy/src/MenuItem/MenuItem.tsx b/packages/mui-joy/src/MenuItem/MenuItem.tsx index 1eff87a1cbdddc..d1ce24622802f0 100644 --- a/packages/mui-joy/src/MenuItem/MenuItem.tsx +++ b/packages/mui-joy/src/MenuItem/MenuItem.tsx @@ -6,7 +6,6 @@ import { unstable_composeClasses as composeClasses } from '@mui/base/composeClas import { useMenuItem } from '@mui/base/useMenuItem'; import { StyledListItemButton } from '../ListItemButton/ListItemButton'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import { useVariantColor } from '../styles/variantColorInheritance'; import { getMenuItemUtilityClass } from './menuItemClasses'; import { MenuItemOwnerState, ExtendMenuItem, MenuItemTypeMap } from './MenuItemProps'; @@ -68,12 +67,10 @@ const MenuItem = React.forwardRef(function MenuItem(inProps, ref) { slotProps = {}, ...other } = props; - const { variant = variantProp, color: inheritedColor = colorProp } = useVariantColor( + const { variant = variantProp, color = colorProp } = useVariantColor( inProps.variant, inProps.color, ); - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, inheritedColor); const { getRootProps, disabled, focusVisible } = useMenuItem({ disabled: disabledProp, diff --git a/packages/mui-joy/src/MenuList/MenuList.test.tsx b/packages/mui-joy/src/MenuList/MenuList.test.tsx index a95592b17bc4b3..a34765831c5d4f 100644 --- a/packages/mui-joy/src/MenuList/MenuList.test.tsx +++ b/packages/mui-joy/src/MenuList/MenuList.test.tsx @@ -1,11 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - screen, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer, screen } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import MenuList, { menuListClasses as classes } from '@mui/joy/MenuList'; @@ -29,8 +24,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyMenuList', classes }); - it('should have root className', () => { const { container } = render(); expect(container.firstChild).to.have.class(classes.root); diff --git a/packages/mui-joy/src/MenuList/MenuList.tsx b/packages/mui-joy/src/MenuList/MenuList.tsx index 5e0f46e8c0540b..6ea40711086783 100644 --- a/packages/mui-joy/src/MenuList/MenuList.tsx +++ b/packages/mui-joy/src/MenuList/MenuList.tsx @@ -7,7 +7,6 @@ import { unstable_composeClasses as composeClasses } from '@mui/base/composeClas import { useMenu, MenuProvider } from '@mui/base/useMenu'; import { ListActionTypes } from '@mui/base/useList'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import { StyledList } from '../List/List'; import ListProvider, { scopedVariables } from '../List/ListProvider'; import GroupListContext from '../List/GroupListContext'; @@ -73,14 +72,12 @@ const MenuList = React.forwardRef(function MenuList(inProps, ref) { children, size = 'md', variant = 'outlined', - color: colorProp = 'neutral', + color = 'neutral', onItemsChange, slots = {}, slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const { contextValue: menuContextValue, diff --git a/packages/mui-joy/src/ModalClose/ModalClose.test.tsx b/packages/mui-joy/src/ModalClose/ModalClose.test.tsx index 4bd68b15827b79..983515365110b1 100644 --- a/packages/mui-joy/src/ModalClose/ModalClose.test.tsx +++ b/packages/mui-joy/src/ModalClose/ModalClose.test.tsx @@ -1,12 +1,7 @@ import * as React from 'react'; import { expect } from 'chai'; import { spy } from 'sinon'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, - fireEvent, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance, fireEvent } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import Modal from '@mui/joy/Modal'; @@ -32,8 +27,6 @@ describe('', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyModalClose', classes }); - describe('prop: variant', () => { it('plain by default', () => { const { getByRole } = render(); diff --git a/packages/mui-joy/src/ModalClose/ModalClose.tsx b/packages/mui-joy/src/ModalClose/ModalClose.tsx index 6c17882eb34635..0234a1129def12 100644 --- a/packages/mui-joy/src/ModalClose/ModalClose.tsx +++ b/packages/mui-joy/src/ModalClose/ModalClose.tsx @@ -7,7 +7,6 @@ import { unstable_capitalize as capitalize } from '@mui/utils'; import { useButton } from '@mui/base/useButton'; import useSlot from '../utils/useSlot'; import { useThemeProps, styled } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import { StyledIconButton } from '../IconButton/IconButton'; import { getModalCloseUtilityClass } from './modalCloseClasses'; import { ModalCloseProps, ModalCloseOwnerState, ModalCloseTypeMap } from './ModalCloseProps'; @@ -96,8 +95,7 @@ const ModalClose = React.forwardRef(function ModalClose(inProps, ref) { const modalDialogVariantColor = React.useContext(ModalDialogVariantColorContext); const variant = inProps.variant ?? modalDialogVariantMapping[modalDialogVariantColor?.variant!] ?? variantProp; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, modalDialogVariantColor?.color ?? colorProp); + const color = inProps.color ?? modalDialogVariantColor?.color ?? colorProp; const modalDialogSize = React.useContext(ModalDialogSizeContext); const size = inProps.size ?? modalDialogSize ?? sizeProp; diff --git a/packages/mui-joy/src/ModalDialog/ModalDialog.test.tsx b/packages/mui-joy/src/ModalDialog/ModalDialog.test.tsx index c07f4e3e5d84e2..3c9c5ac5f38838 100644 --- a/packages/mui-joy/src/ModalDialog/ModalDialog.test.tsx +++ b/packages/mui-joy/src/ModalDialog/ModalDialog.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import ModalDialog, { modalDialogClasses as classes } from '@mui/joy/ModalDialog'; @@ -29,8 +25,6 @@ describe('', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyModalDialog', classes }); - describe('prop: variant', () => { it('plain by default', () => { const { getByRole } = render(); diff --git a/packages/mui-joy/src/ModalDialog/ModalDialog.tsx b/packages/mui-joy/src/ModalDialog/ModalDialog.tsx index 7f7f05d3e259d1..d90d3e72222f1d 100644 --- a/packages/mui-joy/src/ModalDialog/ModalDialog.tsx +++ b/packages/mui-joy/src/ModalDialog/ModalDialog.tsx @@ -12,7 +12,6 @@ import { import { Breakpoint } from '@mui/system'; import { styled, useThemeProps } from '../styles'; import { Theme } from '../styles/types/theme'; -import { ColorInversionProvider, useColorInversion } from '../styles/ColorInversion'; import { getModalDialogUtilityClass } from './modalDialogClasses'; import { ModalDialogProps, ModalDialogOwnerState, ModalDialogTypeMap } from './ModalDialogProps'; import ModalDialogSizeContext from './ModalDialogSizeContext'; @@ -137,7 +136,7 @@ const ModalDialog = React.forwardRef(function ModalDialog(inProps, ref) { children, invertedColors = false, orientation = 'vertical', - color: colorProp = 'neutral', + color = 'neutral', component = 'div', variant = 'outlined', size = 'md', @@ -148,8 +147,6 @@ const ModalDialog = React.forwardRef(function ModalDialog(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, @@ -160,6 +157,7 @@ const ModalDialog = React.forwardRef(function ModalDialog(inProps, ref) { layout, size, variant, + invertedColors, }; const classes = useUtilityClasses(ownerState); @@ -168,7 +166,7 @@ const ModalDialog = React.forwardRef(function ModalDialog(inProps, ref) { const labelledBy = useId(); const describedBy = useId(); const contextValue = React.useMemo( - () => ({ variant, color: color === 'context' ? undefined : color, labelledBy, describedBy }), + () => ({ variant, color, labelledBy, describedBy }), [color, variant, labelledBy, describedBy], ); @@ -187,7 +185,7 @@ const ModalDialog = React.forwardRef(function ModalDialog(inProps, ref) { }, }); - const result = ( + return ( @@ -223,11 +221,6 @@ const ModalDialog = React.forwardRef(function ModalDialog(inProps, ref) { ); - - if (invertedColors) { - return {result}; - } - return result; }) as OverridableComponent; ModalDialog.propTypes /* remove-proptypes */ = { diff --git a/packages/mui-joy/src/Option/Option.tsx b/packages/mui-joy/src/Option/Option.tsx index 2787cf251ed04d..bf7b29f026e3b1 100644 --- a/packages/mui-joy/src/Option/Option.tsx +++ b/packages/mui-joy/src/Option/Option.tsx @@ -7,7 +7,6 @@ import { unstable_useForkRef as useForkRef } from '@mui/utils'; import useSlot from '../utils/useSlot'; import { StyledListItemButton } from '../ListItemButton/ListItemButton'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import { useVariantColor } from '../styles/variantColorInheritance'; import { OptionOwnerState, ExtendOption, OptionTypeMap } from './OptionProps'; import optionClasses, { getOptionUtilityClass } from './optionClasses'; @@ -65,7 +64,7 @@ const Option = React.forwardRef(function Option(inProps, ref: React.ForwardedRef } = props; const row = React.useContext(RowListContext); - const { variant = variantProp, color: inheritedColor = colorProp } = useVariantColor( + const { variant = variantProp, color = colorProp } = useVariantColor( inProps.variant, inProps.color, ); @@ -82,9 +81,6 @@ const Option = React.forwardRef(function Option(inProps, ref: React.ForwardedRef rootRef: combinedRef, }); - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, inheritedColor); - const ownerState: OptionOwnerState = { ...props, disabled, diff --git a/packages/mui-joy/src/Radio/Radio.test.tsx b/packages/mui-joy/src/Radio/Radio.test.tsx index eaf3bf40235e82..cd475f782c6c5f 100644 --- a/packages/mui-joy/src/Radio/Radio.test.tsx +++ b/packages/mui-joy/src/Radio/Radio.test.tsx @@ -1,12 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - act, - createRenderer, - fireEvent, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, act, createRenderer, fireEvent } from '@mui-internal/test-utils'; import Radio, { radioClasses as classes } from '@mui/joy/Radio'; import { ThemeProvider } from '@mui/joy/styles'; @@ -33,8 +27,6 @@ describe('', () => { skip: ['componentProp', 'componentsProp', 'classesRoot', 'propsSpread'], })); - describeJoyColorInversion(, { muiName: 'JoyRadio', classes }); - it('should have the classes required for Radio', () => { expect(classes).to.include.all.keys(['root', 'checked', 'disabled']); }); diff --git a/packages/mui-joy/src/Radio/Radio.tsx b/packages/mui-joy/src/Radio/Radio.tsx index 0cfb23c5aa0b6d..9249c338b7516f 100644 --- a/packages/mui-joy/src/Radio/Radio.tsx +++ b/packages/mui-joy/src/Radio/Radio.tsx @@ -6,7 +6,6 @@ import { unstable_capitalize as capitalize, unstable_useId as useId } from '@mui import { unstable_composeClasses as composeClasses } from '@mui/base'; import { useSwitch } from '@mui/base/useSwitch'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import radioClasses, { getRadioUtilityClass } from './radioClasses'; import { RadioOwnerState, RadioTypeMap } from './RadioProps'; @@ -266,7 +265,6 @@ const Radio = React.forwardRef(function Radio(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); const formControl = React.useContext(FormControlContext); @@ -311,7 +309,7 @@ const Radio = React.forwardRef(function Radio(inProps, ref) { const { getInputProps, checked, disabled, focusVisible } = useSwitch(useRadioProps); - const color = getColor(inProps.color, checked ? activeColor : inactiveColor); + const color = inProps.color ?? (checked ? activeColor : inactiveColor); const ownerState = { ...props, diff --git a/packages/mui-joy/src/Select/Select.test.tsx b/packages/mui-joy/src/Select/Select.test.tsx index 239871e5a8e49f..06c7060b14c3ba 100644 --- a/packages/mui-joy/src/Select/Select.test.tsx +++ b/packages/mui-joy/src/Select/Select.test.tsx @@ -3,7 +3,6 @@ import { expect } from 'chai'; import { spy, stub } from 'sinon'; import { describeConformance, - describeJoyColorInversion, act, createRenderer, fireEvent, @@ -65,12 +64,6 @@ describe('Joy , { - muiName: 'JoySelect', - classes, - portalSlot: 'listbox', - }); - it('should be able to mount the component', () => { render( - {result} + {anchorEl && ( + + + + + {/* for building grouped options */} + {children} + + + + + )} ); }) as SelectComponent; diff --git a/packages/mui-joy/src/Select/SelectProps.ts b/packages/mui-joy/src/Select/SelectProps.ts index 0ab6e0c6822f9b..de93fbe25ae293 100644 --- a/packages/mui-joy/src/Select/SelectProps.ts +++ b/packages/mui-joy/src/Select/SelectProps.ts @@ -215,6 +215,10 @@ export interface SelectOwnerState * If `true`, the select dropdown is open. */ open: boolean; + /** + * @internal + */ + instanceColor?: OverridableStringUnion; } export interface SelectTypeMap< diff --git a/packages/mui-joy/src/Sheet/Sheet.test.tsx b/packages/mui-joy/src/Sheet/Sheet.test.tsx index 6b9c58e3752461..98b0b264b8d67c 100644 --- a/packages/mui-joy/src/Sheet/Sheet.test.tsx +++ b/packages/mui-joy/src/Sheet/Sheet.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import Sheet, { sheetClasses as classes, SheetClassKey } from '@mui/joy/Sheet'; @@ -30,8 +26,6 @@ describe('', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoySheet', classes }); - describe('prop: variant', () => { it('plain by default', () => { const { getByTestId } = render(Hello World); diff --git a/packages/mui-joy/src/Sheet/Sheet.tsx b/packages/mui-joy/src/Sheet/Sheet.tsx index 98a857b28375d2..b5ccd6d94489be 100644 --- a/packages/mui-joy/src/Sheet/Sheet.tsx +++ b/packages/mui-joy/src/Sheet/Sheet.tsx @@ -7,11 +7,11 @@ import { OverridableComponent } from '@mui/types'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { getPath } from '@mui/system'; import { useThemeProps } from '../styles'; +import { applySoftInversion, applySolidInversion } from '../colorInversion'; import styled from '../styles/styled'; import { resolveSxValue } from '../styles/styleUtils'; import { getSheetUtilityClass } from './sheetClasses'; import { SheetProps, SheetOwnerState, SheetTypeMap } from './SheetProps'; -import { ColorInversionProvider, useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; const useUtilityClasses = (ownerState: SheetOwnerState) => { @@ -72,11 +72,17 @@ export const SheetRoot = styled('div', { } as const, { ...theme.typography['body-md'], + ...(ownerState.variant === 'solid' && + ownerState.color && + ownerState.invertedColors && + applySolidInversion(ownerState.color)(theme)), + ...(ownerState.variant === 'soft' && + ownerState.color && + ownerState.invertedColors && + applySoftInversion(ownerState.color)(theme)), + ...theme.variants[ownerState.variant!]?.[ownerState.color!], ...variantStyle, }, - ownerState.color !== 'context' && - ownerState.invertedColors && - theme.colorInversion[ownerState.variant!]?.[ownerState.color!], ]; }); /** @@ -97,7 +103,7 @@ const Sheet = React.forwardRef(function Sheet(inProps, ref) { const { className, - color: colorProp = 'neutral', + color = 'neutral', component = 'div', variant = 'plain', invertedColors = false, @@ -105,8 +111,6 @@ const Sheet = React.forwardRef(function Sheet(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, @@ -127,12 +131,7 @@ const Sheet = React.forwardRef(function Sheet(inProps, ref) { ownerState, }); - const result = ; - - if (invertedColors) { - return {result}; - } - return result; + return ; }) as OverridableComponent; Sheet.propTypes /* remove-proptypes */ = { diff --git a/packages/mui-joy/src/Slider/Slider.test.tsx b/packages/mui-joy/src/Slider/Slider.test.tsx index 8e2a08410db6e6..378a67d41e163b 100644 --- a/packages/mui-joy/src/Slider/Slider.test.tsx +++ b/packages/mui-joy/src/Slider/Slider.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer } from '@mui-internal/test-utils'; import Slider, { sliderClasses as classes } from '@mui/joy/Slider'; import { ThemeProvider } from '@mui/joy/styles'; @@ -40,8 +36,6 @@ describe('', () => { }), ); - describeJoyColorInversion(, { muiName: 'JoySlider', classes }); - it('should render the rail as the first child of the Slider', () => { const { container } = render(); diff --git a/packages/mui-joy/src/Slider/Slider.tsx b/packages/mui-joy/src/Slider/Slider.tsx index 7bd87f18566460..a6e99e0a156353 100644 --- a/packages/mui-joy/src/Slider/Slider.tsx +++ b/packages/mui-joy/src/Slider/Slider.tsx @@ -10,7 +10,6 @@ import { OverridableComponent } from '@mui/types'; import { useSlider, valueToPercent } from '@mui/base/useSlider'; import { isHostComponent } from '@mui/base/utils'; import { useThemeProps, styled, Theme } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; import useSlot from '../utils/useSlot'; import sliderClasses, { getSliderUtilityClass } from './sliderClasses'; import { SliderTypeMap, SliderOwnerState } from './SliderProps'; @@ -232,9 +231,7 @@ const SliderThumb = styled('span', { ...theme.focus.default, outlineOffset: 0, outlineWidth: 'max(4px, var(--Slider-thumbSize) / 3.6)', - ...(ownerState.color !== 'context' && { - outlineColor: `rgba(${theme.vars.palette?.[ownerState.color!]?.mainChannel} / 0.32)`, - }), + outlineColor: `rgba(${theme.vars.palette?.[ownerState.color!]?.mainChannel} / 0.32)`, }, ...(ownerState.orientation === 'horizontal' && { top: '50%', @@ -430,7 +427,7 @@ const Slider = React.forwardRef(function Slider(inProps, ref) { valueLabelDisplay = 'off', valueLabelFormat = Identity, isRtl = false, - color: colorProp = 'primary', + color = 'primary', size = 'md', variant = 'solid', component, @@ -438,8 +435,6 @@ const Slider = React.forwardRef(function Slider(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion('solid'); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, diff --git a/packages/mui-joy/src/SvgIcon/SvgIcon.tsx b/packages/mui-joy/src/SvgIcon/SvgIcon.tsx index d93f587099e265..bd8921c0a9b3d8 100644 --- a/packages/mui-joy/src/SvgIcon/SvgIcon.tsx +++ b/packages/mui-joy/src/SvgIcon/SvgIcon.tsx @@ -60,7 +60,6 @@ const SvgIconRoot = styled('svg', { color: 'inherit', }), ...(ownerState.color !== 'inherit' && - ownerState.color !== 'context' && theme.vars.palette[ownerState.color!] && { color: `rgba(${theme.vars.palette[ownerState.color!]?.mainChannel} / 1)`, }), diff --git a/packages/mui-joy/src/Switch/Switch.test.tsx b/packages/mui-joy/src/Switch/Switch.test.tsx index 014c0baf54c62c..7516d4b6fbe5e7 100644 --- a/packages/mui-joy/src/Switch/Switch.test.tsx +++ b/packages/mui-joy/src/Switch/Switch.test.tsx @@ -2,7 +2,6 @@ import * as React from 'react'; import { expect } from 'chai'; import { describeConformance, - describeJoyColorInversion, act, createRenderer, fireEvent, @@ -40,8 +39,6 @@ describe('', () => { skip: ['componentProp', 'componentsProp', 'classesRoot'], })); - describeJoyColorInversion(, { muiName: 'JoySwitch', classes }); - it('should pass `slotProps` down to slots', () => { const { container } = render( ', () => { }, })); - describeJoyColorInversion(, { - muiName: 'JoyTab', - classes, - wrapper: (node) => {node}, - }); - it('prop: variant', () => { render( diff --git a/packages/mui-joy/src/Tab/Tab.tsx b/packages/mui-joy/src/Tab/Tab.tsx index c9d8342d03b361..90ef14d4e92b59 100644 --- a/packages/mui-joy/src/Tab/Tab.tsx +++ b/packages/mui-joy/src/Tab/Tab.tsx @@ -8,7 +8,6 @@ import { useTab } from '@mui/base/useTab'; import { StyledListItemButton } from '../ListItemButton/ListItemButton'; import { useThemeProps } from '../styles'; import styled from '../styles/styled'; -import { useColorInversion } from '../styles/ColorInversion'; import { getTabUtilityClass } from './tabClasses'; import { TabOwnerState, TabTypeMap } from './TabProps'; import RowListContext from '../List/RowListContext'; @@ -142,7 +141,7 @@ const Tab = React.forwardRef(function Tab(inProps, ref) { component = 'button', orientation = 'horizontal', variant = 'plain', - color: colorProp = 'neutral', + color = 'neutral', disableIndicator = false, indicatorPlacement = row ? 'bottom' : 'right', indicatorInset = false, @@ -150,8 +149,6 @@ const Tab = React.forwardRef(function Tab(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const tabRef = React.useRef(); const handleRef = useForkRef(tabRef, ref) as React.RefCallback; diff --git a/packages/mui-joy/src/TabList/TabList.test.tsx b/packages/mui-joy/src/TabList/TabList.test.tsx index 5dbd76080dcacd..5934491c8255f6 100644 --- a/packages/mui-joy/src/TabList/TabList.test.tsx +++ b/packages/mui-joy/src/TabList/TabList.test.tsx @@ -1,11 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - screen, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer, screen } from '@mui-internal/test-utils'; import { TabsProps } from '@mui/base/Tabs'; import { useTabs, TabsProvider as BaseTabsProvider } from '@mui/base/useTabs'; import { ThemeProvider } from '@mui/joy/styles'; @@ -38,12 +33,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { - muiName: 'JoyTabList', - classes, - wrapper: (node) => {node}, - }); - describe('size', () => { it('uses size from Tabs', () => { render( diff --git a/packages/mui-joy/src/TabList/TabList.tsx b/packages/mui-joy/src/TabList/TabList.tsx index b3617bc2ff7b89..d9987a695d5498 100644 --- a/packages/mui-joy/src/TabList/TabList.tsx +++ b/packages/mui-joy/src/TabList/TabList.tsx @@ -7,7 +7,6 @@ import { OverridableComponent } from '@mui/types'; import { useTabsList, TabsListProvider } from '@mui/base/useTabsList'; import { useThemeProps } from '../styles'; import styled from '../styles/styled'; -import { useColorInversion } from '../styles/ColorInversion'; import { StyledList } from '../List/List'; import ListProvider, { scopedVariables } from '../List/ListProvider'; import SizeTabsContext from '../Tabs/SizeTabsContext'; @@ -114,7 +113,7 @@ const TabList = React.forwardRef(function TabList(inProps, ref) { component = 'div', children, variant = 'plain', - color: colorProp = 'neutral', + color = 'neutral', size: sizeProp, disableUnderline = false, underlinePlacement = orientation === 'horizontal' ? 'bottom' : 'right', @@ -124,8 +123,6 @@ const TabList = React.forwardRef(function TabList(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const size = sizeProp ?? tabsSize; diff --git a/packages/mui-joy/src/TabPanel/TabPanel.tsx b/packages/mui-joy/src/TabPanel/TabPanel.tsx index 8adf06e5966d37..891fb3c9842034 100644 --- a/packages/mui-joy/src/TabPanel/TabPanel.tsx +++ b/packages/mui-joy/src/TabPanel/TabPanel.tsx @@ -7,7 +7,7 @@ import { OverridableComponent } from '@mui/types'; import { useTabPanel } from '@mui/base/useTabPanel'; import { useTabsContext } from '@mui/base/Tabs'; import { styled, useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; + import SizeTabsContext from '../Tabs/SizeTabsContext'; import { getTabPanelUtilityClass } from './tabPanelClasses'; import { TabPanelOwnerState, TabPanelTypeMap } from './TabPanelProps'; @@ -66,7 +66,7 @@ const TabPanel = React.forwardRef(function TabPanel(inProps, ref) { children, value = 0, component, - color: colorProp = 'neutral', + color = 'neutral', variant = 'plain', size: sizeProp, slots = {}, @@ -78,8 +78,6 @@ const TabPanel = React.forwardRef(function TabPanel(inProps, ref) { const { hidden, getRootProps } = useTabPanel({ ...props, value }); const size = sizeProp ?? tabsSize; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, diff --git a/packages/mui-joy/src/Table/Table.test.tsx b/packages/mui-joy/src/Table/Table.test.tsx index 843f73cf9bdf7b..37909e8f400ba6 100644 --- a/packages/mui-joy/src/Table/Table.test.tsx +++ b/packages/mui-joy/src/Table/Table.test.tsx @@ -1,10 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - createRenderer, - describeConformance, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { createRenderer, describeConformance } from '@mui-internal/test-utils'; import { unstable_capitalize as capitalize } from '@mui/utils'; import { ThemeProvider } from '@mui/joy/styles'; import Table, { tableClasses as classes } from '@mui/joy/Table'; @@ -39,8 +35,6 @@ describe('', () => { }, })); - describeJoyColorInversion(
, { muiName: 'JoyTable', classes }); - describe('prop: variant', () => { it('plain by default', () => { const { getByRole } = render(
); diff --git a/packages/mui-joy/src/Table/Table.tsx b/packages/mui-joy/src/Table/Table.tsx index f194b6536960c8..dc842922b93803 100644 --- a/packages/mui-joy/src/Table/Table.tsx +++ b/packages/mui-joy/src/Table/Table.tsx @@ -6,7 +6,7 @@ import { unstable_capitalize as capitalize } from '@mui/utils'; import { unstable_composeClasses as composeClasses } from '@mui/base'; import { OverridableComponent } from '@mui/types'; import { useThemeProps } from '../styles'; -import { useColorInversion } from '../styles/ColorInversion'; + import styled from '../styles/styled'; import { getTableUtilityClass } from './tableClasses'; import { TableProps, TableOwnerState, TableTypeMap } from './TableProps'; @@ -305,7 +305,7 @@ const Table = React.forwardRef(function Table(inProps, ref) { noWrap = false, size = 'md', variant = 'plain', - color: colorProp = 'neutral', + color = 'neutral', stripe, stickyHeader = false, stickyFooter = false, @@ -313,8 +313,6 @@ const Table = React.forwardRef(function Table(inProps, ref) { slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const ownerState = { ...props, diff --git a/packages/mui-joy/src/Tabs/Tabs.test.tsx b/packages/mui-joy/src/Tabs/Tabs.test.tsx index 56052e1b00d5bd..b7951b0200d184 100644 --- a/packages/mui-joy/src/Tabs/Tabs.test.tsx +++ b/packages/mui-joy/src/Tabs/Tabs.test.tsx @@ -1,11 +1,6 @@ import * as React from 'react'; import { expect } from 'chai'; -import { - describeConformance, - createRenderer, - screen, - describeJoyColorInversion, -} from '@mui-internal/test-utils'; +import { describeConformance, createRenderer, screen } from '@mui-internal/test-utils'; import { ThemeProvider } from '@mui/joy/styles'; import Tabs, { tabsClasses as classes } from '@mui/joy/Tabs'; import SizeTabsContext from './SizeTabsContext'; @@ -30,8 +25,6 @@ describe('Joy ', () => { }, })); - describeJoyColorInversion(, { muiName: 'JoyTabs', classes }); - it('prop: variant', () => { render(); expect(screen.getByLabelText('Tabs')).to.have.class(classes.variantOutlined); diff --git a/packages/mui-joy/src/Tabs/Tabs.tsx b/packages/mui-joy/src/Tabs/Tabs.tsx index dcda739ef89c80..a4e7eecb798b9d 100644 --- a/packages/mui-joy/src/Tabs/Tabs.tsx +++ b/packages/mui-joy/src/Tabs/Tabs.tsx @@ -8,7 +8,7 @@ import { useTabs, TabsProvider } from '@mui/base/useTabs'; import { getPath } from '@mui/system'; import { styled, useThemeProps } from '../styles'; import { resolveSxValue } from '../styles/styleUtils'; -import { useColorInversion } from '../styles/ColorInversion'; + import SizeTabsContext from './SizeTabsContext'; import { getTabsUtilityClass } from './tabsClasses'; import { TabsOwnerState, TabsTypeMap } from './TabsProps'; @@ -104,14 +104,12 @@ const Tabs = React.forwardRef(function Tabs(inProps, ref) { onChange, selectionFollowsFocus, variant = 'plain', - color: colorProp = 'neutral', + color = 'neutral', size = 'md', slots = {}, slotProps = {}, ...other } = props; - const { getColor } = useColorInversion(variant); - const color = getColor(inProps.color, colorProp); const defaultValue = defaultValueProp || (valueProp === undefined ? 0 : undefined); const { contextValue } = useTabs({ ...props, orientation, defaultValue }); diff --git a/packages/mui-joy/src/Textarea/Textarea.test.tsx b/packages/mui-joy/src/Textarea/Textarea.test.tsx index 9df58d9be64752..5edf08bfee175e 100644 --- a/packages/mui-joy/src/Textarea/Textarea.test.tsx +++ b/packages/mui-joy/src/Textarea/Textarea.test.tsx @@ -3,7 +3,6 @@ import { expect } from 'chai'; import { spy } from 'sinon'; import { describeConformance, - describeJoyColorInversion, createRenderer, screen, act, @@ -33,8 +32,6 @@ describe('Joy