Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/aparajita/capacitor-dark-mode
Universal, reliable dark mode support for Capacitor apps on the web, iOS and Android
https://github.com/aparajita/capacitor-dark-mode
capacitor capacitor-android capacitor-ios capacitor-plugin capacitor-web
Last synced: 24 days ago
JSON representation
Universal, reliable dark mode support for Capacitor apps on the web, iOS and Android
- Host: GitHub
- URL: https://github.com/aparajita/capacitor-dark-mode
- Owner: aparajita
- License: mit
- Created: 2022-07-23T22:27:36.000Z (over 2 years ago)
- Default Branch: main
- Last Pushed: 2024-07-16T18:59:41.000Z (5 months ago)
- Last Synced: 2024-10-29T01:19:15.513Z (about 1 month ago)
- Topics: capacitor, capacitor-android, capacitor-ios, capacitor-plugin, capacitor-web
- Language: TypeScript
- Homepage:
- Size: 596 KB
- Stars: 24
- Watchers: 2
- Forks: 3
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
Awesome Lists containing this project
- awesome-capacitorjs - @aparajita/capacitor-dark-mode
- awesome-capacitor - Dark Mode - Universal, reliable dark mode support on the web, iOS and Android. ([Aparajita plugins](https://github.com/aparajita?tab=repositories&q=capacitor))
README
# capacitor-dark-mode [![npm version](https://badge.fury.io/js/@aparajita%2Fcapacitor-dark-mode.svg)](https://badge.fury.io/js/@aparajita%2Fcapacitor-dark-mode)
This [Capacitor 6](https://capacitorjs.com) plugin is a complete dark mode solution for Ionic web, iOS and Android.
### ❗️Breaking changes
In order to conform to Ionic 8’s built in dark mode support when importing `@ionic/vue/css/palettes/dark.class.css`, two changes have been made:
- When running on Ionic 8+, the default dark mode class is now `.ion-palette-dark`. If you were using the default `.dark` class, replace all usages of `.dark` in your CSS with `.ion-palette-dark`.
- The dark mode class is now applied to the `html` element instead of the `body`.
In order to conform with the Capacitor 6 listener interface, [`addAppearanceListener`](#addappearancelistener) now returns only a Promise and must be awaited.
## Motivation
On the web and iOS, dark mode works easily with Ionic because browsers and WKWebView correctly handle the `prefers-color-scheme` CSS property. On Android, on the other hand, `prefers-color-scheme` is [well and truly broken](https://developer.android.com/guide/webapps/dark-theme). I have never seen it work reliably in an Ionic app, even with Capacitor 5 and the Android DayNight theme.
With this plugin, you can easily enable and control dark mode in your app across **all** platforms, guaranteed! This means that on Android versions prior to 10 (API 29), which is the first version to support system dark mode, you can allow the user to toggle dark mode.
### Keep it DRY
If you implement dark mode as a user preference, you cannot rely solely on the CSS `prefers-color-scheme` query anyway; you have to use a class to indicate whether or not you are in dark mode. Maintaining an identical set of CSS variables for `prefers-color-scheme: dark` and a dark class selector is error-prone, extra maintenance, and in general violates the DRY principle.
This plugin relies solely on a dark class selector to indicate whether or not you are in dark mode, and manages the dark class for you based on the system dark mode and/or the user preference.
[Installation](#installation) | [Configuration](#configuration) | [Usage](#usage) | [API](#api)
## Features
- Uniform API for enabling and controlling dark mode across all platforms. 👏
- Automatic dark mode detection (in systems that support dark mode). 👀
- Support for user dark mode switching. ☀️🌛
- Support for custom dark mode preference storage. 💾
- Updates the status bar to match the dark mode, even on Android. 🚀
- Custom status bar colors on Android. 🌈
- Register listeners for system dark mode changes. 🔥
- Extensive documentation. 📚## Installation
In your app:
```shell
pnpm add @aparajita/capacitor-dark-mode
```## Configuration
Once the plugin is installed, you need to:
- Provide a dark mode in your CSS if using Ionic < 8, or `import '@ionic/vue/css/palettes/dark.class.css'` in Ionic 8+.
- Initialize the plugin.### Dark mode CSS
This plugin adds or removes a CSS class to the `html` element when necessary. By default, the class is `.dark` on Ionic < 8 and `.ion-palette-dark` on Ionic 8+, but you can configure it to be whatever you want.
> 👉🏽 **Note:** If you are using Tailwind’s [dark mode support](https://tailwindcss.com/docs/dark-mode#toggling-dark-mode-manually), set `darkMode: 'class'` in your Tailwind config file.
It is up to you to configure your CSS to actually implement dark mode when that class is present. A good place to start is the standard [Ionic dark mode](https://ionicframework.com/docs/theming/dark-mode#ionic-dark-theme), which relies on the CSS variables that control Ionic component appearance.
If you have an existing CSS dark theme which relies on `prefers-color-scheme`, you should remove all `@media (prefers-color-scheme: dark)` rules and instead use `html.dark` (or simply `.dark`) as the dark mode selector. There is NO need to duplicate the dark mode in both a `@media (prefers-color-scheme: dark)` block and a `html.dark` block. That’s one of the advantages of using this plugin!
```css
/* Remove all prefers-color-scheme selectors! */
@media (prefers-color-scheme: dark) {
html {
--ion-color-primary: #428cff;
/* ... */
}
}/* Replace with this */
html.dark {
--ion-color-primary: #428cff;
/* ... */
}
```### Plugin configuration
If you are using the default dark mode CSS class and you don’t allow the user to manually set light or dark mode — and thus don’t need to store a preference — you are all set! The plugin does all of the hard work for you.
If you are using a dark mode CSS class other than the default, you need to configure the plugin. You will want to do this just before the app is mounted to avoid any visual glitches. For example, if your app uses a dark mode CSS class of `.dark-mode`, you would configure the plugin like this in a Vue-based Ionic app:
**main.ts**
```typescript
const app = createApp(App).use(IonicVue, config).use(router)router
.isReady()
.then(() => {
// configure() is a synonym for init()
DarkMode.init({ cssClass: 'dark-mode' })
.then(() => {
app.mount('#app')
})
.catch(console.error)
})
.catch(console.error)
```Use the equivalent in a React or Angular-based Ionic app.
> 👉🏽 **Note:** Using a custom dark mode class will not work on Ionic 8+ if you are importing `@ionic/vue/css/palettes/dark.class.css` You must use the default (`.ion-palette-dark`) in that case.
#### Custom preference storage
If you want to store the user’s dark mode preference in a custom location (such as `localStorage`), you must create a getter function that returns the preference and a setter that stores the preference, and pass those functions to the `init` or `configure` method.
**prefs.ts**
```typescript
import type { DarkModeGetterResult } from '@aparajita/capacitor-dark-mode'
import { DarkModeAppearance } from '@aparajita/capacitor-dark-mode'const kDarkModePref = 'dark-mode'
export function getAppearancePref(): DarkModeGetterResult {
return localStorage.getItem(kDarkModePref)
}export function setAppearancePref(appearance: DarkModeAppearance) {
localStorage.setItem(kDarkModePref, appearance)
}
```**main.ts**
```typescript
import { getAppearancePref, setAppearancePref } from './prefs'router
.isReady()
.then(() => {
DarkMode.init({
cssClass: 'dark-mode',
getter: getAppearancePref,
setter: setAppearancePref,
})
.then(() => {
app.mount('#app')
})
.catch(console.error)
})
.catch(console.error)
```The example above uses a synchronous function, but you may also use an async getter that returns a Promise, so there are no constraints on how or where you store the preference.
#### Android status bar customization
On Android, there are several additional options you can pass to `init()/configure()` that control what happens to the status bar when dark mode is toggled.
**syncStatusBar**
If `syncStatusBar` is `true`, the status bar will be updated to match the dark mode. This is the default behavior.**statusBarBackgroundVariable**
When `syncStatusBar` is `true`, by default the status bar background will set to the value of the `--background` CSS variable on the `ion-content` element, which is defined by `ion-content` as:```css
ion-content {
/*
The stock Ionic theme sets --ion-background-color
in dork mode.
*/
--background: var(--ion-background-color, #fff);
}
```If you want to use a different color for the status bar, you can set `statusBarBackgroundVariable` to the name of a different CSS variable. You can then set that variable accordingly in your CSS.
If the value of the variable is not a valid 3 or 6-digit '#'-prefixed hex color, no change is made.
**statusBarStyleGetter**
When `syncStatusBar` is `true` and a valid background color is set, by default the status bar style will be set according to the luminance of the background color:```typescript
// Default threshold is 0.5
const statusBarStyle = isDarkColor(color) ? Style.Dark : Style.Light
```If you want to use a different style, you can set `statusBarStyleGetter` to a function that returns the style to use. The function will be called with the current `Style` (based on the appearance setting, not the background color) and the status bar background color, and should return the `Style` that the status bar should be set to.
For example, you could use `isDarkColor()` (which is exported by the plugin) with a different threshold:
```typescript
import { Style } from '@capacitor/status-bar'const statusBarStyleGetter = (style?: Style, color?: string) => {
if (color) {
const isDark = isDarkColor(color, 0.4)
return isDark ? Style.Dark : Style.Light
}return style
}
```> 👉🏽 **Note:** The getter is also called when `syncStatusBar` is `'textOnly'`.
## Usage
I could spend a lot of time explaining detailed usage, but perhaps the best explanation is a full example that uses the entire plugin API and shows how to handle user dark mode preference changes. Check out the demo app [here](https://github.com/aparajita/capacitor-dark-mode-demo). You will especially want to look at [`prefs.ts`](https://github.com/aparajita/capacitor-dark-mode-demo/blob/main/src/prefs.ts) and [`DarkModeDemo.vue`](https://github.com/aparajita/capacitor-dark-mode-demo/blob/main/src/components/DarkModeDemo.vue).
## API
- [`init(...)`](#init)
- [`configure(...)`](#configure)
- [`isDarkMode()`](#isdarkmode)
- [`setNativeDarkModeListener(...)`](#setnativedarkmodelistener)
- [`addAppearanceListener(...)`](#addappearancelistener)
- [`update(...)`](#update)
- [Interfaces](#interfaces)
- [Type Aliases](#type-aliases)
- [Enums](#enums)### init(...)
```typescript
init(options?: DarkModeOptions) => Promise
```Initializes the plugin and optionally configures the dark mode class and getter used to retrieve the current dark mode state. This should be done BEFORE the app is mounted but AFTER the dom is defined (e.g. at the end of the <body>) to avoid a flash of the wrong mode.
| Param | Type |
| :------ | :--------------------------------------------- |
| options | DarkModeOptions |---
### configure(...)
```typescript
configure(options?: DarkModeOptions) => Promise
```A synonym for `init`.
| Param | Type |
| :------ | :--------------------------------------------- |
| options | DarkModeOptions |---
### isDarkMode()
```typescript
isDarkMode() => Promise
```web: Returns the result of the `prefers-color-scheme: dark` media query.
native: Returns whether the system is currently in dark mode.**Returns:** Promise<IsDarkModeResult>
---
### setNativeDarkModeListener(...)
```typescript
setNativeDarkModeListener(options: Record, callback: DarkModeListener) => Promise
```| Param | Type |
| :------- | :-------------------------------------------------- |
| options | Record<string, unknown> |
| callback | DarkModeListener |**Returns:** Promise<string>
---
### addAppearanceListener(...)
```typescript
addAppearanceListener(listener: DarkModeListener) => Promise
```Adds a listener that will be called whenever the system appearance changes, whether or not the system appearance matches your current appearance. The listener is called AFTER the dark mode class and status bar are updated by the plugin. The listener will be called with `DarkModeListenerData` indicating if the current system appearance is dark.
The returned handle contains a `remove` function which you should be sure to call when the listener is no longer needed, for example when a component is unmounted (which happens a lot with HMR). Otherwise there will be a memory leak and multiple listeners executing the same function.| Param | Type |
| :------- | :----------------------------------------------- |
| listener | DarkModeListener |**Returns:** Promise<DarkModeListenerHandle>
---
### update(...)
```typescript
update(data?: DarkModeListenerData) => Promise
```Adds or removes the dark mode class on the html element depending on the dark mode state. You do NOT need to call this when the system appearance changes.
If you are manually setting the appearance and you have specified a getter function, you should call this method AFTER the value returned by the configured getter changes.
Returns the current appearance.| Param | Type |
| :---- | :------------------------------------------------------- |
| data | DarkModeListenerData |**Returns:** Promise<DarkModeAppearance>
---
### Interfaces
#### DarkModeOptions
The options passed to `configure`.
| Prop | Type | Description |
| :-------------------------- | :--------------------------------------------------------- | :------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
| cssClass | string | The CSS class name to use to toggle dark mode. |
| getter | DarkModeGetter | If set, this function will be called to retrieve the current dark mode state instead of `isDarkMode`. For example, you might want to let the user set dark/light mode manually and store that preference somewhere. If the function wants to signal that no value can be retrieved, it should return null or undefined, in which case `isDarkMode` will be used.
If you are not providing any storage of the dark mode state, don't pass this in the options. |
| setter | DarkModeSetter | If set, this function will be called to set the current dark mode state when `update` is called. For example, you might want to let the user set dark/light mode manually and store that preference somewhere, such as localStorage. |
| disableTransitions | boolean | If true, the plugin will automatically disable all transitions when dark mode is toggled. This is to prevent different elements from switching between light and dark mode at different rates. <ion-item>, for example, by default has a transition on all of its properties.
Set this to false if you want to handle transitions yourself. |
| syncStatusBar | DarkModeSyncStatusBar | Android only
If `statusBarStyleGetter` is set, this option is unused.
If true, on Android the status bar background and content will be synced with the current `DarkModeAppearance`.
If 'textOnly', on Android only the status bar content will be synced with the current `DarkModeAppearance`: a light color when the appearance is dark and vice versa.
On iOS this option is not used, the status bar background is synced with dark mode by the system. |
| statusBarBackgroundVariable | string | Android only
If set, this CSS variable will be used instead of '--background' to set the status bar background color. |
| statusBarStyleGetter | StatusBarStyleGetter | Android only
If set, and `syncStatusBar` is true, this function will be called to retrieve the current status bar style instead of basing it on the dark mode. If the function wants to signal that no value can be retrieved, it should return a falsey value, in which case the current appearance will be used to determine the style. |#### IsDarkModeResult
Result returned by `isDarkMode`.
| Prop | Type |
| :--- | :------ |
| dark | boolean |#### DarkModeListenerData
Your appearance listener callback will receive this data, indicating whether the system is in dark mode or not.
| Prop | Type |
| :--- | :------ |
| dark | boolean |#### DarkModeListenerHandle
When you call `addAppearanceListener`, you get back a handle that you can use to remove the listener. See [addAppearanceListener](#addappearancelistener) for more details.
| Method | Signature |
| :--------- | :------------ |
| **remove** | () => void |### Type Aliases
#### DarkModeGetter
The type of your appearance getter function.
(): DarkModeGetterResult | Promise<DarkModeGetterResult>
#### DarkModeGetterResult
Your appearance getter function should return (directly or as a Promise) either:
- A DarkModeAppearance to signify that is the appearance you want
- null or undefined to signify the system appearance should be used
DarkModeAppearance | null
|#### DarkModeSetter
The type of your appearance setter function.
(appearance: DarkModeAppearance): void | Promise<void>
#### DarkModeSyncStatusBar
Possible values for the syncStatusBar option.
boolean | 'textOnly'
#### StatusBarStyleGetter
The type of your status bar style getter function.
(style?: Style, backgroundColor?: string): StatusBarStyleGetterResult | Promise<StatusBarStyleGetterResult>
#### StatusBarStyleGetterResult
Your style getter function should return (directly or as a Promise) either:
- A Style to signify that is the style you want
- null or undefined to signify the default behavior should be used
Style | null
|#### Record
Construct a type with a set of properties K of type T
{
[P in K]: T;
}#### DarkModeListener
The type of your appearance listener callback.
(data: DarkModeListenerData): void
### Enums
#### DarkModeAppearance
| Members | Value |
| :------ | :------- |
| dark | 'dark' |
| light | 'light' |
| system | 'system' |#### Style
| Members | Value | Description |
| :------ | :-------- | :----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| Dark | "DARK" | Light text for dark backgrounds. |
| Light | "LIGHT" | Dark text for light backgrounds. |
| Default | "DEFAULT" | The style is based on the device appearance. If the device is using Dark mode, the statusbar text will be light. If the device is using Light mode, the statusbar text will be dark. On Android the default will be the one the app was launched with. |