once again change how blog-list is structured

This commit is contained in:
lightling 2024-06-26 17:31:32 -04:00
parent ca91d9fcb5
commit e4da36a23c
6 changed files with 100 additions and 56 deletions

View file

@ -1,6 +1,6 @@
import type { DateRange } from '../dateRange' import type { DateRange } from '../dateRange'
import type { EntryTagCollection } from '../entryTag' import type { EntryTagCollection } from '../entryTag'
import type { EntryWithConfig, EntryWithContent } from './shared' import type { ListWithEntries } from './shared'
/** /**
* This describes aditional information about a blog entry * This describes aditional information about a blog entry
@ -8,6 +8,9 @@ import type { EntryWithConfig, EntryWithContent } from './shared'
* At minimum, the title should be specified. * At minimum, the title should be specified.
*/ */
export type BlogEntry = { export type BlogEntry = {
/**
* Specifies the date of the blog entry
*/
date?: DateRange | string | number date?: DateRange | string | number
/**[Supports Markdown] /**[Supports Markdown]
* Information to summarize an entry * Information to summarize an entry
@ -21,18 +24,22 @@ export type BlogEntry = {
* The title of the blog entry * The title of the blog entry
*/ */
title: string title: string
/**
* Information regarding the thumbnail
*/
thumbnail?: { thumbnail?: {
/**
* Sets the inline-styles for the thumbnail
*/
style: CSSStyleDeclaration style: CSSStyleDeclaration
} }
} /**
* URL to the markdown document of the blog entry
export type BlogEntries = { [key: string]: */
& EntryWithContent url: string
& EntryWithConfig<BlogEntry>
} }
export type BlogList = { export type BlogList = {
entries: BlogEntries
tags?: EntryTagCollection tags?: EntryTagCollection
removeFromView?: boolean removeFromView?: boolean
} } & ListWithEntries<BlogEntry>

View file

@ -1,19 +1,33 @@
/** /**
* Denotes a listing entry that contains content, * Defines entries that are already fetched or are embedded directly in the list.
* which can be defined either directly in the list * Stored in key-value format where the key is the id of the entry,
* or be defined in a separate file * and the value is the entry config itself
* (defined as `T` based on the type of the entries in the implemented list)
*/ */
export type EntryWithContent = { export type ListEntry<T> = { [key: string]: T }
content?: string
contentUrl?: string
}
/** /**
* Denotes a listing entry that contains config, * Defines entries that are fetched from remote config files.
* which can be defined either directly in the list * Stored in key-value format where the key is the id of the entry,
* or be defined in a separate file * and the value is the url to the config file.
*/ */
export type EntryWithConfig<T> = { export type ListRemoteEntries<T> = { [key: string]: string }
config?: T
configUrl?: string /**
* Defines a list-type template that has config entries defined by id.
* One of two must be defined: `entries` or `embeddedEntries`.
* Both can be defined and used simultaneously.
* Note that in the case of an `id` collision
* (i.e. both `entries` and `embeddedEntries` have an entry with the same id),
* the implemented behavior should pick the `embeddedEntry`
*/
export type ListWithEntries<T> = {
/**
* Entries that will be fetched from remote config files
*/
entries?: ListRemoteEntries
/**
* Entries that are embedded directly in the list config
*/
embeddedEntries?: ListEntry
} }

View file

@ -10,4 +10,5 @@ export * from './content/link'
export * from './content/templates/blog-list' export * from './content/templates/blog-list'
export * from './content/templates/gallery-list' export * from './content/templates/gallery-list'
export * from './content/templates/shared'
export * from './content/templates/templateType' export * from './content/templates/templateType'

View file

@ -1,11 +1,10 @@
import DOMPurify from 'dompurify' import DOMPurify from 'dompurify'
import { marked } from 'marked' import { marked } from 'marked'
import yaml from 'js-yaml' import yaml from 'js-yaml'
import type { import type {
EntryWithConfig, ListEntry,
EntryWithContent, ListWithEntries,
} from '@goldenwere/mackenzii-types/src/content/templates/shared' } from '@goldenwere/mackenzii-types'
/** /**
* Config used for DOMPurify. * Config used for DOMPurify.
@ -129,33 +128,57 @@ export const fetchAndParseMarkdown = async (path: string) => {
} }
/** /**
* Fetches content for an entry * Fetches all the entry configs from a given list
* @param entry the entry whose content should be fetched * @param list the list to fetch configs from
* @returns the markdown content for the entry * @returns the resolved configs
*/ */
export const fetchContent = async (entry: EntryWithContent) => { export const fetchConfigsFromList = async <T>(list: ListWithEntries<T>): Promise<{
if (!!entry.content) { ids: string[]
return entry.content entries: ListEntry<T>
} else if (!!entry.contentUrl) { }> => {
return fetchAndParseMarkdown(entry.contentUrl) return new Promise(async (resolve, reject) => {
} let ids: string[] = []
let entries: ListEntry<T> = {}
if (!!list.entries) {
ids = ids.concat(Object.keys(list.entries))
const allEntries = await Promise.all(ids.map(async id => ({
entry: await fetchAndParseYaml<T>(list.entries[id]),
id,
})))
allEntries.forEach((entry) => {
entries[entry.id] = entry.entry
})
}
if (!!list.embeddedEntries) {
ids = ids.concat(Object.keys(list.embeddedEntries))
entries = {
...entries,
...list.embeddedEntries,
}
}
return '' resolve({
entries,
ids,
})
})
} }
/** /**
* Fetches config for an entry * Fetches the config for a given id from the given list
* @param entry the entry whose config should be fetched * @param list the list to query from
* @returns the config object for the entry * @param id the id to query for
* @returns the resolved config
*/ */
export const fetchConfig = async <T>(entry: EntryWithConfig<T>) => { export const fetchConfigByIdFromList = async <T>(list: ListWithEntries<T>, id: string): Promise<T> => {
if (!!entry.config) { return new Promise(async (resolve, reject) => {
return entry.config as T if (!!list.embeddedEntries && list.embeddedEntries[id]) {
} else if (!!entry.configUrl) { resolve(list.embeddedEntries[id])
return fetchAndParseYaml(entry.configUrl) as T } else if (!!list.entries && list.entries[id]) {
} const config = await fetchAndParseYaml<T>(list.entries[id])
resolve(config)
return {} as T }
})
} }
/** /**

View file

@ -6,7 +6,7 @@ import type {
BlogListDefinition, BlogListDefinition,
} from '@goldenwere/mackenzii-types' } from '@goldenwere/mackenzii-types'
import { fetchAndParseYaml, fetchConfig } from 'src/utilities/fetch' import { fetchAndParseYaml, fetchConfigsFromList } 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'
@ -51,12 +51,11 @@ const viewPath = computed(() => `${currentRoute.path}/view`)
} }
onMounted(async () => { onMounted(async () => {
config.value = await fetchAndParseYaml<BlogList>(routeConfig.config) let listConfig = await fetchAndParseYaml<BlogList>(routeConfig.config)
entryIds.value = Object.keys(config.value.entries) const list = await fetchConfigsFromList<BlogEntry>(listConfig)
for (let i = 0; i < entryIds.value.length; ++i) { config.value = listConfig
const id = entryIds.value[i] entryIds.value = list.ids
entries.value[id] = await fetchConfig(config.value.entries[id]) entries.value = list.entries
}
document.title = routeConfig.fullTitle document.title = routeConfig.fullTitle
ready.value = true ready.value = true
}) })

View file

@ -7,7 +7,7 @@ import type {
RoutedWindow, RoutedWindow,
} from '@goldenwere/mackenzii-types' } from '@goldenwere/mackenzii-types'
import { fetchAndParseYaml, fetchConfig, fetchContent } from 'src/utilities/fetch' import { fetchAndParseMarkdown, fetchAndParseYaml, fetchConfigByIdFromList } 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'
@ -29,8 +29,8 @@ const routeSubConfig = routeStore._routes[currentRoute.path]
onMounted(async () => { onMounted(async () => {
const config = await fetchAndParseYaml<BlogList>(routeConfig.config) const config = await fetchAndParseYaml<BlogList>(routeConfig.config)
info.value = await fetchConfig(config.entries[currentRoute.query.id as string]) info.value = await fetchConfigByIdFromList(config, currentRoute.query.id as string)
const md = await fetchContent(config.entries[currentRoute.query.id as string]) const md = await fetchAndParseMarkdown(info.value.url)
content.value = md content.value = md
document.title = routeSubConfig.fullTitle?.replace('$ENTRY', info.value.title) document.title = routeSubConfig.fullTitle?.replace('$ENTRY', info.value.title)
routeStore.setBreadcrumbs(currentRoute, info.value.title) routeStore.setBreadcrumbs(currentRoute, info.value.title)