Design Tokens
Introduction
Design tokens are the core of any design system. They are the smallest pieces of a design system that define the visual properties of a UI. These properties can include colors, typography, spacing, and more. Design tokens are used to maintain consistency across an application and make it easier to update the design system. As Mix is a solution for styling and is used for building design systems, it also supports design tokens.
Getting Started
Mix has built-in support for design tokens. It naturally integrates with five types of them: color
, textStyle
, space
, radius
, and breakpoint
. These tokens are used to define the visual properties in key-value pairs, where the key is the token name and the value is the actual value of the token.
final primary = ColorToken('primary');
final docTheme = MixThemeData(
colors: {
primary: Colors.lightBlue,
},
textStyles: {...},
radii: {...},
spaces: {...},
);
MixTheme and MixThemeData
In the same way we use Theme
and ThemeData
in Flutter, Mix has MixTheme
and MixThemeData
. MixTheme
is a widget that wraps the application and provides the design tokens to all its child widgets.
MixTheme(
data: docTheme,
child: ...
);
MixThemeData
MixThemeData
is a class that defines the values for the design tokens that will be used throughout the application. It has five properties related to Design Tokens: colors
, textStyles
, spaces
, radii
, and breakpoints
. Each property is a map that contains the tokens and their values.
const primary = ColorToken('primary');
const h1 = TextStyleToken('h1');
const radiusMedium = RadiusToken('radiusMedium');
const spaceMedium = SpaceToken('spaceMedium');
final docTheme = MixThemeData(
colors: {
primary: Colors.lightBlue,
},
textStyles: {
h1: const TextStyle(
fontSize: 24,
fontWeight: FontWeight.bold,
),
},
radii: {
radiusMedium: const Radius.circular(10),
},
spaces: {
spaceMedium: 10,
},
);
Also, MixThemeData
has an additional property called defaultOrderOfModifiers
, which is used to override the default order of modifiers. Therefore, if you prefer to use a different order of modifiers, you can set it in the defaultOrderOfModifiers
instead of using the orderOfModifiers
property in each StyledWidget
. To set your own order of modifiers, you just need to pass a list of types that extends WidgetModifierSpec
. For instance:
final docTheme = MixThemeData(
// ...
defaultOrderOfModifiers: const [
SizedBoxModifierSpec,
ClipRectModifierSpec,
TransformModifierSpec,
],
);
Dynamic Themes
Mix allows you to define dynamic themes, which means that the values of the design tokens can be resolved dynamically using a BuildContext (opens in a new tab). To do this, you need to use the ColorResolver
, TextStyleResolver
, RadiusResolver
, SpaceResolver
and BreakpointResolver
classes. These classes are used to resolve the values of the design tokens based on the current BuildContext
.
As an example, let's say you want to define a surface
color token that changes based on the platform brightness. You can do this by using the ColorResolver
class:
final surface = ColorToken('surface');
final docTheme = MixThemeData(
colors: {
surface: ColorResolver((context) {
if (MediaQuery.of(context).platformBrightness == Brightness.dark) {
return Colors.black;
}
return Colors.white;
}),
},
);
Applying design tokens
After defining the design tokens, you can use them in your application to style your widgets. All attributes that accept a token as a value have a ref
method that allows you to access the token value. For instance:
$box.color.ref(ColorToken)
$text.style.ref(TextStyleToken)
$box.borderRadius.ref(RadiusToken)
$box.padding.all.ref(SpaceToken) // You can replace `all` with any other padding property
In a real example:
Box(
style: Style(
$box.color.ref(primary),
$box.height(100),
$box.width(100),
),
);
Mix's tokens are also flexible enough to be used in Flutter's widgets, you just need to use the resolve
method to get the token value.
Container(
height: 200,
width: 200,
color: primary.resolve(context),
)
Next Steps
Now that you know how design tokens work in Mix, you can start defining your own tokens and using them in your application. For a step-by-step tutorial on how to theme your app with Mix, check the Theming guide.