content warning prompt for routes
This commit is contained in:
parent
8225d594e4
commit
3b2273ed25
4 changed files with 175 additions and 12 deletions
67
src/components/shared/warning-prompt.vue
Normal file
67
src/components/shared/warning-prompt.vue
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
|
||||||
|
import { storage } from 'src/utilities/fetch'
|
||||||
|
|
||||||
|
import type { WarningModal } from 'content/routes.js'
|
||||||
|
|
||||||
|
import EmbedableContent from './embedable-content.vue'
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
storageId: string
|
||||||
|
warning: WarningModal
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const emits = defineEmits<{
|
||||||
|
(e: 'acknowledged'): void
|
||||||
|
(e: 'canceled'): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const router = useRouter()
|
||||||
|
const rememberChecked = defineModel('rememberChecked', { type: Boolean })
|
||||||
|
|
||||||
|
const onLeave = (event: Event) => {
|
||||||
|
event.preventDefault()
|
||||||
|
router.push({ path: props.warning?.leave?.url || '/' })
|
||||||
|
}
|
||||||
|
|
||||||
|
const onAcknowledge = (event: Event) => {
|
||||||
|
event.preventDefault()
|
||||||
|
console.log(rememberChecked.value)
|
||||||
|
if (rememberChecked.value) {
|
||||||
|
storage.write(`${props.storageId}::rememberWarning`, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
emits('acknowledged')
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template lang="pug">
|
||||||
|
.warning-prompt
|
||||||
|
EmbedableContent(
|
||||||
|
:content='warning.prompt'
|
||||||
|
)
|
||||||
|
.actions
|
||||||
|
button(
|
||||||
|
@click='onLeave($event)'
|
||||||
|
)
|
||||||
|
span {{ warning.leave.text }}
|
||||||
|
button(
|
||||||
|
@click='onAcknowledge($event)'
|
||||||
|
)
|
||||||
|
span {{ warning.acknowledge }}
|
||||||
|
.input.labeled-checkbox
|
||||||
|
label(
|
||||||
|
for='warning-prompt-checkbox'
|
||||||
|
) {{ warning.remember }}
|
||||||
|
input(
|
||||||
|
type='checkbox'
|
||||||
|
name='warning-prompt-checkbox'
|
||||||
|
id='warning-prompt-checkbox'
|
||||||
|
v-model='rememberChecked'
|
||||||
|
)
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped lang="sass">
|
||||||
|
|
||||||
|
</style>
|
27
src/content-env.d.ts
vendored
27
src/content-env.d.ts
vendored
|
@ -1,4 +1,17 @@
|
||||||
declare module 'content/routes.js' {
|
declare module 'content/routes.js' {
|
||||||
|
/**
|
||||||
|
* Defines the structure of the warning modal that shows when a route has a content warning of some sort.
|
||||||
|
*/
|
||||||
|
export type WarningModal = {
|
||||||
|
prompt: string
|
||||||
|
leave: {
|
||||||
|
url: string,
|
||||||
|
text: string,
|
||||||
|
},
|
||||||
|
acknowledge: string,
|
||||||
|
remember: string,
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the available `views` that can be used to set up routes.
|
* Defines the available `views` that can be used to set up routes.
|
||||||
* Each `Template` has different configuration options
|
* Each `Template` has different configuration options
|
||||||
|
@ -16,6 +29,7 @@ declare module 'content/routes.js' {
|
||||||
stylesheetUrls: string[]
|
stylesheetUrls: string[]
|
||||||
template: Template
|
template: Template
|
||||||
title: string
|
title: string
|
||||||
|
warning: boolean | WarningModal
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -88,6 +102,19 @@ declare module 'content/routes.js' {
|
||||||
} | {
|
} | {
|
||||||
children: HeaderEntry[]
|
children: HeaderEntry[]
|
||||||
})
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines global values for the site.
|
||||||
|
*/
|
||||||
|
type SiteGlobals = {
|
||||||
|
id: string
|
||||||
|
warning: WarningModal
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Global values the site uses.
|
||||||
|
*/
|
||||||
|
const siteGlobals: SiteGlobals
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The header the app uses.
|
* The header the app uses.
|
||||||
|
|
68
src/main.vue
68
src/main.vue
|
@ -3,26 +3,51 @@ 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 { storage } from './utilities/fetch'
|
||||||
import { useRouteStore } from 'src/routes'
|
import { useRouteStore } from 'src/routes'
|
||||||
|
|
||||||
import HeaderLink from 'src/components/shared/header-link.vue'
|
import type { WarningModal } from 'content/routes.js'
|
||||||
|
|
||||||
const ready = ref(false)
|
import HeaderLink from 'src/components/shared/header-link.vue'
|
||||||
|
import WarningPrompt from 'src/components/shared/warning-prompt.vue'
|
||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const routeStore = useRouteStore()
|
const routeStore = useRouteStore()
|
||||||
const headerConfig = routeStore._header
|
const headerConfig = routeStore._header
|
||||||
|
const globalConfig = routeStore._globals
|
||||||
|
|
||||||
const refresh = async () => {
|
let currentRoute = getCurrentRoute()
|
||||||
const currentRoute = getCurrentRoute()
|
let routeConfig = routeStore._routes[currentRoute.path]
|
||||||
const routeConfig = routeStore._routes[currentRoute.path]
|
let rememberWarning = false
|
||||||
|
|
||||||
|
const ready = ref(false)
|
||||||
|
const acknowledged = ref(false)
|
||||||
|
const storageId = ref('')
|
||||||
|
const warning = ref({} as WarningModal)
|
||||||
|
|
||||||
|
const determineWarning = () => {
|
||||||
|
if (!routeConfig.warning || routeStore.doesRouteRememberWarning(currentRoute.path)) {
|
||||||
|
acknowledged.value = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
rememberWarning = storage.read(`${storageId.value}::rememberWarning`) || false
|
||||||
|
if (rememberWarning) {
|
||||||
|
acknowledged.value = true
|
||||||
|
routeStore.rememberRouteWarning(currentRoute.path)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
warning.value = routeConfig.warning === true
|
||||||
|
? globalConfig.warning
|
||||||
|
: routeConfig.warning
|
||||||
|
}
|
||||||
|
|
||||||
|
const determineStylesheets = (stylesheetUrls?: string[]) => {
|
||||||
const staleStylesheets = document.head.querySelectorAll('link[rel="stylesheet"]')
|
const staleStylesheets = document.head.querySelectorAll('link[rel="stylesheet"]')
|
||||||
staleStylesheets.forEach(stylesheet => {
|
staleStylesheets.forEach(stylesheet => {
|
||||||
document.head.removeChild(stylesheet)
|
document.head.removeChild(stylesheet)
|
||||||
})
|
})
|
||||||
|
|
||||||
routeConfig?.stylesheetUrls?.forEach(stylesheet => {
|
stylesheetUrls?.forEach(stylesheet => {
|
||||||
const newElement = document.createElement('link')
|
const newElement = document.createElement('link')
|
||||||
newElement.setAttribute('rel', 'stylesheet')
|
newElement.setAttribute('rel', 'stylesheet')
|
||||||
newElement.setAttribute('href', stylesheet)
|
newElement.setAttribute('href', stylesheet)
|
||||||
|
@ -30,12 +55,29 @@ const refresh = async () => {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const refresh = async () => {
|
||||||
|
ready.value = false
|
||||||
|
acknowledged.value = false
|
||||||
|
currentRoute = getCurrentRoute()
|
||||||
|
routeConfig = routeStore._routes[currentRoute.path]
|
||||||
|
storageId.value = `${globalConfig.id}::${currentRoute.path}`
|
||||||
|
|
||||||
|
determineWarning()
|
||||||
|
determineStylesheets(routeConfig.stylesheetUrls)
|
||||||
|
|
||||||
|
ready.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
const onAcknowledgedWarning = () => {
|
||||||
|
acknowledged.value = true
|
||||||
|
routeStore.rememberRouteWarning(currentRoute.path)
|
||||||
|
}
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
await refresh()
|
await refresh()
|
||||||
router.afterEach(async (to, from) => {
|
router.afterEach(async (to, from) => {
|
||||||
await refresh()
|
await refresh()
|
||||||
})
|
})
|
||||||
ready.value = true
|
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -49,11 +91,19 @@ onMounted(async () => {
|
||||||
v-for='entry in headerConfig'
|
v-for='entry in headerConfig'
|
||||||
:entry='entry'
|
:entry='entry'
|
||||||
)
|
)
|
||||||
#main-entry(
|
main(
|
||||||
v-if='ready'
|
v-if='ready'
|
||||||
)
|
)
|
||||||
main
|
#main-entry(
|
||||||
|
v-if='acknowledged'
|
||||||
|
)
|
||||||
router-view
|
router-view
|
||||||
|
WarningPrompt(
|
||||||
|
v-else
|
||||||
|
:storageId='storageId'
|
||||||
|
:warning='warning'
|
||||||
|
@acknowledged='onAcknowledgedWarning()'
|
||||||
|
)
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped lang="sass">
|
<style scoped lang="sass">
|
||||||
|
|
|
@ -1,6 +1,16 @@
|
||||||
import { defineStore } from 'pinia'
|
import { defineStore } from 'pinia'
|
||||||
import { type RouteRecordRaw, type Router } from 'vue-router'
|
import { type RouteRecordRaw } from 'vue-router'
|
||||||
import { header, routes, type HeaderEntry, type ProjectListDefinition, type RouteDefinition, type Template, GalleryListDefinition } from 'content/routes.js'
|
import {
|
||||||
|
header,
|
||||||
|
routes,
|
||||||
|
siteGlobals,
|
||||||
|
type GalleryListDefinition,
|
||||||
|
type HeaderEntry,
|
||||||
|
type ProjectListDefinition,
|
||||||
|
type RouteDefinition,
|
||||||
|
type SiteGlobals,
|
||||||
|
type Template,
|
||||||
|
} from 'content/routes.js'
|
||||||
|
|
||||||
const markdownBody = () => import ('./views/markdown.vue')
|
const markdownBody = () => import ('./views/markdown.vue')
|
||||||
const projectListBody = () => import ('./views/project-list.vue')
|
const projectListBody = () => import ('./views/project-list.vue')
|
||||||
|
@ -52,9 +62,17 @@ export const useRouteStore = defineStore('routeStore', {
|
||||||
state: () => ({
|
state: () => ({
|
||||||
_header: [] as HeaderEntry[],
|
_header: [] as HeaderEntry[],
|
||||||
_routes: {} as Record<string, RouteRecordRaw & RouteDefinition>,
|
_routes: {} as Record<string, RouteRecordRaw & RouteDefinition>,
|
||||||
|
_globals: {} as SiteGlobals,
|
||||||
|
|
||||||
|
_routesAlreadyWarned: {} as Record<string, boolean>
|
||||||
}),
|
}),
|
||||||
actions: {
|
actions: {
|
||||||
|
doesRouteRememberWarning(route: string) {
|
||||||
|
return this._routesAlreadyWarned[route]
|
||||||
|
},
|
||||||
|
rememberRouteWarning(route: string) {
|
||||||
|
this._routesAlreadyWarned[route] = true
|
||||||
|
}
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -73,6 +91,7 @@ export const initializeRouteStore = (routerRoutes: readonly RouteRecordRaw[]) =>
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
routeStore._header = header
|
routeStore._header = header
|
||||||
|
routeStore._globals = siteGlobals
|
||||||
}
|
}
|
||||||
|
|
||||||
export type RouteStoreDefinition = Omit<
|
export type RouteStoreDefinition = Omit<
|
||||||
|
|
Loading…
Add table
Reference in a new issue