Skip to main content
Version: v8

Dark Mode

Ionic makes it easy to change the palettes of your app, including supporting dark color schemes. Dark mode is a display setting that changes all of an app's views to a dark palette. It has system-wide support on iOS and Android, making it highly desirable for developers to add to their apps.

Enabling Dark Palette

There are three provided ways to enable the dark palette in an app: always, based on system settings, or by using a CSS class.

Always

The default palette provided with Ionic Framework is a light palette, consisting of a light background and dark text. However, the default palette can be changed to the dark palette by importing the following stylesheet in the appropriate files:

@import '@ionic/angular/css/palettes/dark.always.css';

This sets the application colors and stepped colors on the :root selector.

The following example will always display the dark palette, regardless of the system settings for dark mode.

<ion-header class="ion-no-border">
<ion-toolbar>
<ion-buttons slot="start">
<ion-back-button default-href="#"></ion-back-button>
</ion-buttons>
<ion-title>Display</ion-title>
<ion-buttons slot="end">
<ion-button color="dark">
<ion-icon slot="icon-only" ios="person-circle-outline" md="person-circle"></ion-icon>
</ion-button>
</ion-buttons>
</ion-toolbar>
</ion-header>

<ion-content>
<ion-list-header>Appearance</ion-list-header>
<ion-list [inset]="true">
<ion-item [button]="true">Text Size</ion-item>
<ion-item>
<ion-toggle justify="space-between">Bold Text</ion-toggle>
</ion-item>
</ion-list>

<ion-list-header>Brightness</ion-list-header>
<ion-list [inset]="true">
<ion-item>
<ion-range value="40">
<ion-icon name="sunny-outline" slot="start"></ion-icon>
<ion-icon name="sunny" slot="end"></ion-icon>
</ion-range>
</ion-item>
<ion-item>
<ion-toggle justify="space-between" checked>True Tone</ion-toggle>
</ion-item>
</ion-list>

<ion-list [inset]="true">
<ion-item [button]="true">
<ion-label>Night Shift</ion-label>
<ion-text slot="end" color="medium">9:00 PM to 8:00 AM</ion-text>
</ion-item>
</ion-list>
</ion-content>
Important

Avoid targeting the .ios or .md selectors to override the Ionic dark palette, as these classes are added to each component and will take priority over globally defined variables. Instead, we can target the mode-specific classes on the :root element.

System

The system approach to enable dark mode involves checking the system settings for the user's preferred color scheme. This is the default when starting a new Ionic Framework app. Importing the following stylesheet in the appropriate file will automatically retrieve the user's preference from the system settings and apply the dark palette when dark mode is preferred:

@import '@ionic/angular/css/palettes/dark.system.css';

This sets the application colors and stepped colors when the CSS media query for prefers-color-scheme is dark. The prefers-color-scheme media query is supported by all modern browsers. If support for older browser is required, we recommend using the CSS class approach.

The following example uses the system settings to decide when to show dark mode.

info

Not sure how to change the system settings? Here's how to enable dark mode on Windows 11 and on macOS.

<ion-header class="ion-no-border">
<ion-toolbar>
<ion-buttons slot="start">
<ion-back-button default-href="#"></ion-back-button>
</ion-buttons>
<ion-title>Display</ion-title>
<ion-buttons slot="end">
<ion-button color="dark">
<ion-icon slot="icon-only" ios="person-circle-outline" md="person-circle"></ion-icon>
</ion-button>
</ion-buttons>
</ion-toolbar>
</ion-header>

<ion-content>
<ion-list-header>Appearance</ion-list-header>
<ion-list [inset]="true">
<ion-item [button]="true">Text Size</ion-item>
<ion-item>
<ion-toggle justify="space-between">Bold Text</ion-toggle>
</ion-item>
</ion-list>

<ion-list-header>Brightness</ion-list-header>
<ion-list [inset]="true">
<ion-item>
<ion-range value="40">
<ion-icon name="sunny-outline" slot="start"></ion-icon>
<ion-icon name="sunny" slot="end"></ion-icon>
</ion-range>
</ion-item>
<ion-item>
<ion-toggle justify="space-between" checked>True Tone</ion-toggle>
</ion-item>
</ion-list>

<ion-list [inset]="true">
<ion-item [button]="true">
<ion-label>Night Shift</ion-label>
<ion-text slot="end" color="medium">9:00 PM to 8:00 AM</ion-text>
</ion-item>
</ion-list>
</ion-content>
Important

Avoid targeting the .ios or .md selectors to override the Ionic dark palette, as these classes are added to each component and will take priority over globally defined variables. Instead, we can target the mode-specific classes on the :root element.

CSS Class

