2022-04-05 11:50:15 +08:00

57 lines
1.6 KiB
TypeScript

import { usePreferredDark, useStorage } from '@vueuse/core'
import { computed, inject, onMounted, onUnmounted, provide, watch } from 'vue'
import type { InjectionKey, WritableComputedRef } from 'vue'
import { useThemeLocaleData } from './themeData'
export type DarkModeRef = WritableComputedRef<boolean>
export const darkModeSymbol: InjectionKey<DarkModeRef> = Symbol(
__VUEPRESS_DEV__ ? 'darkMode' : ''
)
export const useDarkMode = (): DarkModeRef => {
const isDarkMode = inject(darkModeSymbol)
if (!isDarkMode) {
throw new Error('useDarkMode() is called without provider')
}
return isDarkMode
}
export const setupDarkMode = (): void => {
const themeLocale = useThemeLocaleData()
const isDarkPreferred = usePreferredDark()
const darkStorage = useStorage('vuepress-color-scheme', 'auto')
const isDarkMode = computed<boolean>({
get() {
if (!themeLocale.value.darkMode) {
return false
}
if (darkStorage.value === 'auto') {
return isDarkPreferred.value
}
return darkStorage.value === 'dark'
},
set(val) {
if (val === isDarkPreferred.value) {
darkStorage.value = 'auto'
} else {
darkStorage.value = val ? 'dark' : 'light'
}
},
})
provide(darkModeSymbol, isDarkMode)
updateHtmlDarkClass(isDarkMode)
}
export const updateHtmlDarkClass = (isDarkMode: DarkModeRef): void => {
const update = (value = isDarkMode.value): void => {
const htmlEl = window?.document.querySelector('html')
htmlEl?.classList.toggle('dark', value)
}
onMounted(() => {
watch(isDarkMode, update, { immediate: true })
})
onUnmounted(() => update())
}