2021-11-22 22:12:54 +01:00
|
|
|
import {computed, watch, readonly} from 'vue'
|
2022-04-11 22:08:28 +02:00
|
|
|
import {useStorage, createSharedComposable, usePreferredColorScheme, tryOnMounted} from '@vueuse/core'
|
|
|
|
import type {BasicColorSchema} from '@vueuse/core'
|
2021-11-22 22:12:54 +01:00
|
|
|
|
|
|
|
const STORAGE_KEY = 'color-scheme'
|
|
|
|
|
2022-01-16 22:24:51 +01:00
|
|
|
const DEFAULT_COLOR_SCHEME_SETTING: BasicColorSchema = 'light'
|
2021-11-22 22:12:54 +01:00
|
|
|
|
|
|
|
const CLASS_DARK = 'dark'
|
|
|
|
const CLASS_LIGHT = 'light'
|
|
|
|
|
|
|
|
// This is built upon the vueuse useDark
|
|
|
|
// Main differences:
|
|
|
|
// - usePreferredColorScheme
|
|
|
|
// - doesn't allow setting via the `isDark` ref.
|
|
|
|
// - instead the store is exposed
|
|
|
|
// - value is synced via `createSharedComposable`
|
|
|
|
// https://github.com/vueuse/vueuse/blob/main/packages/core/useDark/index.ts
|
|
|
|
export const useColorScheme = createSharedComposable(() => {
|
2022-01-16 22:24:51 +01:00
|
|
|
const store = useStorage<BasicColorSchema>(STORAGE_KEY, DEFAULT_COLOR_SCHEME_SETTING)
|
2021-11-22 22:12:54 +01:00
|
|
|
|
|
|
|
const preferredColorScheme = usePreferredColorScheme()
|
|
|
|
|
|
|
|
const isDark = computed<boolean>(() => {
|
|
|
|
if (store.value !== 'auto') {
|
|
|
|
return store.value === 'dark'
|
|
|
|
}
|
|
|
|
|
|
|
|
const autoColorScheme = preferredColorScheme.value === 'no-preference'
|
|
|
|
? DEFAULT_COLOR_SCHEME_SETTING
|
|
|
|
: preferredColorScheme.value
|
|
|
|
return autoColorScheme === 'dark'
|
|
|
|
})
|
|
|
|
|
|
|
|
function onChanged(v: boolean) {
|
|
|
|
const el = window?.document.querySelector('html')
|
|
|
|
el?.classList.toggle(CLASS_DARK, v)
|
|
|
|
el?.classList.toggle(CLASS_LIGHT, !v)
|
|
|
|
}
|
|
|
|
|
|
|
|
watch(isDark, onChanged, { flush: 'post' })
|
|
|
|
|
|
|
|
tryOnMounted(() => onChanged(isDark.value))
|
|
|
|
|
|
|
|
return {
|
|
|
|
store,
|
|
|
|
isDark: readonly(isDark),
|
|
|
|
}
|
|
|
|
})
|