Configuring In-Application Theme Skinning
Overview
For applications developed with ArkTS, you can switch themes within the application, such as toggling between dark and light modes or changing the skin. Note that this feature is limited to in-application use and does not apply at the UIAbility or window level. It is not available for the C API and Node-API.
Customizing Theme Colors
To implement theme switching for your application, you must define custom theme colors. Use CustomTheme to specify the colors you wish to modify. Its properties are optional. Only values that need to be changed from system defaults should be specified. Unmodified values inherit from the system's default settings. For details, see System Default Token Color Values. The following is an example of how to define custom theme colors:
import { CustomColors, CustomTheme } from '@kit.ArkUI';
export class AppColors implements CustomColors {
// Custom theme colors
public brand: ResourceColor = '#FF75D9';
// Use $r to set the primary warning color to different colors in dark and light modes.
public warning: ResourceColor = $r('sys.color.ohos_id_color_warning');
}
export class AppTheme implements CustomTheme {
public colors: AppColors = new AppColors();
}
export let gAppTheme: CustomTheme = new AppTheme();
Setting Custom Theme Colors for Application Components
-
When setting custom theme colors at the page entry point, ensure that ThemeControl.setDefaultTheme is executed before the page is built.
Use the onWillApplyTheme callback to allow custom components to access the currently active Theme object.
// Index.ets import { Theme, ThemeControl } from '@kit.ArkUI'; import { gAppTheme } from './AppTheme'; // Execute ThemeControl before the page builds. ThemeControl.setDefaultTheme(gAppTheme); @Entry @Component struct DisplayPage { @State menuItemColor: ResourceColor = $r('sys.color.background_primary'); onWillApplyTheme(theme: Theme) { this.menuItemColor = theme.colors.backgroundPrimary; } build() { Column() { List({ space: 10 }) { ListItem() { Column({ space: '5vp' }) { Text('Color mode') .margin({ top: '5vp', left: '14fp' }) .width('100%') Row() { Column() { Text('Light') .fontSize('16fp') .textAlign(TextAlign.Start) .alignSelf(ItemAlign.Center) Radio({ group: 'light or dark', value: 'light' }) .checked(true) } .width('50%') Column() { Text('Dark') .fontSize('16fp') .textAlign(TextAlign.Start) .alignSelf(ItemAlign.Center) Radio({ group: 'light or dark', value: 'dark' }) } .width('50%') } } .width('100%') .height('90vp') .borderRadius('10vp') .backgroundColor(this.menuItemColor) } ListItem() { Column() { Text('Brightness') .width('100%') .margin({ top: '5vp', left: '14fp' }) Slider({ value: 40, max: 100 }) } .width('100%') .height('70vp') .borderRadius('10vp') .backgroundColor(this.menuItemColor) } ListItem() { Column() { Row() { Column({ space: '5vp' }) { Text('Touch sensitivity') .fontSize('16fp') .textAlign(TextAlign.Start) .width('100%') Text('Increase the touch sensitivity of your screen' + ' for use with screen protectors') .fontSize('12fp') .fontColor(Color.Blue) .textAlign(TextAlign.Start) .width('100%') } .alignSelf(ItemAlign.Center) .margin({ left: '14fp' }) .width('75%') Toggle({ type: ToggleType.Switch, isOn: true }) .margin({ right: '14fp' }) .alignSelf(ItemAlign.Center) } .width('100%') .height('80vp') } .width('100%') .borderRadius('10vp') .backgroundColor(this.menuItemColor) } ListItem() { Column() { Text('Warning') .width('100%') .margin({ top: '5vp', left: '14fp' }) Button('Text') .type(ButtonType.Capsule) .role(ButtonRole.ERROR) .width('40%') } .width('100%') .height('70vp') .borderRadius('10vp') .backgroundColor(this.menuItemColor) } } } .padding('10vp') .backgroundColor('#dcdcdc') .width('100%') .height('100%') } } -
When setting custom theme colors in UIAbility, call ThemeControl.setDefaultTheme within the completion callback of windowStage.loadContent in the onWindowStageCreate() API.
// EntryAbility.ets import {AbilityConstant, UIAbility, Want } from '@kit.AbilityKit'; import { hilog } from '@kit.PerformanceAnalysisKit'; import { window, CustomColors, ThemeControl } from '@kit.ArkUI'; class AppColors implements CustomColors { fontPrimary = 0xFFD53032; iconOnPrimary = 0xFFD53032; iconFourth = 0xFFD53032; } const abilityThemeColors = new AppColors(); export default class EntryAbility extends UIAbility { onCreate(want: Want, launchParam: AbilityConstant.LaunchParam) { hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onCreate'); } onDestroy() { hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onDestroy'); } onWindowStageCreate(windowStage: window.WindowStage) { // The main window is created. Set a main page for this ability. hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onWindowStageCreate'); windowStage.loadContent('pages/Index', (err, data) => { if (err.code) { hilog.error(0x0000, 'testTag', 'Failed to load the content. Cause: %{public}s', JSON.stringify(err) ?? ''); return; } hilog.info(0x0000, 'testTag', 'Succeeded in loading the content. Data: %{public}s', JSON.stringify(data) ?? ''); // Call setDefaultTheme in the onWindowStageCreate() API. ThemeControl.setDefaultTheme({ colors: abilityThemeColors }); hilog.info(0x0000, 'testTag', '%{public}s', 'ThemeControl.setDefaultTheme done'); }); } }
NOTE
-
When the parameter passed to setDefaultTheme is undefined, the previously applied custom theme is cleared. The color values corresponding to the theme tokens will then revert to the system defaults defined in System Default Token Color Values.
-
setDefaultTheme must be called after ArkUI initialization completes, specifically within the windowStage.loadContent completion callback.
-
Setting a Custom Theme Style for Specific Application Pages
You can use WithTheme to apply the color scheme of a custom theme to the default styles of internal components. Within the scope of WithTheme, the color scheme of components will be adjusted according to the theme's color scheme.
NOTE
When using WithTheme in the custom node BuilderNode, to ensure the correct display effect, it is necessary to manually pass system environment change events to trigger a full update of the node. For details, see the BuilderNode system environment change section.
As demonstrated in the example, WithTheme({ theme: this.CustomTheme }) applies the custom theme styling to all components within its scope. Theme styles can be dynamically updated by modifying this.CustomTheme. The onWillApplyTheme callback allows custom components to access the currently active Theme object.
import { CustomColors, CustomTheme, Theme } from '@kit.ArkUI';
import { common } from '@kit.AbilityKit';
// Replace $r('app.color.xxx') with the actual resource file.
class AppColors implements CustomColors {
public fontPrimary: ResourceColor = $r('app.color.brand_purple');
public backgroundEmphasize: ResourceColor = $r('app.color.brand_purple');
}
class AppColorsSec implements CustomColors {
public fontPrimary: ResourceColor = $r('app.color.brand');
public backgroundEmphasize: ResourceColor = $r('app.color.brand');
}
class AppTheme implements CustomTheme {
public colors: AppColors = new AppColors();
}
class AppThemeSec implements CustomTheme {
public colors: AppColors = new AppColorsSec();
}
@Entry
@Component
struct DisplayPage1 {
@State customTheme: CustomTheme = new AppTheme();
// Replace $r('app.string.SetCustomThemeStyle') with the actual resource file. In this example, the value for the resource file is "Set a custom theme style for specific pages".
@State message: ResourceStr = $r('app.string.SetCustomThemeStyle');
count = 0;
build() {
WithTheme({ theme: this.customTheme }) {
Row(){
Column() {
Text('WithTheme')
.fontSize(30)
.margin({bottom: 10})
Text(this.message)
.margin({bottom: 10})
Button('change theme').onClick(() => {
this.count++;
if (this.count > 1) {
this.count = 0;
}
switch (this.count) {
case 0:
this.customTheme = new AppTheme();
break;
case 1:
this.customTheme = new AppThemeSec();
break;
default:
break;
}
})
}
.width('100%')
}
.height('100%')
.width('100%')
}
}
}