While the previous approaches are excellent for enabling the dark palette through file imports alone, there are scenarios where you may need more control over its application. In cases where you need to apply the dark palette conditionally, such as through a toggle, or if you want to extend the functionality based on system settings, we provide a dark palette class file. This file applies the dark palette when a specific class is added to an app. Importing the following stylesheet into the appropriate file will provide the necessary styles for using the dark palette with the class:

@import '@ionic/angular/css/palettes/dark.class.css';

This sets the application colors and stepped colors on the .ion-palette-dark selector, which must be applied to the app by the developer.

The following example combines site settings, system settings, and the toggle to decide when to show dark mode. The site's palette takes precedence over system settings. If your system settings differ from the site's palette when the demo loads, it will use the site's palette.

info

Not sure how to change the system settings? Here's how to enable dark mode on Windows 11 and on macOS.

<ion-header class="ion-no-border">
<ion-toolbar>
<ion-buttons slot="start">
<ion-back-button default-href="#"></ion-back-button>
</ion-buttons>
<ion-title>Display</ion-title>
<ion-buttons slot="end">
<ion-button color="dark">
<ion-icon slot="icon-only" ios="person-circle-outline" md="person-circle"></ion-icon>
</ion-button>
</ion-buttons>
</ion-toolbar>
</ion-header>

<ion-content>
<ion-list-header>Appearance</ion-list-header>
<ion-list [inset]="true">
<ion-item>
<ion-toggle [(ngModel)]="paletteToggle" (ionChange)="toggleChange($event)" justify="space-between"
>Dark Mode</ion-toggle
>
</ion-item>
</ion-list>

<ion-list [inset]="true">
<ion-item [button]="true">Text Size</ion-item>
<ion-item>
<ion-toggle justify="space-between">Bold Text</ion-toggle>
</ion-item>
</ion-list>

<ion-list-header>Brightness</ion-list-header>
<ion-list [inset]="true">
<ion-item>
<ion-range value="40">
<ion-icon name="sunny-outline" slot="start"></ion-icon>
<ion-icon name="sunny" slot="end"></ion-icon>
</ion-range>
</ion-item>
<ion-item>
<ion-toggle justify="space-between" checked>True Tone</ion-toggle>
</ion-item>
</ion-list>

<ion-list [inset]="true">
<ion-item [button]="true">
<ion-label>Night Shift</ion-label>
<ion-text slot="end" color="medium">9:00 PM to 8:00 AM</ion-text>
</ion-item>
</ion-list>
</ion-content>
Important

The .ion-palette-dark class must be added to the html element in order to work with the imported dark palette.

Adjusting System UI Components

When developing a dark palette, you may notice that certain system UI components are not adjusting to dark mode properly. To fix this you will need to specify the color-scheme. See the browser compatibility for color-scheme for details on cross browser support.

While you may be mainly using Ionic components instead of only native components, color-scheme can also affect aspects of your application such as the scrollbar. In order to use color-scheme you will need to add the following HTML to the head of your application:

<meta name="color-scheme" content="light dark" />

This allows the page to indicate which color scheme it is comfortable being rendered with. Alternatively, you can add the following CSS to do this on a per-element basis:

color-scheme: light dark;
Default scrollbarScrollbar with color-scheme
A default light-themed scrollbar in an application interface.A dark-themed scrollbar in an application interface, demonstrating the effect of the &#39;color-scheme&#39; property.

For more information regarding color-scheme, please see the Web.dev guide on color schemes.

note

color-scheme does not apply to the keyboard. For details on how dark mode works with the keyboard, see Keyboard Documentation.

note

For developers looking to customize the theme color under the status bar in Safari on iOS 15 or the toolbar in Safari on macOS, see theme-color Meta.

Ionic Dark Palette

Ionic has a recommended dark palette that can be enabled in three different ways: always, based on system settings, or by using a CSS class. Each of these methods involves importing the dark palette file with the corresponding name.

The contents of each file are included below for reference. These variables are set by importing the relevant dark palette file and do not need to be copied into an app. For more information on the variables being changed, including additional variables for further customization, refer to the Themes section.

The always dark palette behaves in the following ways:

  1. Sets the Ionic colors for all modes to complement a dark palette in the :root selector. The :root selector is identical to the selector html, except that its specificity is higher.
  2. Setting variables for the dark palette on ios devices using the :root.ios selector.
  3. Setting variables for the dark palette on md devices using the :root.md selector.
caution

It is important to pay attention to the specificity if you want to override any of the Ionic dark palette variables. For example, because the --ion-item-background variable is set for each mode, it cannot be overridden in the :root selector. A higher specificity selector, such as :root.ios, is required.

info

The contents of Ionic's dark palette can be viewed on GitHub. The CSS used to apply the always dark palette can be found in the repository.