Makepad 2.0 Theme System
Overview
The Makepad 2.0 theme system provides a comprehensive set of design tokens accessed
through theme.* variables in Splash scripts. It delivers consistent styling for
colors, typography, spacing, and widget states across your entire application.
Three built-in themes are available:
mod.themes.dark-- dark desktop theme (default)mod.themes.light-- light desktop thememod.themes.skeleton-- minimal skeleton theme with hardcoded values
Golden rule: Always use theme.* variables instead of hardcoded values for any
color, font size, or spacing in production UIs. This ensures your app automatically
supports theme switching and maintains visual consistency.
Theme Setup in App::run
The theme must be loaded before widgets are loaded. The standard pattern is:
impl App {
fn run(vm: &mut ScriptVm) -> Self {
// Step 1: Load theme definitions (dark, light, skeleton)
crate::makepad_widgets::theme_mod(vm);
// Step 2: Select active theme (MUST come before widgets_mod)
script_eval!(vm, {
mod.theme = mod.themes.light // or mod.themes.dark
});
// Step 3: Load widget definitions (they reference mod.theme)
crate::makepad_widgets::widgets_mod(vm);
// Step 4: Load your app's script_mod
App::from_script_mod(vm, self::script_mod)
}
}
If you skip steps 1-2, the default theme is dark (set inside theme_mod).
The counter example uses the simplified one-liner crate::makepad_widgets::script_mod(vm)
which bundles steps 1-3 with the default dark theme.
How It Works Internally
theme_mod() does the following:
- Calls
makepad_draw::script_mod(vm)to load drawing primitives - Creates the
mod.themesmodule - Loads
theme_desktop_dark,theme_desktop_light, andtheme_desktop_skeleton - Sets
mod.theme = mod.themes.darkas the default
Then widgets_mod() creates mod.prelude.widgets_internal with theme: mod.theme,
making theme.* available in all widget scripts that use mod.prelude.widgets.*.
Theme Global Parameters
Each theme defines tunable global parameters that control the overall feel:
| Parameter | Purpose | Default (dark/light) |
|-----------|---------|---------------------|
| color_contrast | Controls color palette spread | 1.0 |
| color_tint | Tint applied to backgrounds | #0000ff |
| color_tint_amount | How much tint to apply (0-1) | 0.0 |
| space_factor | Base spacing multiplier | 6.0 |
| corner_radius | Base corner radius | 2.5 |
| beveling | Bevel intensity | 0.75 |
| font_size_base | Base font size in px | 10.0 |
| font_size_contrast | Font size step between levels | 2.5 |
Theme Color Variables -- Primary
These are the colors you will use most often in application code:
| Variable | Purpose | Light Appearance | Dark Appearance |
|----------|---------|-----------------|-----------------|
| theme.color_bg_app | App background | Light gray (~#DDD) | Dark gray (~#333) |
| theme.color_fg_app | Foreground layer | Slightly darker | Slightly lighter |
| theme.color_bg_container | Card/container bg | Semi-transparent light | Semi-transparent dark |
| theme.color_bg_even | Alternating row (even) | Lighter | Darker |
| theme.color_bg_odd | Alternating row (odd) | Darker | Lighter |
| theme.color_bg_highlight | Highlight background | White-ish #FFFFFF22 | White-ish low opacity |
| theme.color_bg_highlight_inline | Inline highlight | color_d_1 | color_d_3 |
| theme.color_bg_unfocussed | Unfocused highlight | 85% of bg_highlight | 85% of bg_highlight |
| theme.color_app_caption_bar | Caption bar bg | Transparent | Transparent |
| theme.color_white | Pure white | #FFFFFF | #FFFFFF |
| theme.color_makepad | Makepad brand | #FF5C39 | #FF5C39 |
Theme Color Variables -- Text and Labels
| Variable | Purpose |
|----------|---------|
| theme.color_label_inner | Primary text on inner elements (buttons, labels) |
| theme.color_label_inner_hover | Text on hover |
| theme.color_label_inner_down | Text when pressed |
| theme.color_label_inner_focus | Text when focused |
| theme.color_label_inner_active | Text when active/selected |
| theme.color_label_inner_inactive | Secondary/muted text |
| theme.color_label_inner_disabled | Disabled text |
| theme.color_label_outer | Primary text on outer elements (tabs, headers) |
| theme.color_label_outer_off | Outer text when off |
| theme.color_label_outer_disabled | Disabled outer text |
| theme.color_text | General text color |
| theme.color_text_hover | Text on hover |
| theme.color_text_focus | Text on focus |
| theme.color_text_disabled | Disabled text |
| theme.color_text_placeholder | Placeholder text |
| theme.color_text_meta | Metadata text |
| theme.color_text_cursor | Text cursor color |
Theme Color Variables -- Widget States
Outset colors (buttons, raised elements):
| Variable | Purpose |
|----------|---------|
| theme.color_outset | Default button background |
| theme.color_outset_hover | Button on hover |
| theme.color_outset_down | Button when pressed |
| theme.color_outset_active | Active toggle state |
| theme.color_outset_focus | Focused button |
| theme.color_outset_disabled | Disabled button |
| theme.color_outset_inactive | Inactive button |
Inset colors (text inputs, checkboxes, radio buttons):
| Variable | Purpose |
|----------|---------|
| theme.color_inset | Default input background |
| theme.color_inset_hover | Input on hover |
| theme.color_inset_focus | Input on focus |
| theme.color_inset_disabled | Disabled input |
| theme.color_inset_empty | Empty input |
Selection and highlight:
| Variable | Purpose |
|----------|---------|
| theme.color_selection_focus | Text selection highlight |
| theme.color_selection_hover | Selection on hover |
| theme.color_highlight | General accent/highlight |
| theme.color_cursor | Cursor color |
| theme.color_cursor_focus | Focused cursor |
Theme Color Variables -- Semantic/Status
| Variable | Purpose | Value |
|----------|---------|-------|
| theme.color_error | Error state | Red (#C00) |
| theme.color_warning | Warning state | Orange (#FA0) |
| theme.color_high | High severity | Red (#C00) |
| theme.color_mid | Medium severity | Orange (#FA0) |
| theme.color_low | Low severity | Yellow-green (#8A0) |
| theme.color_panic | Panic/critical | Magenta (#f0f) |
Theme Color Variables -- Bevel System
The theme has a layered bevel system for 3D-like widget effects:
| Group | Variables | Purpose |
|-------|-----------|---------|
| color_bevel* | _hover, _focus, _active, _down, _disabled | Flat bevel |
| color_bevel_inset_1* | Same suffixes | Inner shadow (layer 1) |
| color_bevel_inset_2* | Same suffixes | Inner highlight (layer 2) |
| color_bevel_outset_1* | Same suffixes | Outer highlight (layer 1) |
| color_bevel_outset_2* | Same suffixes | Outer shadow (layer 2) |
Theme Color Variables -- Additional Widget Colors
| Variable Group | Purpose |
|----------------|---------|
| theme.color_icon* | Icon colors (default, inactive, active, disabled) |
| theme.color_mark* | Checkmark/radio mark colors |
| theme.color_val* | Progress bar and slider fill colors |
| theme.color_handle* | Slider handle colors |
| theme.color_shadow* | Shadow effects |
| theme.color_drag_quad | Drag preview overlay |
| theme.color_dock_tab_active | Active dock tab background |
Theme Font Variables
Font Sizes
Font sizes are computed from font_size_base and font_size_contrast:
| Variable | Formula | Default Value |
|----------|---------|---------------|
| theme.font_size_1 | base + 8 * contrast | 30.0 (largest heading) |
| theme.font_size_2 | base + 4 * contrast | 20.0 (medium heading) |
| theme.font_size_3 | base + 2 * contrast | 15.0 (small heading) |
| theme.font_size_4 | base + 1 * contrast | 12.5 (subheading) |
| theme.font_size_p | base | 10.0 (body text) |
| theme.font_size_code | fixed | 9.0 (monospace code) |
Font Styles (TextStyle objects)
| Variable | Description | Font File |
|----------|-------------|-----------|
| theme.font_regular | Regular weight body text | IBMPlexSans-Text.ttf |
| theme.font_bold | Bold/semibold text | IBMPlexSans-SemiBold.ttf |
| theme.font_italic | Italic text | IBMPlexSans-Italic.ttf |
| theme.font_bold_italic | Bold italic text | IBMPlexSans-BoldItalic.ttf |
| theme.font_code | Monospace code font | LiberationMono-Regular.ttf |
| theme.font_label | Label text (legacy) | IBMPlexSans-Text.ttf |
| theme.font_icons | Icon font (FontAwesome) | fa-solid-900.ttf |
Each font style includes multi-language support:
- Latin: IBM Plex Sans family
- Chinese: LXGW WenKai family
- Emoji: Noto Color Emoji
Line Spacing Constants
| Variable | Value | Purpose |
|----------|-------|---------|
| theme.font_wdgt_line_spacing | 1.2 | Widget text |
| theme.font_hl_line_spacing | 1.05 | Heading text |
| theme.font_longform_line_spacing | 1.2 | Long-form text |
Theme Spacing Variables
Base Spacing
Spacing is derived from space_factor (default 6.0):
| Variable | Formula | Default Value |
|----------|---------|---------------|
| theme.space_1 | 0.5 * space_factor | 3.0 (extra small) |
| theme.space_2 | 1.0 * space_factor | 6.0 (small/standard) |
| theme.space_3 | 1.5 * space_factor | 9.0 (medium) |
Margin/Padding Presets (Inset objects)
All-sides presets:
| Variable | Description | Values |
|----------|-------------|--------|
| theme.mspace_1 | XS padding all sides | 3px each |
| theme.mspace_2 | SM padding all sides | 6px each |
| theme.mspace_3 | MD padding all sides | 9px each |
Horizontal-only presets:
| Variable | Description | Values |
|----------|-------------|--------|
| theme.mspace_h_1 | XS horizontal padding | left/right: 3px, top/bottom: 0 |
| theme.mspace_h_2 | SM horizontal padding | left/right: 6px, top/bottom: 0 |
| theme.mspace_h_3 | MD horizontal padding | left/right: 9px, top/bottom: 0 |
Vertical-only presets:
| Variable | Description | Values |
|----------|-------------|--------|
| theme.mspace_v_1 | XS vertical padding | top/bottom: 3px, left/right: 0 |
| theme.mspace_v_2 | SM vertical padding | top/bottom: 6px, left/right: 0 |
| theme.mspace_v_3 | MD vertical padding | top/bottom: 9px, left/right: 0 |
Dimension Variables
| Variable | Purpose | Default |
|----------|---------|---------|
| theme.data_item_height | Standard data row height | ~23px |
| theme.data_icon_width | Standard icon width | ~16px |
| theme.data_icon_height | Standard icon height | ~22px |
| theme.container_corner_radius | Container border radius | 5.0 |
| theme.textselection_corner_radius | Text selection radius | 1.25 |
| theme.tab_height | Tab bar height | 36.0 |
Using Theme Variables in Splash
Colors
// Background color
draw_bg.color: theme.color_bg_container
// Text color
draw_text.color: theme.color_label_inner
// Muted/secondary text
draw_text.color: theme.color_label_inner_inactive
// Color math -- multiply for opacity
draw_text.color: theme.color_label_inner_inactive * 0.8
// Semantic colors
draw_bg.color: theme.color_warning
draw_bg.color: theme.color_error
Font Sizes
// Body text size
draw_text.text_style.font_size: theme.font_size_p
// Heading sizes
draw_text.text_style.font_size: theme.font_size_1 // Largest
draw_text.text_style.font_size: theme.font_size_2 // Medium
draw_text.text_style.font_size: theme.font_size_3 // Small
draw_text.text_style.font_size: theme.font_size_4 // Sub-heading
// Code font size
draw_text.text_style.font_size: theme.font_size_code
Font Styles
// Bold text with custom size (override with {} syntax)
draw_text.text_style: theme.font_bold{font_size: theme.font_size_2}
// Bold text keeping default size
draw_text.text_style: theme.font_bold{}
// Code font
draw_text.text_style: theme.font_code{}
// Override font size using +: merge syntax
draw_text +: {text_style +: {font_size: theme.font_size_3}}
Spacing
// Uniform padding
padding: theme.mspace_2
// Padding with overrides
padding: theme.mspace_2{left: theme.space_3, right: theme.space_3}
// Horizontal-only padding with custom values
padding: theme.mspace_h_1{left: theme.space_2, right: theme.space_2}
// Spacing between children
spacing: theme.space_2
// Computed spacing
width: Fill height: 9. * theme.space_1
padding: theme.mspace_3{left: theme.space_3 * 2, right: theme.space_3 * 2}
Color Syntax Reference
Splash supports multiple color formats (not theme-specific but essential):
| Syntax | Example | Description |
|--------|---------|-------------|
| #RGB | #f00 | Short hex (red) |
| #RRGGBB | #ff0000 | Full hex |
| #RRGGBBAA | #ff000080 | Hex with alpha |
| #xRRGGBB | #x2ecc71 | Hex starting with e (prefix #x) |
| #N | #D | Grayscale shorthand |
| vec4(r,g,b,a) | vec4(1.0, 0.0, 0.0, 1.0) | RGBA float (0.0-1.0) |
| theme.* | theme.color_highlight | Theme variable |
Important: When a hex color starts with the letter e, use the #x prefix
to avoid ambiguity with scientific notation. For example, #x2ecc71 not #2ecc71.
Color math is supported:
// Multiply for opacity
theme.color_label_inner_inactive * 0.8
// Mix two colors
mix(theme.color_w, theme.color_b, 0.5)
Theme Switching at Runtime
You can switch themes dynamically in Rust event handlers:
impl MatchEvent for App {
fn handle_actions(&mut self, cx: &mut Cx, actions: &Actions) {
if self.ui.button(cx, ids!(toggle_theme)).clicked(actions) {
script_eval!(cx, {
mod.theme = mod.themes.dark // or mod.themes.light
});
// All widgets using theme.* will pick up new values
// You may need to trigger a re-render
}
}
}
Or switch before widgets load (in App::run) for a static theme choice:
fn run(vm: &mut ScriptVm) -> Self {
crate::makepad_widgets::theme_mod(vm);
script_eval!(vm, {
mod.theme = mod.themes.light
});
crate::makepad_widgets::widgets_mod(vm);
App::from_script_mod(vm, self::script_mod)
}
Complete Theme-Aware UI Example
This example demonstrates a themed card list using only theme variables:
script_mod! {
use mod.prelude.widgets.*
let CardItem = RoundedView{
width: Fill height: Fit
padding: theme.mspace_2{left: theme.space_3, right: theme.space_3}
flow: Right spacing: theme.space_2
align: Align{y: 0.5}
draw_bg.color: theme.color_bg_container
draw_bg.border_radius: theme.container_corner_radius
icon := Label{
width: 24 height: 24
text: ""
}
content := View{
width: Fill height: Fit
flow: Down spacing: theme.space_1
title := Label{
text: "Title"
draw_text.color: theme.color_label_inner
draw_text.text_style: theme.font_bold{font_size: theme.font_size_4}
}
subtitle := Label{
text: "Subtitle"
draw_text.color: theme.color_label_inner_inactive
draw_text.text_style.font_size: theme.font_size_p
}
}
badge := RoundedView{
width: Fit height: Fit
padding: theme.mspace_h_1
draw_bg.color: theme.color_bg_highlight_inline
draw_bg.border_radius: 4.0
badge_label := Label{
text: "new"
draw_text.color: theme.color_highlight
draw_text.text_style.font_size: theme.font_size_code
draw_text.text_style: theme.font_bold{}
}
}
}
startup() do #(App::script_component(vm)){
ui: Root{
main_window := Window{
pass.clear_color: theme.color_bg_app
window.inner_size: vec2(400, 600)
body +: {
width: Fill height: Fill
flow: Down spacing: 0
// Header
SolidView{
width: Fill height: Fit
padding: theme.mspace_3
draw_bg.color: theme.color_app_caption_bar
Label{
text: "My Cards"
draw_text.color: theme.color_label_inner
draw_text.text_style: theme.font_bold{font_size: theme.font_size_2}
}
}
// Card list
ScrollYView{
width: Fill height: Fill
padding: theme.mspace_2
flow: Down spacing: theme.space_1
CardItem{title.text: "First Card" subtitle.text: "Description here"}
CardItem{title.text: "Second Card" subtitle.text: "Another card"}
}
// Footer
SolidView{
width: Fill height: Fit
padding: theme.mspace_2
draw_bg.color: theme.color_bg_container
Label{
text: "2 items"
draw_text.color: theme.color_label_inner_inactive
draw_text.text_style.font_size: theme.font_size_code
}
}
}
}
}
}
}
Best Practices
-
ALWAYS use
theme.*for colors in production apps -- never hardcode#ff0000whentheme.color_errorexists. This enables theme switching and accessibility. -
Use theme fonts for consistent typography -- prefer
theme.font_bold{font_size: theme.font_size_2}over manually specifying font families. -
Use theme spacing for consistent layout --
theme.mspace_2andtheme.space_2create a harmonious rhythm. Avoid magic numbers likepadding: Inset{top: 8, ...}. -
Override with
{}syntax -- extend theme values without replacing them:theme.font_bold{font_size: 20}keeps the bold font family but overrides size.theme.mspace_2{left: theme.space_3}keeps top/right/bottom but overrides left. -
Use merge
+:for partial overrides -- when you only want to change one nested property:draw_text +: {text_style +: {font_size: theme.font_size_3}} -
Choose the right text color variable:
theme.color_label_innerfor primary UI text (buttons, labels)theme.color_label_inner_inactivefor secondary/muted texttheme.color_textfor general content texttheme.color_text_placeholderfor placeholder text
-
Use state-aware color variants -- widgets that change on hover/focus/press should use the matching
_hover,_focus,_downsuffixes. -
Multiply for subtle opacity --
theme.color_label_inner_inactive * 0.8creates a subtler variant without a new variable. -
Select theme before
widgets_mod-- the theme must be set betweentheme_mod()andwidgets_mod()calls, so widget definitions pick up the correct theme values. -
For simple apps, use
crate::makepad_widgets::script_mod(vm)which loads everything with the default dark theme in one call.
Source Files
- Theme dark:
widgets/src/theme_desktop_dark.rs - Theme light:
widgets/src/theme_desktop_light.rs - Theme skeleton:
widgets/src/theme_desktop_skeleton.rs - Theme loader:
widgets/src/lib.rs(theme_modandwidgets_modfunctions) - Example usage:
examples/todo/src/app.rs(light theme with full theme variable usage) - Example usage:
examples/counter/src/app.rs(default dark theme)