Guides

Theming

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 component
  • theme: 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.
OptionTypeDescription
colors.themeColor or RampColor used for all "theme" keys in the theme
colors.dangerColor or RampColor used for all "danger" keys in the theme
colors.successColor or RampColor used for all "success" keys in the theme
colors.warningColor or RampColor used for all "warning" keys in the theme
colors.blackstringColor used for black in the theme
colors.grayRampColor used for all "gray" values in the theme
colors.whitestringColor used for black in the theme
overridesObject<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:

    1. Each ramp contains 10 color values, each spaced roughly evenly from one another.
    2. Each ramp's inflection point (where text color changes from black to white to meet accessible contrast ratio standards) occurs at the60 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 the inflection 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 #

VariableValue
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 #

VariableValue
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_10.1875em

breakpoint #

VariableValue
breakpoint_narrow512px
breakpoint_medium768px
breakpoint_wide1024px

depth #

VariableValue
boxShadow_10 1px 2px 0 rgba(0,0,0,0.2), 0 2px 4px 0 rgba(0,0,0,0.2)
boxShadow_20 2px 4px 0 rgba(0,0,0,0.2), 0 4px 8px 0 rgba(0,0,0,0.2)
boxShadow_30 4px 8px 0 rgba(0,0,0,0.2), 0 8px 16px 0 rgba(0,0,0,0.2)
boxShadow_40 8px 16px 0 rgba(0,0,0,0.2), 0 20px 16px -8px rgba(0,0,0,0.2)
boxShadow_50 16px 24px 0 rgba(0,0,0,0.2), 0 32px 24px -16px rgba(0,0,0,0.2)
boxShadow_focusInner#ffffff
zIndex_100100
zIndex_16001600
zIndex_200200
zIndex_400400
zIndex_800800

heading #

VariableValue
h1_color#333840
h1_fontSize2.125em
h1_fontWeight800
h2_color#58606e
h2_fontSize1.75em
h2_fontWeight700
h3_color#58606e
h3_fontSize1.375em
h3_fontWeight700
h4_color#58606e
h4_fontSize1.125em
h4_fontWeight700
h5_color#333840
h5_fontSize0.875em
h5_fontWeight700
h6_color#58606e
h6_fontSize0.875em
h6_fontWeight400

icon #

VariableValue
icon_color#58606e
icon_color_theme#3272d9
icon_color_danger#de1b1b
icon_color_success#2a854e
icon_color_warning#ad5f00

input #

VariableValue
input_backgroundColor#ffffff
input_backgroundColor_disabled#ebeff5
input_color_placeholder#8e99ab

panel #

VariableValue
panel_backgroundColor#ffffff
panel_backgroundColor_inverted#434a54
panel_borderColor#ebeff5
panel_borderColor_inverted#434a54

size #

VariableValue
size_small1.5em
size_medium2em
size_large2.5em
size_jumbo3.25em

space #

VariableValue
space_inline_xxs0.125em
space_inline_xs0.25em
space_inline_sm0.5em
space_inline_md1em
space_inline_lg1.5em
space_inline_xl2em
space_inline_xxl4em
space_inset_sm0.5em
space_inset_md1em
space_inset_lg1.5em
space_stack_xxs0.125em
space_stack_xs0.25em
space_stack_sm0.5em
space_stack_md1em
space_stack_lg1.5em
space_stack_xl2em
space_stack_xxl4em

typography #

VariableValue
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_base16px
fontSize_mouse0.6875em
fontSize_prose1em
fontSize_ui0.875em
fontWeight_regular400
fontWeight_semiBold600
fontWeight_bold700
fontWeight_extraBold800
lineHeight1.25
lineHeight_heading1.25
lineHeight_heading_small1.5
lineHeight_prose1.5

well #

VariableValue
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 #

VariableValue
directionltr

Colors #

VariableValue
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