Setting the Color Mode for Application Pages
Using WithTheme, you can set three color modes: following the system mode, light mode, and dark mode.
Within the scope of WithTheme, component styles adapt to the specified color mode by accessing the corresponding system and application resource values. This means that the color schemes of components are adjusted according to the chosen color mode.
In the example below, components within the scope are set to dark mode using WithTheme({ colorMode: ThemeColorMode.DARK }).
For the light and dark modes to take effect, add a dark.json resource file.

Example of the dark.json file content:
{
"color": [
{
"name": "start_window_background",
"value": "#000000"
}
]
}
import { ThemeControl } from '@kit.ArkUI';
ThemeControl.setDefaultTheme(undefined);
@Entry
@Component
struct DisplayPage3 {
@State message: string = 'Hello World';
@State colorMode: ThemeColorMode = ThemeColorMode.DARK;
build() {
WithTheme({ colorMode: this.colorMode }) {
Row() {
Column() {
Text(this.message)
.fontSize(50)
.fontWeight(FontWeight.Bold)
Button('Switch ColorMode').onClick(() => {
if (this.colorMode === ThemeColorMode.LIGHT) {
this.colorMode = ThemeColorMode.DARK;
} else if (this.colorMode === ThemeColorMode.DARK) {
this.colorMode = ThemeColorMode.LIGHT;
}
})
}
.width('100%')
}
.backgroundColor($r('sys.color.background_primary'))
.height('100%')
// Extend the safe area to implement an immersive effect that adapts to light and dark mode changes.
.expandSafeArea(
[SafeAreaType.SYSTEM], [SafeAreaEdge.TOP, SafeAreaEdge.END, SafeAreaEdge.BOTTOM, SafeAreaEdge.START])
}
}
}

System Default Token Color Values
| Token | Category | Light | Description | Dark | Description |
|---|---|---|---|---|---|
| theme.colors.brand | Brand color. | #ff0a59f7 | ![]() |
#ff317af7 | ![]() |
| theme.colors.warning | Alert color. | #ffe84026 | ![]() |
#ffd94838 | ![]() |
| theme.colors.alert | Warning color. | #ffed6f21 | ![]() |
#ffdb6b42 | ![]() |
| theme.colors.confirm | Confirmation color. | #ff64bb5c | ![]() |
#ff5be854 | ![]() |
| theme.colors.fontPrimary | Primary text color. | #e5000000 | ![]() |
#e5ffffff | ![]() |
| theme.colors.fontSecondary | Secondary text color. | #99000000 | ![]() |
#99ffffff | ![]() |
| theme.colors.fontTertiary | Tertiary text color. | #66000000 | ![]() |
#66ffffff | ![]() |
| theme.colors.fontFourth | Quaternary text color. | #33000000 | ![]() |
#33ffffff | ![]() |
| theme.colors.fontEmphasize | Highlight text color. | #ff0a59f7 | ![]() |
#ff317af7 | ![]() |
| theme.colors.fontOnPrimary | Primary text invert color. | #ffffffff | ![]() |
#ff000000 | ![]() |
| theme.colors.fontOnSecondary | Secondary text invert color. | #99ffffff | ![]() |
#99000000 | ![]() |
| theme.colors.fontOnTertiary | Tertiary text invert color. | #66ffffff | ![]() |
#66000000 | ![]() |
| theme.colors.fontOnFourth | Quaternary text invert color. | #33ffffff | ![]() |
#33000000 | ![]() |
| theme.colors.iconPrimary | Primary icon color. | #e5000000 | ![]() |
#e5ffffff | ![]() |
| theme.colors.iconSecondary | Secondary icon color. | #99000000 | ![]() |
#99ffffff | ![]() |
| theme.colors.iconTertiary | Tertiary icon color. | #66000000 | ![]() |
#66ffffff | ![]() |
| theme.colors.iconFourth | Quaternary icon color. | #33000000 | ![]() |
#33ffffff | ![]() |
| theme.colors.iconEmphasize | Emphasis icon color. | #ff0a59f7 | ![]() |
#ff317af7 | ![]() |
| theme.colors.iconSubEmphasize | Emphasis auxiliary icon color. | #660a59f7 | ![]() |
#66317af7 | ![]() |
| theme.colors.iconOnPrimary | Primary icon invert color. | #ffffffff | ![]() |
#ff000000 | ![]() |
| theme.colors.iconOnSecondary | Secondary icon invert color. | #99ffffff | ![]() |
#99000000 | ![]() |
| theme.colors.iconOnTertiary | Tertiary icon invert color. | #66ffffff | ![]() |
#66000000 | ![]() |
| theme.colors.iconOnFourth | Quaternary icon invert color. | #33ffffff | ![]() |
#33000000 | ![]() |
| theme.colors.backgroundPrimary | Primary background color (solid, opaque). | #ffffffff | ![]() |
#ffe5e5e5 | ![]() |
| theme.colors.backgroundSecondary | Secondary background color (solid, opaque). | #fff1f3f5 | ![]() |
#ff191a1c | ![]() |
| theme.colors.backgroundTertiary | Tertiary background color (solid, opaque). | #ffe5e5ea | ![]() |
#ff202224 | ![]() |
| theme.colors.backgroundFourth | Quaternary background color (solid, opaque). | #ffd1d1d6 | ![]() |
#ff2e3033 | ![]() |
| theme.colors.backgroundEmphasize | Emphasis background color (solid, opaque). | #ff0a59f7 | ![]() |
#ff317af7 | ![]() |
| theme.colors.compForegroundPrimary | Foreground. | #ff000000 | ![]() |
#ffe5e5e5 | ![]() |
| theme.colors.compBackgroundPrimary | White background. | #ffffffff | ![]() |
#ffffffff | ![]() |
| theme.colors.compBackgroundPrimaryTran | White transparent background. | #ffffffff | ![]() |
#33ffffff | ![]() |
| theme.colors.compBackgroundPrimaryContrary | Always-on background. | #ffffffff | ![]() |
#ffe5e5e5 | ![]() |
| theme.colors.compBackgroundGray | Gray background. | #fff1f3f5 | ![]() |
#ffe5e5ea | ![]() |
| theme.colors.compBackgroundSecondary | Secondary background. | #19000000 | ![]() |
#19ffffff | ![]() |
| theme.colors.compBackgroundTertiary | Tertiary background. | #0c000000 | ![]() |
#0cffffff | ![]() |
| theme.colors.compBackgroundEmphasize | Emphasis background. | #ff0a59f7 | ![]() |
#ff317af7 | ![]() |
| theme.colors.compBackgroundNeutral | Black, neutral, emphasis background. | #ff000000 | ![]() |
#ffffffff | ![]() |
| theme.colors.compEmphasizeSecondary | 20% emphasis background. | #330a59f7 | ![]() |
#33317af7 | ![]() |
| theme.colors.compEmphasizeTertiary | 10% emphasis background. | #190a59f7 | ![]() |
#19317af7 | ![]() |
| theme.colors.compDivider | Divider color. | #33000000 | ![]() |
#33ffffff | ![]() |
| theme.colors.compCommonContrary | Common invert color. | #ffffffff | ![]() |
#ff000000 | ![]() |
| theme.colors.compBackgroundFocus | Background color in the focused state. | #fff1f3f5 | ![]() |
#ff000000 | ![]() |
| theme.colors.compFocusedPrimary | Primary inverted color in the focused state. | #e5000000 | ![]() |
#e5ffffff | ![]() |
| theme.colors.compFocusedSecondary | Secondary inverted color in the focused state. | #99000000 | ![]() |
#99ffffff | ![]() |
| theme.colors.compFocusedTertiary | Tertiary inverted color in the focused state. | #66000000 | ![]() |
#66ffffff | ![]() |
| theme.colors.interactiveHover | Common interactive color for the hover state. | #0c000000 | ![]() |
#0cffffff | ![]() |
| theme.colors.interactivePressed | Common interactive color for the pressed state. | #19000000 | ![]() |
#19ffffff | ![]() |
| theme.colors.interactiveFocus | Common interactive color for the focused state. | #ff0a59f7 | ![]() |
#ff317af7 | ![]() |
| theme.colors.interactiveActive | Common interactive color for the active state. | #ff0a59f7 | ![]() |
#ff317af7 | ![]() |
| theme.colors.interactiveSelect | Common interactive color for the selected state. | #33000000 | ![]() |
#33ffffff | ![]() |
| theme.colors.interactiveClick | Common interactive color for the clicked state. | #19000000 | ![]() |
#19ffffff | ![]() |
For details about the components that can be affected by each token color value, see Colors.


































