mirror of
https://github.com/elk-zone/elk.git
synced 2025-01-12 14:57:21 +03:00
69 lines
2.3 KiB
Vue
69 lines
2.3 KiB
Vue
<script setup lang="ts">
|
|
import type { ThemeColors } from '~/composables/settings'
|
|
import { THEME_COLORS } from '~/constants'
|
|
|
|
const themes = await import('~/constants/themes.json').then((r) => {
|
|
const map = new Map<'dark' | 'light', [string, ThemeColors][]>([['dark', []], ['light', []]])
|
|
const themes = r.default as [string, ThemeColors][]
|
|
for (const [key, theme] of themes) {
|
|
map.get('dark')!.push([key, theme])
|
|
map.get('light')!.push([key, {
|
|
...theme,
|
|
'--c-primary': `color-mix(in srgb, ${theme['--c-primary']}, black 25%)`,
|
|
}])
|
|
}
|
|
return map
|
|
})
|
|
|
|
const settings = useUserSettings()
|
|
|
|
const media = useMediaQuery('(prefers-color-scheme: dark)')
|
|
|
|
const colorMode = useColorMode()
|
|
|
|
const useThemes = shallowRef<[string, ThemeColors][]>([])
|
|
|
|
watch(() => colorMode.preference, (cm) => {
|
|
const dark = cm === 'dark' || (cm === 'system' && media.value)
|
|
const newThemes = dark ? themes.get('dark')! : themes.get('light')!
|
|
const key = settings.value.themeColors?.['--theme-color-name'] || THEME_COLORS.defaultTheme
|
|
for (const [k, theme] of newThemes) {
|
|
if (k === key) {
|
|
settings.value.themeColors = theme
|
|
break
|
|
}
|
|
}
|
|
useThemes.value = newThemes
|
|
}, { immediate: true, flush: 'post' })
|
|
|
|
const currentTheme = computed(() => settings.value.themeColors?.['--theme-color-name'] || THEME_COLORS.defaultTheme)
|
|
|
|
function updateTheme(theme: ThemeColors) {
|
|
settings.value.themeColors = theme
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<section space-y-2>
|
|
<h2 id="interface-tc" font-medium>
|
|
{{ $t('settings.interface.theme_color') }}
|
|
</h2>
|
|
<div flex="~ gap4 wrap" p2 role="group" aria-labelledby="interface-tc">
|
|
<button
|
|
v-for="[key, theme] in useThemes" :key="key"
|
|
:style="{
|
|
'--rgb-primary': theme['--rgb-primary'],
|
|
'background': theme['--c-primary'],
|
|
'--local-ring-color': theme['--c-primary'],
|
|
}"
|
|
type="button"
|
|
:class="currentTheme === theme['--theme-color-name'] ? 'ring-2' : 'scale-90'"
|
|
:aria-pressed="currentTheme === theme['--theme-color-name'] ? 'true' : 'false'"
|
|
:title="theme['--theme-color-name']"
|
|
w-8 h-8 rounded-full transition-all
|
|
ring="$local-ring-color offset-3 offset-$c-bg-base"
|
|
@click="updateTheme(theme)"
|
|
/>
|
|
</div>
|
|
</section>
|
|
</template>
|