From 650df38f6c0bdc7c0cf36b7e56bf2e34742ab4be Mon Sep 17 00:00:00 2001 From: Lightling Date: Tue, 30 Jul 2024 22:01:03 -0400 Subject: [PATCH] begin refactoring lists - async fetching of list entries is now done when the tile is mounted for blog-lists - started removing recursion from galleries --- .../src/content/templates/blog-list.d.ts | 39 +---- .../src/content/templates/gallery-list.d.ts | 85 ++------- libs/types/src/content/templates/shared.d.ts | 63 ++++++- projects/frontend/src/utilities/fetch.ts | 67 ++----- .../frontend/src/views/blog/blog-list.vue | 23 +-- .../frontend/src/views/blog/blog-tile.vue | 18 +- .../src/views/gallery/gallery-list.vue | 164 +----------------- 7 files changed, 111 insertions(+), 348 deletions(-) diff --git a/libs/types/src/content/templates/blog-list.d.ts b/libs/types/src/content/templates/blog-list.d.ts index 4c5286f..14b5d12 100644 --- a/libs/types/src/content/templates/blog-list.d.ts +++ b/libs/types/src/content/templates/blog-list.d.ts @@ -1,45 +1,12 @@ -import type { DateRange } from '../dateRange' -import type { EntryTagCollection } from '../entryTag' -import type { ListWithEntries } from './shared' +import type { MediaEntry } from './shared' /** - * This describes aditional information about a blog entry + * This describes additional information about a blog entry * to display when listing the entry on the blog-list page. - * At minimum, the title should be specified. */ export type BlogEntry = { - /** - * Specifies the date of the blog entry - */ - date?: DateRange | string | number - /**[Supports Markdown] - * Information to summarize an entry - */ - description?: string - /** - * Tags that correspond to filters on the blog-list page if defined - */ - tags?: string[] - /**[Supports Markdown] - * The title of the blog entry - */ - title: string - /** - * Information regarding the thumbnail - */ - thumbnail?: { - /** - * Sets the inline-styles for the thumbnail - */ - style: CSSStyleDeclaration - } /** * URL to the markdown document of the blog entry */ url: string -} - -export type BlogList = { - tags?: EntryTagCollection - removeFromView?: boolean -} & ListWithEntries +} & MediaEntry diff --git a/libs/types/src/content/templates/gallery-list.d.ts b/libs/types/src/content/templates/gallery-list.d.ts index 2d32a45..f51e1b6 100644 --- a/libs/types/src/content/templates/gallery-list.d.ts +++ b/libs/types/src/content/templates/gallery-list.d.ts @@ -1,83 +1,22 @@ -import type { EntryTagCollection } from '../entryTag' -import type { ListWithEntries } from './shared' +import type { MediaEntry } from './shared' /** - * A partial definition of a {@link GalleryEntry} - * which defines the properties of an entry - * that can be passed down from a top-level entry down to its variants + * Defines the supported formats for a gallery entry */ -export type GalleryEntryInheritedProperties = { - /** - * css background applied to the image (useful for transparent images) - */ - background?: string - /** - * [SUPPORTS MARKDOWN] a place for the siteowner to describe the entry - */ - description?: string - /** - * a key-value pair set of general-purpose fields to additionally describe the entry - * @example entry.fields = { - * 'date': '1960/01/01', - * 'rating': 'general', - * } - */ - fields?: Record - /** - * array of tag-ids that apply to the art; - * this is used to allow a visitor to filter out entries of a specific tag - * @see {@link GalleryList.tags} - */ - tags: string[] - /** - * css background applied to the thumbnail (useful for transparent images) - */ - thumbnailBackground?: string - /** - * the position of the thumbnail; - * this will be applied as css `object-position` - */ - thumbnailPosition?: string - /** - * the title of the entry - */ - title: string | null | undefined - /** - * any content warnings that apply to the entry, - * which will be used to apply the `.warning` classname - * to the DOM of a gallery tile - * and will be displayed with the tile - */ - warning?: string -} & ListWithEntries +export type GalleryEntryFormat = + | 'image' /** - * Defines an entry in a gallery that can be displayed in a tiled manner - * and can be clicked by a visitor to display its variants or the entry itself if there are none + * This describes additional information about a gallery entry + * to display when listing the entry on the gallery-list page. */ -export type GalleryEntry = GalleryEntryInheritedProperties & { +export type GalleryEntry = { /** - * the url to the thumbnail to show for the entry in the gallery tile + * The kind of media the entry is */ - thumbnailUrl?: string + format: GalleryEntryFormat /** - * the url to the entry itself + * URL to the markdown document of the gallery entry */ - url?: string -} - -/** - * Defines the model of the `GalleryList` template - */ -export type GalleryList = { - /** - * the tags to use for filtering entries - */ - tags?: EntryTagCollection - /** - * whether or not tag filtering removes entries completely from view; - * if false, they will apply a class selector instead - * in order to manually style (CSS filtering/opacity/etc.) - */ - removeFromView?: boolean -} & ListWithEntries + url: string +} & MediaEntry diff --git a/libs/types/src/content/templates/shared.d.ts b/libs/types/src/content/templates/shared.d.ts index 7451d9b..eeef640 100644 --- a/libs/types/src/content/templates/shared.d.ts +++ b/libs/types/src/content/templates/shared.d.ts @@ -1,3 +1,6 @@ +import type { DateRange } from '../dateRange' +import type { EntryTagCollection } from '../entryTag' + /** * Defines entries that are already fetched or are embedded directly in the list. * Stored in key-value format where the key is the id of the entry, @@ -32,9 +35,61 @@ export type ListWithEntries = { embeddedEntries?: ListEntries } -export type ListEntriesWithNestedEntries = { [key: string]: T & ListWithEntries } +/** + * The resolved instance of a list, which will flatten + */ +export type ResolvedListEntries = ListEntries> -export type ListWithNestedEntries = { - entries?: ListRemoteEntries - embeddedEntries?: ListEntriesWithNestedEntries +/** + * Defines a list that supports tagging the entries + */ +export type ListWithTags = { + /** + * the tags to use for filtering entries + */ + tags?: EntryTagCollection + /** + * whether or not tag filtering removes entries completely from view; + * if false, they will apply a class selector instead + * in order to manually style (CSS filtering/opacity/etc.) + */ + removeFromView?: boolean +} & ListWithEntries + +/** + * Defines an entry with common media-related fields + */ +export type MediaEntry = { + /** + * Specifies the date of the entry + */ + date?: DateRange | string | number + /**[Supports Markdown] + * Information to summarize an entry + */ + description?: string + /** + * Tags that correspond to filters on the list page if defined + */ + tags?: string[] + /**[Supports Markdown] + * The title of the entry + */ + title: string + /** + * Information regarding the thumbnail + */ + thumbnail?: { + /** + * Sets the inline-styles for the thumbnail + */ + style: CSSStyleDeclaration + } + /** + * any content warnings that apply to the entry, + * which will be used to apply the `.warning` classname + * to the DOM of a tile + * and will be displayed with the tile + */ + warning?: string } diff --git a/projects/frontend/src/utilities/fetch.ts b/projects/frontend/src/utilities/fetch.ts index a0d8043..cfe8ab5 100644 --- a/projects/frontend/src/utilities/fetch.ts +++ b/projects/frontend/src/utilities/fetch.ts @@ -3,9 +3,8 @@ import { marked } from 'marked' import yaml from 'js-yaml' import type { ListEntries, - ListEntriesWithNestedEntries, ListWithEntries, - ListWithNestedEntries, + ResolvedListEntries, } from '@goldenwere/mackenzii-types' import { useRouteStore } from 'src/routes' @@ -151,29 +150,24 @@ export const fetchAndParseMarkdown = async (path: string) => { * @param list the list to fetch configs from * @returns the resolved configs */ -export const fetchConfigsFromList = async (list: ListWithEntries | ListWithNestedEntries): Promise<{ +export const fetchConfigsFromList = async (list: ListWithEntries): Promise<{ ids: string[] - entries: ListEntries | ListEntriesWithNestedEntries + entries: ResolvedListEntries }> => { return new Promise(async (resolve, reject) => { let ids: string[] = [] - let entries: ListEntries = {} + let entries: ResolvedListEntries = {} if (!!list.entries) { ids = ids.concat(Object.keys(list.entries)) - const allEntries = await Promise.all(ids.map(async id => ({ - entry: await fetchAndParseYaml((list.entries!)[id]), - id, - }))) - allEntries.forEach((entry) => { - entries[entry.id] = entry.entry + ids.forEach(id => { + entries[id] = fetchAndParseYaml((list.entries!)[id]) }) } if (!!list.embeddedEntries) { ids = ids.concat(Object.keys(list.embeddedEntries)) - entries = { - ...entries, - ...list.embeddedEntries, - } + ids.forEach(id => { + entries[id] = new Promise((resolve) => resolve(list.embeddedEntries![id])) + }) } resolve({ @@ -189,7 +183,7 @@ export const fetchConfigsFromList = async (list: ListWithEntries | ListWit * @param id the id to query for * @returns the resolved config */ -export const fetchConfigByIdFromList = async (list: ListWithEntries | ListWithNestedEntries, id: string): Promise => { +export const fetchConfigByIdFromList = async (list: ListWithEntries, id: string): Promise => { return new Promise(async (resolve, reject) => { if (!!list.embeddedEntries && list.embeddedEntries[id]) { resolve(list.embeddedEntries[id]) @@ -200,47 +194,6 @@ export const fetchConfigByIdFromList = async (list: ListWithEntries | List }) } -export const fetchNestedConfigs = async ( - list: ListWithNestedEntries, - idsByDepth: string[], - proccessCallback?: (parent: ListEntriesWithNestedEntries, child: ListEntriesWithNestedEntries) => ListEntriesWithNestedEntries, -): Promise> => { - /* DOCUMENT STRUCTURE: - ...listConfig - entries: - 'key1': 'url' - ...entryConfig (T) - entries: - 'key1-1': 'url' - ...entryConfig (T) - embeddedEntries: - 'key1-2': - ...entryConfig (T) - embeddedEntries: - 'key2': - ...entryConfig (T) - embeddedEntries: - 'key2-1': - ...entryConfig (T) - idsByDepth STRUCTURE: - [ 'key1', 'key2-1' ] - */ - - return new Promise(async (resolve, reject) => { - let idsToLoop = [ ...idsByDepth ] - let current = await fetchConfigByIdFromList(list, idsToLoop.shift()!) as ListEntriesWithNestedEntries - if (idsToLoop.length > 0) { - let next = await fetchNestedConfigs(current as any, idsToLoop) as ListEntriesWithNestedEntries - if (!!proccessCallback) { - next = proccessCallback(current, next) - } - resolve(next) - } else { - resolve(current) - } - }) -} - /** * Fetches content type from an image * @param path the path of the file to check diff --git a/projects/frontend/src/views/blog/blog-list.vue b/projects/frontend/src/views/blog/blog-list.vue index f3eab25..840a5a4 100644 --- a/projects/frontend/src/views/blog/blog-list.vue +++ b/projects/frontend/src/views/blog/blog-list.vue @@ -1,9 +1,10 @@