Theming is a core concept in Mineral UI. Themes provide a consistent look and feel across pages with varied functionality. Mineral UI makes it simple to implement and maintain theming across your app.
Common Scenarios #
Theme your entire app #
Wrap your app in a ThemeProvider in order for styles to be properly applied. The ThemeProvider provides the theme to the tree of Mineral UI components, and any other Emotion components contained within.
import React from 'react';
import { render } from 'react-dom';
import Button from 'mineral-ui/Button';
import { ThemeProvider } from 'mineral-ui/themes';
function App() {
return (
<ThemeProvider>
<Button>
Hello World
</Button>
</ThemeProvider>
);
}
render(<App />, document.getElementById('app'));
Theme a portion of your app #
ThemeProviders may be nested in order to apply a custom theme to a portion of
your app. Nested ThemeProviders shallowly merge their theme with the parent
theme. The theme itself is a simple shallow object of variables that are shared
across components.
See the default mineralTheme below for an example.
<ThemeProvider>
<ThemeProvider theme={{ color_themePrimary: 'darkgray' }}>
<nav>Navigation<nav>
</ThemeProvider>
<main>The main part of your app</main>
</ThemeProvider>
Theme a component #
Each component has a set of component-level theme variables that may be overridden to adjust styles on a per component basis. These are documented on the individual component pages, e.g. Button theme variables.
To theme a component, use themed as shown below. It is effectively the same as wrapping your component with a ThemeProvider.
import { themed } from 'mineral-ui/themes';
const MyButton = themed(Button)({
Button_backgroundColor: 'tomato'
});
Create your own theme #
Use createTheme in order to create a custom theme. Once created, this theme can be applied using a ThemeProvider.
Use a custom color palette #
Each theme variable with a color value derives that value from a color ramp in
Mineral UI's palette. Those colors can be derived
from a different palette color (e.g. "danger" variables use 'bronze'
instead
of 'red'
, below) or a custom color ramp
can be used (e.g. myAppColor
is used for "theme" variables, below).
import React from 'react';
import { render } from 'react-dom';
import Button from 'mineral-ui/Button';
import { createTheme, ThemeProvider } from 'mineral-ui/themes';
const myAppColor = {
[10]: '#faf0f4',
[20]: '#fad4e4',
[30]: '#fab4d1',
[40]: '#f78bb8',
[50]: '#ed5393',
[60]: '#d6246e',
[70]: '#b01355',
[80]: '#8a1244',
[90]: '#611535',
[100]: '#421527'
}
const myTheme = createTheme({
colors: {
theme: myAppColor,
danger: 'bronze',
warning: 'dusk',
success: 'teal'
}
});
function App() {
return (
<ThemeProvider theme={myTheme}>
<Button>
Hello World
</Button>
</ThemeProvider>
);
}
render(<App />, document.getElementById('app'));
API #
<ThemeProvider theme />
#
This component takes a theme property and provides it to the tree of components contained within. When nested, child themes will be shallowly merged with the parent theme.
See the previous examples and the ThemeProvider page for more details.
themed(component)(theme)
#
This function is useful when you need to override component-level theme variables. It is effectively the same as wrapping a ThemeProvider around a single component.
Parameters
component
: A React componenttheme
: A shallow object of theme variables or a function that accepts props and returns a shallow object of theme variables
Returns
- The original React component wrapped in a ThemeProvider with the merged theme provided.
Example
import { themed } from 'mineral-ui/themes';
// The \`theme\` prop is the theme available from the nearest ThemeProvider in
// the component tree
const MyButton = themed(Button)(({ theme }) => ({
Button_backgroundColor: theme.color_theme_10
}));
createTheme(options)
#
This function is useful when you want to create a new theme that uses a different color scheme or otherwise overrides default values.
Parameters
options
: Optional. An object with the following shape. All properties are optional.
Option | Type | Description |
---|---|---|
colors.theme | Color or Ramp | Color used for all "theme" keys in the theme |
colors.danger | Color or Ramp | Color used for all "danger" keys in the theme |
colors.success | Color or Ramp | Color used for all "success" keys in the theme |
colors.warning | Color or Ramp | Color used for all "warning" keys in the theme |
colors.black | string | Color used for black in the theme |
colors.gray | Ramp | Color used for all "gray" values in the theme |
colors.white | string | Color used for black in the theme |
overrides | Object | <key, value> pairs of specific theme overrides |
Types
- Color: String matching a color ramp name in the Mineral UI palette
Ramp: Object matching this shape:
const myRamp = { [10]: '<color>', [20]: '<color>', [30]: '<color>', [40]: '<color>', [50]: '<color>', [60]: '<color>', [70]: '<color>', [80]: '<color>', [90]: '<color>', [100]: '<color>', inflection: 70 };
Inflection Point
Note the optional
inflection
property. Mineral UI's color ramps are designed to provide an accessible level of contrast, which they accomplish through two constraints:- Each ramp contains 10 color values, each spaced roughly evenly from one another.
- Each ramp's inflection point (where text color changes from black to white to meet accessible contrast ratio standards) occurs at the
60
value.
Your custom ramp must meet the first constraint of 10 values. But the inflection point, while advised to be
60
, may be different. You may define it with theinflection
property to ensure adequate contrast.
Returns
- A new theme object
Example
import { createTheme } from 'mineral-ui/themes';
const myTheme = createTheme({
colors: { theme: 'dusk' },
overrides: { fontFamily: 'Comic Sans MS' }
});
withTheme(component)
#
A higher order component (HOC) that enables theme access inside any component.
Parameters
component
: A React component
Returns
- The original React component with theme provided as a prop.
Example
import { withTheme } from 'emotion-theming';
const Direction = withTheme(({ theme }) => {
return <span>{theme.direction}</span>;
});
Theme Structure #
Mineral UI themes are shallow objects of variables that are shared across
components. The values in the table below are from the default mineralTheme.
Note the naming convention: [target]_[property]_[variation]_[state]
.
background #
Variable | Value |
---|---|
backgroundColor | #ffffff |
backgroundColor_active | #ebeff5 |
backgroundColor_disabled | #ebeff5 |
backgroundColor_focus | #ffffff |
backgroundColor_hover | #f5f7fa |
backgroundColor_theme_selected | #f0f5fc |
backgroundColor_theme_selectedActive | #accbfc |
backgroundColor_theme_selectedHover | #cfe0fc |
backgroundColor_themePrimary | #3272d9 |
backgroundColor_themePrimary_active | #1d5bbf |
backgroundColor_themePrimary_focus | #3272d9 |
backgroundColor_themePrimary_hover | #5691f0 |
backgroundColor_danger_active | #fad4d4 |
backgroundColor_danger_focus | #faf0f0 |
backgroundColor_danger_hover | #faf0f0 |
backgroundColor_dangerPrimary | #de1b1b |
backgroundColor_dangerPrimary_active | #b80d0d |
backgroundColor_dangerPrimary_focus | #de1b1b |
backgroundColor_dangerPrimary_hover | #f55353 |
backgroundColor_success_active | #abedc5 |
backgroundColor_success_focus | #e1faeb |
backgroundColor_success_hover | #e1faeb |
backgroundColor_successPrimary | #2a854e |
backgroundColor_successPrimary_active | #20693d |
backgroundColor_successPrimary_focus | #2a854e |
backgroundColor_successPrimary_hover | #3ba164 |
backgroundColor_warning_active | #fad8af |
backgroundColor_warning_focus | #fcf2e6 |
backgroundColor_warning_hover | #fcf2e6 |
backgroundColor_warningPrimary | #ad5f00 |
backgroundColor_warningPrimary_active | #8a4d03 |
backgroundColor_warningPrimary_focus | #ad5f00 |
backgroundColor_warningPrimary_hover | #cf7911 |
border #
Variable | Value |
---|---|
borderColor | #c8d1e0 |
borderColor_theme | #3272d9 |
borderColor_theme_active | #1d5bbf |
borderColor_theme_focus | #1d5bbf |
borderColor_theme_hover | #5691f0 |
borderColor_danger | #de1b1b |
borderColor_danger_active | #b80d0d |
borderColor_danger_focus | #b80d0d |
borderColor_danger_hover | #f55353 |
borderColor_success | #2a854e |
borderColor_success_active | #20693d |
borderColor_success_focus | #20693d |
borderColor_success_hover | #3ba164 |
borderColor_warning | #ad5f00 |
borderColor_warning_active | #8a4d03 |
borderColor_warning_focus | #8a4d03 |
borderColor_warning_hover | #cf7911 |
borderRadius_1 | 0.1875em |
breakpoint #
Variable | Value |
---|---|
breakpoint_narrow | 512px |
breakpoint_medium | 768px |
breakpoint_wide | 1024px |
depth #
Variable | Value |
---|---|
boxShadow_1 | 0 1px 2px 0 rgba(0,0,0,0.2), 0 2px 4px 0 rgba(0,0,0,0.2) |
boxShadow_2 | 0 2px 4px 0 rgba(0,0,0,0.2), 0 4px 8px 0 rgba(0,0,0,0.2) |
boxShadow_3 | 0 4px 8px 0 rgba(0,0,0,0.2), 0 8px 16px 0 rgba(0,0,0,0.2) |
boxShadow_4 | 0 8px 16px 0 rgba(0,0,0,0.2), 0 20px 16px -8px rgba(0,0,0,0.2) |
boxShadow_5 | 0 16px 24px 0 rgba(0,0,0,0.2), 0 32px 24px -16px rgba(0,0,0,0.2) |
boxShadow_focusInner | #ffffff |
zIndex_100 | 100 |
zIndex_1600 | 1600 |
zIndex_200 | 200 |
zIndex_400 | 400 |
zIndex_800 | 800 |
heading #
Variable | Value |
---|---|
h1_color | #333840 |
h1_fontSize | 2.125em |
h1_fontWeight | 800 |
h2_color | #58606e |
h2_fontSize | 1.75em |
h2_fontWeight | 700 |
h3_color | #58606e |
h3_fontSize | 1.375em |
h3_fontWeight | 700 |
h4_color | #58606e |
h4_fontSize | 1.125em |
h4_fontWeight | 700 |
h5_color | #333840 |
h5_fontSize | 0.875em |
h5_fontWeight | 700 |
h6_color | #58606e |
h6_fontSize | 0.875em |
h6_fontWeight | 400 |
icon #
Variable | Value |
---|---|
icon_color | #58606e |
icon_color_theme | #3272d9 |
icon_color_danger | #de1b1b |
icon_color_success | #2a854e |
icon_color_warning | #ad5f00 |
input #
Variable | Value |
---|---|
input_backgroundColor | #ffffff |
input_backgroundColor_disabled | #ebeff5 |
input_color_placeholder | #8e99ab |
panel #
Variable | Value |
---|---|
panel_backgroundColor | #ffffff |
panel_backgroundColor_inverted | #434a54 |
panel_borderColor | #ebeff5 |
panel_borderColor_inverted | #434a54 |
size #
Variable | Value |
---|---|
size_small | 1.5em |
size_medium | 2em |
size_large | 2.5em |
size_jumbo | 3.25em |
space #
Variable | Value |
---|---|
space_inline_xxs | 0.125em |
space_inline_xs | 0.25em |
space_inline_sm | 0.5em |
space_inline_md | 1em |
space_inline_lg | 1.5em |
space_inline_xl | 2em |
space_inline_xxl | 4em |
space_inset_sm | 0.5em |
space_inset_md | 1em |
space_inset_lg | 1.5em |
space_stack_xxs | 0.125em |
space_stack_xs | 0.25em |
space_stack_sm | 0.5em |
space_stack_md | 1em |
space_stack_lg | 1.5em |
space_stack_xl | 2em |
space_stack_xxl | 4em |
typography #
Variable | Value |
---|---|
color | #333840 |
color_inverted | #ffffff |
color_disabled | #afbacc |
color_mouse | #58606e |
color_readOnly | #58606e |
color_required | #de1b1b |
color_theme | #3272d9 |
color_themePrimary | #ffffff |
color_theme_active | #1d5bbf |
color_theme_focus | #3272d9 |
color_theme_hover | #5691f0 |
color_danger | #de1b1b |
color_dangerPrimary | #ffffff |
color_danger_active | #b80d0d |
color_danger_focus | #de1b1b |
color_danger_hover | #f55353 |
color_success | #2a854e |
color_successPrimary | #ffffff |
color_success_active | #20693d |
color_success_focus | #2a854e |
color_success_hover | #3ba164 |
color_warning | #ad5f00 |
color_warningPrimary | #ffffff |
color_warning_active | #8a4d03 |
color_warning_focus | #ad5f00 |
color_warning_hover | #cf7911 |
fontFamily | "Open Sans" |
fontFamily_system | -apple-system, system-ui, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol" |
fontFamily_monospace | "SF Mono", "Droid Sans Mono", "Source Code Pro", Monaco, Consolas, "Courier New", Courier, monospace |
fontSize_base | 16px |
fontSize_mouse | 0.6875em |
fontSize_prose | 1em |
fontSize_ui | 0.875em |
fontWeight_regular | 400 |
fontWeight_semiBold | 600 |
fontWeight_bold | 700 |
fontWeight_extraBold | 800 |
lineHeight | 1.25 |
lineHeight_heading | 1.25 |
lineHeight_heading_small | 1.5 |
lineHeight_prose | 1.5 |
well #
Variable | Value |
---|---|
well_backgroundColor | #ebeff5 |
well_backgroundColor_danger | #fad4d4 |
well_backgroundColor_success | #abedc5 |
well_backgroundColor_warning | #fad8af |
well_borderColor_danger | #fa8e8e |
well_borderColor_success | #57c282 |
well_borderColor_warning | #e89c3f |
Other #
Variable | Value |
---|---|
direction | ltr |
Colors #
Variable | Value |
---|---|
color_theme_10 | #f0f5fc |
color_theme_20 | #cfe0fc |
color_theme_30 | #accbfc |
color_theme_40 | #84b1fa |
color_theme_50 | #5691f0 |
color_theme_60 | #3272d9 |
color_theme_70 | #1d5bbf |
color_theme_80 | #114599 |
color_theme_90 | #103570 |
color_theme_100 | #15233b |
color_white | #ffffff |
color_gray_10 | #f5f7fa |
color_gray_20 | #ebeff5 |
color_gray_30 | #dde3ed |
color_gray_40 | #c8d1e0 |
color_gray_50 | #afbacc |
color_gray_60 | #8e99ab |
color_gray_70 | #707a8a |
color_gray_80 | #58606e |
color_gray_90 | #434a54 |
color_gray_100 | #333840 |
color_black | #1d1f24 |