get gallery functionality working again
- redefine gallery types - re-implement gallery list, tile, and view - no carousel yet for variants, but WithoutVariants works when url specified
This commit is contained in:
parent
07b37e0805
commit
8ac3ed16ff
9 changed files with 225 additions and 145 deletions
|
@ -6,17 +6,55 @@ import type { MediaEntry } from './shared'
|
||||||
export type GalleryEntryFormat =
|
export type GalleryEntryFormat =
|
||||||
| 'image'
|
| 'image'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This describes additional information about a gallery entry
|
* Defines properties shared among the different versions of a {@link GalleryEntry}
|
||||||
* to display when listing the entry on the gallery-list page.
|
*/
|
||||||
*/
|
export type GalleryEntrySharedProperties = {
|
||||||
export type GalleryEntry = {
|
|
||||||
/**
|
/**
|
||||||
* The kind of media the entry is
|
* The kind of media the entry is
|
||||||
*/
|
*/
|
||||||
format: GalleryEntryFormat
|
format: GalleryEntryFormat
|
||||||
|
} & MediaEntry
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This describes additional information about a gallery entry which does not contain any variants.
|
||||||
|
* It contains a URL to the media that the entry represents.
|
||||||
|
*/
|
||||||
|
export type GalleryEntryWithoutVariants = {
|
||||||
/**
|
/**
|
||||||
* URL to the markdown document of the gallery entry
|
* URL to the entry
|
||||||
*/
|
*/
|
||||||
url: string
|
url: string
|
||||||
} & MediaEntry
|
} & GalleryEntrySharedProperties
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This describes additional information about a gallery entry which contains variants.
|
||||||
|
* It contains a collection of ids which map to a gallery entry without variants.
|
||||||
|
*/
|
||||||
|
export type GalleryEntryWithVariants = {
|
||||||
|
/**
|
||||||
|
* The id of the entry from the {@link GalleryEntry variants} collection to display when the page first loads.
|
||||||
|
* If this is not defined, the first entry will be chosen instead.
|
||||||
|
*/
|
||||||
|
defaultVariant?: string
|
||||||
|
/**
|
||||||
|
* Variants are a collection of alternate media to display for the entry.
|
||||||
|
* This changes based on the {@link GalleryEntryFormat format} of the entry:
|
||||||
|
* - For music, this is treated like a tracklist for an album,
|
||||||
|
* with the {@link GalleryEntry.defaultVariant default} or selected variant being treated as the current track in the player
|
||||||
|
* - For all other media, this is put into a carousel,
|
||||||
|
* where the {@link GalleryEntry.defaultVariant default} or selected variant is displayed in the main viewer
|
||||||
|
*/
|
||||||
|
variants: Record<string, GalleryEntryWithoutVariants>
|
||||||
|
} & GalleryEntrySharedProperties
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This describes the overall structure of a gallery entry
|
||||||
|
* to display when listing the entry on the gallery-list page.
|
||||||
|
* Note that a {@link GalleryEntryWithVariants} will take priority
|
||||||
|
* over a {@link GalleryEntryWithoutVariants}
|
||||||
|
* if the parsed config results in one with fields from both types.
|
||||||
|
*/
|
||||||
|
export type GalleryEntry =
|
||||||
|
| GalleryEntryWithoutVariants
|
||||||
|
| GalleryEntryWithVariants
|
||||||
|
|
11
projects/frontend/src/templates/link.pug
Normal file
11
projects/frontend/src/templates/link.pug
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
mixin link
|
||||||
|
router-link.router-link.link(
|
||||||
|
v-if='isInternal'
|
||||||
|
:to='{ path: viewPath, query: { id: id } }'
|
||||||
|
)
|
||||||
|
block
|
||||||
|
a.link(
|
||||||
|
v-else
|
||||||
|
:href='href'
|
||||||
|
)
|
||||||
|
block
|
|
@ -27,7 +27,6 @@ type BlogDisplayedEntries = ResolvedListEntries<BlogEntry & {
|
||||||
isHidden?: boolean
|
isHidden?: boolean
|
||||||
}>
|
}>
|
||||||
|
|
||||||
const entryIds = ref([] as string[])
|
|
||||||
const entries = ref({} as BlogDisplayedEntries)
|
const entries = ref({} as BlogDisplayedEntries)
|
||||||
const ready = ref(false)
|
const ready = ref(false)
|
||||||
const currentRoute = getCurrentRoute()
|
const currentRoute = getCurrentRoute()
|
||||||
|
@ -57,7 +56,6 @@ onMounted(async () => {
|
||||||
let listConfig = await fetchAndParseYaml<BlogList>(routeConfig.config)
|
let listConfig = await fetchAndParseYaml<BlogList>(routeConfig.config)
|
||||||
const list = await fetchConfigsFromList<BlogEntry>(listConfig)
|
const list = await fetchConfigsFromList<BlogEntry>(listConfig)
|
||||||
config.value = listConfig
|
config.value = listConfig
|
||||||
entryIds.value = list.ids
|
|
||||||
entries.value = list.entries
|
entries.value = list.entries
|
||||||
document.title = routeConfig.fullTitle
|
document.title = routeConfig.fullTitle
|
||||||
ready.value = true
|
ready.value = true
|
||||||
|
@ -71,15 +69,15 @@ onMounted(async () => {
|
||||||
v-if='ready'
|
v-if='ready'
|
||||||
)
|
)
|
||||||
Transition(
|
Transition(
|
||||||
v-for='id in entryIds'
|
v-for='(entry, id) in entries'
|
||||||
)
|
)
|
||||||
BlogTile(
|
BlogTile(
|
||||||
v-if='!entries[id].isHidden || !config.removeFromView'
|
v-if='!entry.isHidden || !config.removeFromView'
|
||||||
:class='{ hidden: entries[id].isHidden && !config.removeFromView }'
|
:class='{ hidden: entry.isHidden && !config.removeFromView }'
|
||||||
:id='id'
|
:id='id'
|
||||||
:viewPath='viewPath'
|
:viewPath='viewPath'
|
||||||
:isInternal='true'
|
:isInternal='true'
|
||||||
:entry='entries[id]'
|
:entry='entry'
|
||||||
)
|
)
|
||||||
Transition
|
Transition
|
||||||
FilterPanel(
|
FilterPanel(
|
||||||
|
|
|
@ -17,7 +17,7 @@ const resolved = ref({} as BlogEntry)
|
||||||
const thumbnail = computed(() => resolved.value.thumbnail)
|
const thumbnail = computed(() => resolved.value.thumbnail)
|
||||||
const description = computed(() => marked.parse(resolved.value.description || ''))
|
const description = computed(() => marked.parse(resolved.value.description || ''))
|
||||||
const date = computed(() => !!resolved.value.date ? getFormattedDate(resolved.value.date) : null)
|
const date = computed(() => !!resolved.value.date ? getFormattedDate(resolved.value.date) : null)
|
||||||
const title = computed(() => marked.parse(resolved.value.title || ''))
|
const title = computed(() => marked.parse(resolved.value.title || props.id))
|
||||||
const href = computed(() => `${props.viewPath}?id=${props.id}`)
|
const href = computed(() => `${props.viewPath}?id=${props.id}`)
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { onMounted, ref } from 'vue'
|
import { onMounted, ref } from 'vue'
|
||||||
import type {
|
import type {
|
||||||
BlogList,
|
ListWithEntries,
|
||||||
BlogEntry,
|
BlogEntry,
|
||||||
BlogListDefinition,
|
BlogListDefinition,
|
||||||
RoutedWindow,
|
RoutedWindow,
|
||||||
|
@ -20,7 +20,7 @@ const emits = defineEmits<{
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const ready = ref(false)
|
const ready = ref(false)
|
||||||
const info = ref(null! as BlogEntry)
|
const resolved = ref({} as BlogEntry)
|
||||||
const content = ref('')
|
const content = ref('')
|
||||||
const currentRoute = getCurrentRoute()
|
const currentRoute = getCurrentRoute()
|
||||||
const routeStore = useRouteStore()
|
const routeStore = useRouteStore()
|
||||||
|
@ -28,15 +28,15 @@ const routeConfig = routeStore._routes[currentRoute.path.substring(0, currentRou
|
||||||
const routeSubConfig = routeStore._routes[currentRoute.path]
|
const routeSubConfig = routeStore._routes[currentRoute.path]
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
const config = await fetchAndParseYaml<BlogList>(routeConfig.config)
|
const config = await fetchAndParseYaml<ListWithEntries<BlogEntry>>(routeConfig.config)
|
||||||
info.value = await fetchConfigByIdFromList(config, currentRoute.query.id as string)
|
resolved.value = await fetchConfigByIdFromList(config, currentRoute.query.id as string)
|
||||||
const md = await fetchAndParseMarkdown(info.value.url)
|
const md = await fetchAndParseMarkdown(resolved.value.url)
|
||||||
content.value = md
|
content.value = md
|
||||||
document.title = routeSubConfig.fullTitle?.replace('$ENTRY', info.value.title)
|
document.title = routeSubConfig.fullTitle?.replace('$ENTRY', resolved.value.title)
|
||||||
routeStore.setBreadcrumbs(currentRoute, info.value.title)
|
routeStore.setBreadcrumbs(currentRoute, resolved.value.title)
|
||||||
window.routeConfig = {...routeConfig}
|
window.routeConfig = {...routeConfig}
|
||||||
window.routeSubConfig = {...routeSubConfig}
|
window.routeSubConfig = {...routeSubConfig}
|
||||||
window.routeContentConfig = {...info.value}
|
window.routeContentConfig = {...resolved.value}
|
||||||
|
|
||||||
ready.value = true
|
ready.value = true
|
||||||
emits('loaded')
|
emits('loaded')
|
||||||
|
|
|
@ -1,11 +1,76 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { onMounted, ref } from 'vue'
|
import { computed, onMounted, ref } from 'vue'
|
||||||
|
import type {
|
||||||
|
GalleryEntry,
|
||||||
|
GalleryListDefinition,
|
||||||
|
ListWithTags,
|
||||||
|
ResolvedListEntries,
|
||||||
|
} from '@goldenwere/mackenzii-types'
|
||||||
|
import { fetchAndParseYaml, fetchConfigsFromList, storage } from 'src/utilities/fetch'
|
||||||
|
import { getCurrentRoute } from 'src/utilities/vuetils'
|
||||||
|
import { useRouteStore } from 'src/routes'
|
||||||
|
|
||||||
|
import FilterPanel from 'src/components/shared/filter-panel.vue'
|
||||||
|
import GalleryTile from './gallery-tile.vue'
|
||||||
|
|
||||||
|
type GalleryList = ListWithTags<GalleryEntry>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A wrapper around {@link GalleryEntries} for the app's use only which adds additional fields
|
||||||
|
* in order for the app to effectively display the entries.
|
||||||
|
*/
|
||||||
|
type GalleryDisplayedEntries = ResolvedListEntries<GalleryEntry & {
|
||||||
|
/**
|
||||||
|
* specifies whether the entry is hidden by the tags selected by a visitor
|
||||||
|
*/
|
||||||
|
isHidden?: boolean
|
||||||
|
}>
|
||||||
|
|
||||||
|
const entries = ref({} as GalleryDisplayedEntries)
|
||||||
const ready = ref(false)
|
const ready = ref(false)
|
||||||
|
const currentRoute = getCurrentRoute()
|
||||||
|
const routeStore = useRouteStore()
|
||||||
|
const routeConfig = routeStore._routes[currentRoute.path] as GalleryListDefinition
|
||||||
|
const config = ref(null as GalleryList | null)
|
||||||
|
const globalConfig = routeStore._globals
|
||||||
|
const storageId = `${globalConfig.id}`
|
||||||
|
const viewPath = computed(() => `${currentRoute.path}/view`)
|
||||||
|
const hasWarnings = computed(async () => !!Object.values(entries.value).find(async other => !!(await other).warning))
|
||||||
|
const hideWarnings = defineModel('hideWarnings', { type: Boolean })
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handler for a tag being selected;
|
||||||
|
* updates the visibility state of the current entries
|
||||||
|
* @param tagsToggled: the tags currently toggled in the filter panel
|
||||||
|
*/
|
||||||
|
const onToggledTagsChanged = async (tagsToggled: string[]) => {
|
||||||
|
if (tagsToggled.length < 1) {
|
||||||
|
Object.keys(entries.value).forEach(async entryId => {
|
||||||
|
(await entries.value[entryId]).isHidden = false
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
Object.keys(entries.value).forEach(async entryId => {
|
||||||
|
(await entries.value[entryId]).isHidden = !(await entries.value[entryId]).tags?.some(own => tagsToggled.includes(own))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handler for the toggle for hiding/showing warnings;
|
||||||
|
* updates localstorage with the state of the checkbox
|
||||||
|
* so that it is saved between page loads
|
||||||
|
* @param event the event context which invoked this handler
|
||||||
|
*/
|
||||||
|
const onHideWarningsToggled = (event: Event) => {
|
||||||
|
storage.write(`${storageId}::hideWarnings`, (event.target as HTMLInputElement).checked)
|
||||||
|
}
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
ready.value = false
|
let listConfig = await fetchAndParseYaml<GalleryList>(routeConfig.config)
|
||||||
|
const list = await fetchConfigsFromList<GalleryEntry>(listConfig)
|
||||||
|
config.value = listConfig
|
||||||
|
entries.value = list.entries
|
||||||
|
document.title = routeConfig.fullTitle
|
||||||
ready.value = true
|
ready.value = true
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
@ -32,13 +97,12 @@ onMounted(async () => {
|
||||||
Transition
|
Transition
|
||||||
FilterPanel(
|
FilterPanel(
|
||||||
v-if='ready && !!config.tags'
|
v-if='ready && !!config.tags'
|
||||||
:ref='filterPanelRef'
|
|
||||||
:tags='config.tags'
|
:tags='config.tags'
|
||||||
@toggledTagsChanged='onToggledTagsChanged($event)'
|
@toggledTagsChanged='onToggledTagsChanged($event)'
|
||||||
)
|
)
|
||||||
Transition
|
Transition
|
||||||
.gallery(
|
.gallery(
|
||||||
v-if='galleryReady'
|
v-if='ready'
|
||||||
)
|
)
|
||||||
Transition(
|
Transition(
|
||||||
v-for='(entry, id) in entries'
|
v-for='(entry, id) in entries'
|
||||||
|
@ -46,9 +110,10 @@ onMounted(async () => {
|
||||||
GalleryTile(
|
GalleryTile(
|
||||||
v-if='!entry.isHidden || !config.removeFromView'
|
v-if='!entry.isHidden || !config.removeFromView'
|
||||||
:class='{ hidden: entry.isHidden && !config.removeFromView }'
|
:class='{ hidden: entry.isHidden && !config.removeFromView }'
|
||||||
:entry='entry'
|
|
||||||
:id='id'
|
|
||||||
:hideWarnings='hideWarnings'
|
:hideWarnings='hideWarnings'
|
||||||
@click='onTileClicked($event)'
|
:id='id'
|
||||||
|
:viewPath='viewPath'
|
||||||
|
:isInternal='true'
|
||||||
|
:entry='entry'
|
||||||
)
|
)
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -1,54 +1,80 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed } from 'vue'
|
import { computed, onMounted, ref, toRaw } from 'vue'
|
||||||
|
import { marked } from 'marked'
|
||||||
import type {
|
import type {
|
||||||
GalleryEntry,
|
GalleryEntry,
|
||||||
|
GalleryEntryWithVariants,
|
||||||
|
GalleryEntryWithoutVariants,
|
||||||
} from '@goldenwere/mackenzii-types'
|
} from '@goldenwere/mackenzii-types'
|
||||||
|
|
||||||
import { getTitleFromEntryOrId } from './gallery-utilities'
|
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
entry: GalleryEntry,
|
|
||||||
id: string,
|
id: string,
|
||||||
|
viewPath: string,
|
||||||
hideWarnings: boolean,
|
hideWarnings: boolean,
|
||||||
|
isInternal: boolean,
|
||||||
|
entry: Promise<GalleryEntry>
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
defineEmits<{
|
defineEmits<{
|
||||||
(e: 'click', value: { event: Event, id: string }): void
|
(e: 'click', value: { event: Event, id: string }): void
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const title = computed(() => getTitleFromEntryOrId(props.entry, props.id))
|
const resolved = ref({} as GalleryEntry)
|
||||||
const styles = computed(() => {
|
const hasVariants = computed(() => !!(resolved.value as GalleryEntryWithVariants).variants)
|
||||||
const stylesReturn: Record<string, string> = {}
|
const description = computed(() => marked.parse(resolved.value.description || '') as string)
|
||||||
if (!!props.entry.thumbnailBackground) {
|
const title = computed(() => marked.parse(resolved.value.title || props.id) as string)
|
||||||
stylesReturn.background = props.entry.thumbnailBackground
|
const warning = computed(() => resolved.value.warning)
|
||||||
|
|
||||||
|
const thumbnail = computed(() => {
|
||||||
|
if (!!resolved.value.thumbnail) {
|
||||||
|
return toRaw(resolved.value.thumbnail.style)
|
||||||
|
} else if (hasVariants) {
|
||||||
|
const resolvedCast = resolved.value as GalleryEntryWithVariants
|
||||||
|
if (resolvedCast.defaultVariant) {
|
||||||
|
return {
|
||||||
|
'background-image': `url("${resolvedCast.variants[resolvedCast.defaultVariant].url}")`,
|
||||||
|
'background-position': 'center center',
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
'background-image': `url("${(resolved.value as GalleryEntryWithoutVariants).url}")`,
|
||||||
|
'background-position': 'center center',
|
||||||
|
}
|
||||||
}
|
}
|
||||||
stylesReturn['object-position'] = props.entry.thumbnailPosition || 'center center'
|
})
|
||||||
return stylesReturn
|
|
||||||
|
onMounted(async () => {
|
||||||
|
resolved.value = await props.entry
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template lang="pug">
|
<template lang="pug">
|
||||||
.gallery-embed(
|
include /src/templates/link.pug
|
||||||
@click='$emit("click", { event: $event, id })'
|
.gallery-embed
|
||||||
)
|
.thumbnail-wrapper(
|
||||||
.image-wrapper(
|
:class='{ warning: !!warning && !hideWarnings }'
|
||||||
:class='{ warning: !!entry.warning && !hideWarnings }'
|
|
||||||
)
|
)
|
||||||
img(
|
+link
|
||||||
:src='entry.thumbnailUrl || entry.url'
|
.thumbnail(
|
||||||
:alt='entry.description || id'
|
:style='thumbnail'
|
||||||
:style='styles'
|
)
|
||||||
)
|
|
||||||
.caption-wrapper
|
.caption-wrapper
|
||||||
p.variants(
|
p.title(
|
||||||
v-if='!!entry.variants'
|
v-html='title'
|
||||||
) has variants
|
)
|
||||||
p {{ title }}
|
|
||||||
p.warning(
|
p.warning(
|
||||||
v-if='!!entry.warning'
|
v-if='!!warning'
|
||||||
) {{ entry.warning }}
|
) {{ warning }}
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped lang="sass">
|
<style scoped lang="sass">
|
||||||
|
.thumbnail-wrapper
|
||||||
|
.thumbnail
|
||||||
|
height: 100%
|
||||||
|
width: 100%
|
||||||
|
display: block
|
||||||
|
background-size: cover
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -1,54 +0,0 @@
|
||||||
import type {
|
|
||||||
GalleryEntry,
|
|
||||||
GalleryEntryInheritedProperties,
|
|
||||||
} from '@goldenwere/mackenzii-types'
|
|
||||||
|
|
||||||
import { deepCopy } from 'src/utilities/dom'
|
|
||||||
|
|
||||||
export const getTitleFromEntryOrId = (entry: GalleryEntry, id: string) => (
|
|
||||||
entry.title !== undefined
|
|
||||||
? entry.title === '' || entry.title === null
|
|
||||||
? 'untitled'
|
|
||||||
: entry.title
|
|
||||||
: id
|
|
||||||
)
|
|
||||||
|
|
||||||
export const amendVariantsWithDefaults = (parent: GalleryEntryInheritedProperties, children: GalleryEntryInheritedProperties) => {
|
|
||||||
const _children = deepCopy(children)
|
|
||||||
if (!!_children) {
|
|
||||||
Object.keys(children).forEach(id => {
|
|
||||||
_children[id] = _amendVariantWithDefaults(parent, children[id])
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return _children
|
|
||||||
}
|
|
||||||
|
|
||||||
export const _amendVariantWithDefaults = (parent: GalleryEntryInheritedProperties, variant: GalleryEntryInheritedProperties) => {
|
|
||||||
if (variant.title === undefined && (!!parent.title || parent.title === null || parent.title === '')) {
|
|
||||||
variant.title = parent.title
|
|
||||||
}
|
|
||||||
if (!variant.description && !!parent.description) {
|
|
||||||
variant.description = parent.description
|
|
||||||
}
|
|
||||||
if (variant.warning === undefined && !!parent.warning) {
|
|
||||||
variant.warning = parent.warning
|
|
||||||
}
|
|
||||||
if (!variant.fields && !!parent.fields) {
|
|
||||||
variant.fields = parent.fields
|
|
||||||
}
|
|
||||||
if (!variant.tags && !!parent.tags) {
|
|
||||||
variant.tags = parent.tags
|
|
||||||
}
|
|
||||||
if (!variant.thumbnailPosition && !!parent.thumbnailPosition) {
|
|
||||||
variant.thumbnailPosition = parent.thumbnailPosition
|
|
||||||
}
|
|
||||||
if (!variant.thumbnailBackground && !!parent.thumbnailBackground) {
|
|
||||||
variant.thumbnailBackground = parent.thumbnailBackground
|
|
||||||
}
|
|
||||||
if (!variant.background && !!parent.background) {
|
|
||||||
variant.background = parent.background
|
|
||||||
}
|
|
||||||
|
|
||||||
return variant
|
|
||||||
}
|
|
|
@ -1,14 +1,16 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, onMounted, ref } from 'vue'
|
import { computed, onMounted, ref } from 'vue'
|
||||||
|
import { marked } from 'marked'
|
||||||
import type {
|
import type {
|
||||||
GalleryEntry,
|
GalleryEntry,
|
||||||
GalleryList,
|
GalleryEntryWithVariants,
|
||||||
|
GalleryEntryWithoutVariants,
|
||||||
GalleryListDefinition,
|
GalleryListDefinition,
|
||||||
|
ListWithEntries,
|
||||||
RoutedWindow,
|
RoutedWindow,
|
||||||
} from '@goldenwere/mackenzii-types'
|
} from '@goldenwere/mackenzii-types'
|
||||||
|
|
||||||
import { _amendVariantWithDefaults, getTitleFromEntryOrId } from './gallery-utilities'
|
import { fetchAndParseYaml, fetchConfigByIdFromList } from 'src/utilities/fetch'
|
||||||
import { fetchAndParseYaml, fetchConfigByIdFromList, fetchNestedConfigs } from 'src/utilities/fetch'
|
|
||||||
import { getCurrentRoute } from 'src/utilities/vuetils'
|
import { getCurrentRoute } from 'src/utilities/vuetils'
|
||||||
import { useRouteStore } from 'src/routes'
|
import { useRouteStore } from 'src/routes'
|
||||||
|
|
||||||
|
@ -16,46 +18,39 @@ import EmbedableContent from 'src/components/shared/embedable-content.vue'
|
||||||
|
|
||||||
declare const window: RoutedWindow
|
declare const window: RoutedWindow
|
||||||
|
|
||||||
const props = defineProps<{
|
|
||||||
variants: string[]
|
|
||||||
}>()
|
|
||||||
|
|
||||||
const emits = defineEmits<{
|
const emits = defineEmits<{
|
||||||
(e: 'loaded'): void
|
(e: 'loaded'): void
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const ready = ref(false)
|
const ready = ref(false)
|
||||||
const config = ref(null! as GalleryList)
|
const resolved = ref({} as GalleryEntry)
|
||||||
const entry = ref(null! as GalleryEntry)
|
|
||||||
const id = ref('')
|
|
||||||
const currentRoute = getCurrentRoute()
|
const currentRoute = getCurrentRoute()
|
||||||
const routeStore = useRouteStore()
|
const routeStore = useRouteStore()
|
||||||
const routeConfig = routeStore._routes[currentRoute.path.substring(0, currentRoute.path.length - 5)] as GalleryListDefinition
|
const routeConfig = routeStore._routes[currentRoute.path.substring(0, currentRoute.path.length - 5)] as GalleryListDefinition
|
||||||
const routeSubConfig = routeStore._routes[currentRoute.path]
|
const routeSubConfig = routeStore._routes[currentRoute.path]
|
||||||
const title = ref('')
|
const description = computed(() => marked.parse(resolved.value.description || '') as string)
|
||||||
const styles = computed(() => {
|
const title = computed(() => marked.parse(resolved.value.title || currentRoute.query.id!.toString()) as string)
|
||||||
const stylesReturn: Record<string, string> = {}
|
const url = computed(() => (resolved.value as GalleryEntryWithoutVariants).url)
|
||||||
if (!!entry.value.background) {
|
const hasVariants = computed(() => !!(resolved.value as GalleryEntryWithVariants).variants)
|
||||||
stylesReturn.background = entry.value.background
|
const fields = computed(() => {
|
||||||
|
const toReturn = {} as Record<string, any>
|
||||||
|
if (!!resolved.value.date) {
|
||||||
|
toReturn.date = resolved.value.date
|
||||||
}
|
}
|
||||||
return stylesReturn
|
if (!!resolved.value.warning) {
|
||||||
|
toReturn.warning = resolved.value.warning
|
||||||
|
}
|
||||||
|
return toReturn
|
||||||
})
|
})
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
let listConfig = await fetchAndParseYaml<GalleryList>(routeConfig.config)
|
let config = await fetchAndParseYaml<ListWithEntries<GalleryEntry>>(routeConfig.config)
|
||||||
config.value = listConfig
|
resolved.value = await fetchConfigByIdFromList(config, currentRoute.query.id as string)
|
||||||
let ids = props.variants
|
document.title = routeSubConfig.fullTitle?.replace('$ENTRY', resolved.value.title)
|
||||||
let viewId = ids.pop()!
|
routeStore.setBreadcrumbs(currentRoute, resolved.value.title)
|
||||||
let entries = await fetchNestedConfigs(listConfig, ids, _amendVariantWithDefaults)
|
|
||||||
let finalEntry = await fetchConfigByIdFromList<GalleryEntry>(entries, viewId)
|
|
||||||
entry.value = _amendVariantWithDefaults(entries, finalEntry)
|
|
||||||
id.value = props.variants[props.variants.length - 1]
|
|
||||||
title.value = getTitleFromEntryOrId(entry.value, id.value)
|
|
||||||
document.title = routeSubConfig.fullTitle?.replace('$ENTRY', title.value)
|
|
||||||
routeStore.setBreadcrumbs(currentRoute, title.value)
|
|
||||||
window.routeConfig = {...routeConfig}
|
window.routeConfig = {...routeConfig}
|
||||||
window.routeSubConfig = {...routeSubConfig}
|
window.routeSubConfig = {...routeSubConfig}
|
||||||
window.routeContentConfig = {...entry.value}
|
window.routeContentConfig = {...resolved.value}
|
||||||
|
|
||||||
ready.value = true
|
ready.value = true
|
||||||
emits('loaded')
|
emits('loaded')
|
||||||
|
@ -70,20 +65,21 @@ onMounted(async () => {
|
||||||
)
|
)
|
||||||
.view-outlet
|
.view-outlet
|
||||||
img(
|
img(
|
||||||
:src='entry.url || entry.thumbnailUrl'
|
:src='url'
|
||||||
:style='styles'
|
|
||||||
)
|
)
|
||||||
.view-content
|
.view-content
|
||||||
p {{ title }}
|
p(
|
||||||
|
v-html='title'
|
||||||
|
)
|
||||||
dl.info
|
dl.info
|
||||||
.info-entry(
|
.info-entry(
|
||||||
v-for='(field, key) in entry.fields'
|
v-for='(field, key) in fields'
|
||||||
:class='key'
|
:class='key'
|
||||||
)
|
)
|
||||||
dt {{ key }}
|
dt {{ key }}
|
||||||
dd {{ field }}
|
dd {{ field }}
|
||||||
EmbedableContent(
|
EmbedableContent(
|
||||||
:content='entry.description'
|
:content='description'
|
||||||
)
|
)
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue