support theming
This commit is contained in:
parent
48b8f132d6
commit
fb120750ab
4 changed files with 58 additions and 9 deletions
39
src/components/shared/theme-picker.vue
Normal file
39
src/components/shared/theme-picker.vue
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { onMounted, ref } from 'vue'
|
||||||
|
|
||||||
|
import { injectStylesheet } from 'src/utilities/dom'
|
||||||
|
import { storage } from 'src/utilities/fetch'
|
||||||
|
import { useRouteStore } from 'src/routes'
|
||||||
|
|
||||||
|
const routeStore = useRouteStore()
|
||||||
|
const globalConfig = routeStore._globals
|
||||||
|
const options = ref(globalConfig.themes)
|
||||||
|
const currentTheme = ref('')
|
||||||
|
let node: HTMLLinkElement
|
||||||
|
|
||||||
|
const onThemeChosen = (event: Event) => {
|
||||||
|
storage.write(`${globalConfig.id}::currentTheme`, currentTheme.value)
|
||||||
|
node.setAttribute('href', globalConfig.themes[currentTheme.value].url)
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
currentTheme.value = storage.read(`${globalConfig.id}::currentTheme`) || Object.keys(globalConfig.themes)[0]
|
||||||
|
node = injectStylesheet(globalConfig.themes[currentTheme.value].url, 'theme-stylesheet')
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template lang="pug">
|
||||||
|
.theme-picker
|
||||||
|
select(
|
||||||
|
v-model='currentTheme'
|
||||||
|
@change='onThemeChosen($event)'
|
||||||
|
)
|
||||||
|
option(
|
||||||
|
v-for='(option, key) in options'
|
||||||
|
:value='key'
|
||||||
|
) {{ option.displayName|| option }}
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped lang="sass">
|
||||||
|
|
||||||
|
</style>
|
11
src/main.vue
11
src/main.vue
|
@ -3,12 +3,14 @@ import { onMounted, ref } from 'vue'
|
||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
|
|
||||||
import { getCurrentRoute } from 'src/utilities/vuetils'
|
import { getCurrentRoute } from 'src/utilities/vuetils'
|
||||||
|
import { injectStylesheet } from 'src/utilities/dom'
|
||||||
import { storage } from './utilities/fetch'
|
import { storage } from './utilities/fetch'
|
||||||
import { useRouteStore } from 'src/routes'
|
import { useRouteStore } from 'src/routes'
|
||||||
|
|
||||||
import type { WarningModal } from 'content/routes.js'
|
import type { WarningModal } from 'content/routes.js'
|
||||||
|
|
||||||
import HeaderLink from 'src/components/shared/header-link.vue'
|
import HeaderLink from 'src/components/shared/header-link.vue'
|
||||||
|
import ThemePicker from 'src/components/shared/theme-picker.vue'
|
||||||
import WarningPrompt from 'src/components/shared/warning-prompt.vue'
|
import WarningPrompt from 'src/components/shared/warning-prompt.vue'
|
||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
|
@ -40,14 +42,6 @@ const determineWarning = () => {
|
||||||
: routeConfig.warning
|
: routeConfig.warning
|
||||||
}
|
}
|
||||||
|
|
||||||
const injectStylesheet = (url: string, className: string) => {
|
|
||||||
const newElement = document.createElement('link')
|
|
||||||
newElement.setAttribute('rel', 'stylesheet')
|
|
||||||
newElement.setAttribute('href', url)
|
|
||||||
newElement.classList.add(className)
|
|
||||||
document.head.appendChild(newElement)
|
|
||||||
}
|
|
||||||
|
|
||||||
const determineStylesheets = (stylesheetUrls?: string[]) => {
|
const determineStylesheets = (stylesheetUrls?: string[]) => {
|
||||||
const staleStylesheets = document.head.querySelectorAll('link.page-stylesheet[rel="stylesheet"]')
|
const staleStylesheets = document.head.querySelectorAll('link.page-stylesheet[rel="stylesheet"]')
|
||||||
staleStylesheets.forEach(stylesheet => {
|
staleStylesheets.forEach(stylesheet => {
|
||||||
|
@ -105,6 +99,7 @@ onMounted(async () => {
|
||||||
v-for='entry in globalConfig.header'
|
v-for='entry in globalConfig.header'
|
||||||
:entry='entry'
|
:entry='entry'
|
||||||
)
|
)
|
||||||
|
ThemePicker
|
||||||
main(
|
main(
|
||||||
v-if='ready'
|
v-if='ready'
|
||||||
)
|
)
|
||||||
|
|
|
@ -71,7 +71,7 @@ export const useRouteStore = defineStore('routeStore', {
|
||||||
},
|
},
|
||||||
rememberRouteWarning(route: string) {
|
rememberRouteWarning(route: string) {
|
||||||
this._routesAlreadyWarned[route] = true
|
this._routesAlreadyWarned[route] = true
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,21 @@ import { type DateRange } from 'src/types/shared/dateRange'
|
||||||
|
|
||||||
export const deepCopy = rfdc()
|
export const deepCopy = rfdc()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injects a stylesheet link into th ehead of the document
|
||||||
|
* @param url the url of the stylesheet
|
||||||
|
* @param className the classname to give the stylesheet
|
||||||
|
*/
|
||||||
|
export const injectStylesheet = (url: string, className: string) => {
|
||||||
|
const newElement = document.createElement('link')
|
||||||
|
newElement.setAttribute('rel', 'stylesheet')
|
||||||
|
newElement.setAttribute('href', url)
|
||||||
|
newElement.classList.add(className)
|
||||||
|
document.head.appendChild(newElement)
|
||||||
|
|
||||||
|
return newElement
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Runs element.querySelector for every selector in the order they were passed in until an element is found
|
* Runs element.querySelector for every selector in the order they were passed in until an element is found
|
||||||
* @param element the element to query children from
|
* @param element the element to query children from
|
||||||
|
|
Loading…
Add table
Reference in a new issue