Skip to main content

Theming

OnlyNative UI ships with Material Design 3 out of the box, but the theme engine is design-system agnostic — you can customize MD3, generate branded themes from a seed color, or build an entirely custom design system.

Material Design 3 (default)

ThemeProvider

Wrap your app with ThemeProvider to supply the MD3 theme:

import { ThemeProvider } from '@onlynative/core'

export default function App() {
return (
<ThemeProvider>
{/* All components use the default light theme */}
</ThemeProvider>
)
}

Dark mode

Pass the built-in dark theme:

import { ThemeProvider, darkTheme } from '@onlynative/core'

<ThemeProvider theme={darkTheme}>
{/* Dark mode */}
</ThemeProvider>

Switch between light and dark based on system preference:

import { useColorScheme } from 'react-native'
import { ThemeProvider, lightTheme, darkTheme } from '@onlynative/core'

export default function App() {
const colorScheme = useColorScheme()
const theme = colorScheme === 'dark' ? darkTheme : lightTheme

return (
<ThemeProvider theme={theme}>
{/* Follows system theme */}
</ThemeProvider>
)
}

Override specific tokens

Spread the base theme and override individual tokens:

import { lightTheme } from '@onlynative/core'
import type { Theme } from '@onlynative/core'

const brandTheme: Theme = {
...lightTheme,
colors: {
...lightTheme.colors,
primary: '#006A6A',
onPrimary: '#FFFFFF',
},
}

<ThemeProvider theme={brandTheme}>
{/* Components use your custom primary color */}
</ThemeProvider>

Generate a theme from a seed color

createMaterialTheme generates a complete MD3 light and dark theme from a single hex color using Google's HCT color space. All 69 color roles are derived automatically:

import { createMaterialTheme } from '@onlynative/core/create-theme'
import { ThemeProvider } from '@onlynative/core'

const { lightTheme, darkTheme } = createMaterialTheme('#006A6A')

<ThemeProvider theme={lightTheme}>
{/* Full MD3 palette generated from #006A6A */}
</ThemeProvider>

You can also pass a custom font family:

const { lightTheme, darkTheme } = createMaterialTheme('#006A6A', {
fontFamily: 'Inter',
})

This is the easiest way to create a branded theme — pick your brand color and the entire palette is generated for you. See the Fonts guide for full details on loading and using custom fonts.

Access theme values

Use the useTheme hook in any component:

import { useTheme } from '@onlynative/core'

function MyComponent() {
const theme = useTheme()
return (
<View style={{ backgroundColor: theme.colors.surface }}>
<Text style={{ color: theme.colors.onSurface }}>Hello</Text>
</View>
)
}

Material preset

All MD3 values are also available as a grouped object:

import { material } from '@onlynative/core'

material.lightTheme
material.darkTheme
material.defaultTopAppBarTokens

Custom design systems

The theme engine supports any design system, not just MD3. Use BaseTheme, defineTheme, and ThemeProvider to build your own.

BaseTheme

All themes extend BaseTheme:

interface BaseTheme {
colors: Record<string, string>
typography: Record<string, TextStyle>
shape: Shape
spacing: Spacing
stateLayer: StateLayer
elevation: Elevation
motion: Motion
}

Define a custom theme

Use defineTheme for type-safe theme creation:

import { defineTheme } from '@onlynative/core'
import type { BaseTheme } from '@onlynative/core'

interface BrandTheme extends BaseTheme {
colors: {
brand: string
brandMuted: string
background: string
surface: string
text: string
textSecondary: string
border: string
error: string
success: string
[key: string]: string
}
typography: {
heading: TextStyle
subheading: TextStyle
body: TextStyle
caption: TextStyle
[key: string]: TextStyle
}
}

const brandTheme = defineTheme<BrandTheme>({
colors: {
brand: '#FF6B00',
brandMuted: '#FFF3E0',
background: '#FFFFFF',
surface: '#F5F5F5',
text: '#1A1A1A',
textSecondary: '#757575',
border: '#E0E0E0',
error: '#D32F2F',
success: '#388E3C',
},
typography: {
heading: { fontSize: 24, fontWeight: '700', lineHeight: 32 },
subheading: { fontSize: 18, fontWeight: '600', lineHeight: 24 },
body: { fontSize: 16, fontWeight: '400', lineHeight: 22 },
caption: { fontSize: 12, fontWeight: '400', lineHeight: 16 },
},
shape: { none: 0, extraSmall: 4, small: 8, medium: 12, large: 16, extraLarge: 28, full: 9999 },
spacing: { xs: 4, sm: 8, md: 16, lg: 24, xl: 32 },
stateLayer: { pressed: 0.12, focused: 0.12, hovered: 0.08, disabled: 0.38 },
elevation: {
level0: {},
level1: { shadowColor: '#000', shadowOffset: { width: 0, height: 1 }, shadowOpacity: 0.1, shadowRadius: 3, elevation: 1 },
level2: { shadowColor: '#000', shadowOffset: { width: 0, height: 2 }, shadowOpacity: 0.15, shadowRadius: 6, elevation: 3 },
level3: { shadowColor: '#000', shadowOffset: { width: 0, height: 4 }, shadowOpacity: 0.2, shadowRadius: 8, elevation: 6 },
},
motion: {
duration: { short: 100, medium: 300, long: 500 },
easing: { standard: 'ease-in-out', accelerate: 'ease-in', decelerate: 'ease-out' },
},
})

ThemeProvider

Use ThemeProvider for custom design systems:

import { ThemeProvider } from '@onlynative/core'

<ThemeProvider theme={brandTheme}>
{/* Your app */}
</ThemeProvider>

Access custom theme values

Pass your theme type as a generic to useTheme:

import { useTheme } from '@onlynative/core'

function MyComponent() {
const theme = useTheme<BrandTheme>()
// theme.colors.brand is typed as string
return (
<View style={{ backgroundColor: theme.colors.background }}>
<Text style={[theme.typography.body, { color: theme.colors.text }]}>
Hello
</Text>
</View>
)
}

Theme structure reference

Token groupDescription
colorsDesign-system specific color roles (Record<string, string>)
typographyType scale variants (Record<string, TextStyle>)
shapeCorner radius tokens (none through full)
spacingSpacing scale (xs, sm, md, lg, xl)
elevationShadow levels 0-3
stateLayerOpacity values for pressed, focused, hovered, disabled
motionDuration and easing tokens

Type hierarchy

  • BaseTheme — Generic base. Any design system extends this.
  • Theme — MD3 theme. Extends BaseTheme with 69 color roles, 15 typography variants, optional topAppBar tokens. MaterialTheme is an identical alias — use it to disambiguate in multi-design-system codebases.

Summary

GoalAPI
Use MD3 defaults<ThemeProvider>
Dark mode<ThemeProvider theme={darkTheme}>
Override a few MD3 colorsSpread lightTheme and override
Branded MD3 theme from one colorimport { createMaterialTheme } from '@onlynative/core/create-theme'
Custom fontcreateMaterialTheme('#color', { fontFamily: 'Inter' }) — see Fonts
Fully custom design systemdefineTheme + <ThemeProvider> + useTheme<T>()