update tagging in cms
- tags are now on global config and not per-list - tags are now stored in a regular array rather than key-value object - tags can be fetched externally or embedded in the global config
This commit is contained in:
parent
9ce4e38301
commit
8267ab8e82
6 changed files with 45 additions and 53 deletions
2
libs/types/src/config/globals.d.ts
vendored
2
libs/types/src/config/globals.d.ts
vendored
|
@ -1,4 +1,5 @@
|
||||||
import type { HeaderEntry } from './navigation'
|
import type { HeaderEntry } from './navigation'
|
||||||
|
import type { MediaEntryTag } from '../content/entryTag'
|
||||||
import type { RouteCollection } from './routing'
|
import type { RouteCollection } from './routing'
|
||||||
import type { SiteThemeList } from './themes'
|
import type { SiteThemeList } from './themes'
|
||||||
import type { WarningModal } from './warnings'
|
import type { WarningModal } from './warnings'
|
||||||
|
@ -12,5 +13,6 @@ export type SiteGlobals = {
|
||||||
id: string
|
id: string
|
||||||
stylesheetUrls: string[]
|
stylesheetUrls: string[]
|
||||||
themes: SiteThemeList
|
themes: SiteThemeList
|
||||||
|
tags?: string | MediaEntryTag[]
|
||||||
warning: WarningModal
|
warning: WarningModal
|
||||||
}
|
}
|
||||||
|
|
20
libs/types/src/content/entryTag.d.ts
vendored
20
libs/types/src/content/entryTag.d.ts
vendored
|
@ -2,7 +2,11 @@
|
||||||
* Defines a tag used by entries in a gallery-list/article-list,
|
* Defines a tag used by entries in a gallery-list/article-list,
|
||||||
* used for filtering entries from view
|
* used for filtering entries from view
|
||||||
*/
|
*/
|
||||||
export type EntryTag = {
|
export type MediaEntryTag = {
|
||||||
|
/**
|
||||||
|
* specifies the id for the tag
|
||||||
|
*/
|
||||||
|
tagId: string
|
||||||
/**
|
/**
|
||||||
* specifies a category that the tag belongs to
|
* specifies a category that the tag belongs to
|
||||||
* in order to optionally organize them in the view;
|
* in order to optionally organize them in the view;
|
||||||
|
@ -11,19 +15,13 @@ export type EntryTag = {
|
||||||
* placed before any other sections formed by categories (if any)
|
* placed before any other sections formed by categories (if any)
|
||||||
*/
|
*/
|
||||||
category?: string
|
category?: string
|
||||||
|
/**
|
||||||
|
* can be used to describe a tag when hovering over it in the UI
|
||||||
|
*/
|
||||||
|
description?: string
|
||||||
/**
|
/**
|
||||||
* specifies the name that the tag will appear as in the DOM;
|
* specifies the name that the tag will appear as in the DOM;
|
||||||
* if not specified, the id of the tag will be used in its place
|
* if not specified, the id of the tag will be used in its place
|
||||||
*/
|
*/
|
||||||
displayName?: string
|
displayName?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Defines the list of tags in a gallery-list/article-list,
|
|
||||||
* a key-value object where the value is the entry,
|
|
||||||
* and the key represents the id of a tag;
|
|
||||||
* the id of a tag must be unique,
|
|
||||||
* and the ids specified in a gallery/article entry must match
|
|
||||||
* the ids specified in `EntryTagCollection` in order for them to work effectively
|
|
||||||
*/
|
|
||||||
export type EntryTagCollection = { [id: string]: EntryTag }
|
|
||||||
|
|
17
libs/types/src/content/templates/shared.d.ts
vendored
17
libs/types/src/content/templates/shared.d.ts
vendored
|
@ -1,5 +1,4 @@
|
||||||
import type { DateRange } from '../dateRange'
|
import type { DateRange } from '../dateRange'
|
||||||
import type { EntryTagCollection } from '../entryTag'
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines entries that are already fetched or are embedded directly in the list.
|
* Defines entries that are already fetched or are embedded directly in the list.
|
||||||
|
@ -40,22 +39,6 @@ export type ListWithEntries<T> = {
|
||||||
*/
|
*/
|
||||||
export type ResolvedListEntries<T> = ListEntries<Promise<T>>
|
export type ResolvedListEntries<T> = ListEntries<Promise<T>>
|
||||||
|
|
||||||
/**
|
|
||||||
* Defines a list that supports tagging the entries
|
|
||||||
*/
|
|
||||||
export type ListWithTags<T> = {
|
|
||||||
/**
|
|
||||||
* 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<T>
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines a list that has warnings on its entries
|
* Defines a list that has warnings on its entries
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -1,27 +1,28 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed } from 'vue'
|
import { computed, ref, onMounted } from 'vue'
|
||||||
import type { EntryTagCollection } from '@goldenwere/mackenzii-types'
|
import type { MediaEntryTag } from '@goldenwere/mackenzii-types'
|
||||||
|
import { fetchAndParseConfig } from 'src/utilities/fetch'
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
tags: EntryTagCollection
|
tags: string | MediaEntryTag[]
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
const emits = defineEmits<{
|
const emits = defineEmits<{
|
||||||
(e: 'toggledTagsChanged', value: string[]): void
|
(e: 'toggledTagsChanged', value: string[]): void
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
|
const tagsLoaded = ref([] as MediaEntryTag[])
|
||||||
const tagsByCategory = computed(() => {
|
const tagsByCategory = computed(() => {
|
||||||
const value: { [category: string]: Record<string, string> } = { 'NoCategory': {}}
|
const value: { [category: string]: MediaEntryTag[] } = { 'NoCategory': []}
|
||||||
|
|
||||||
Object.keys(props.tags).forEach(id => {
|
tagsLoaded.value.forEach((tag) => {
|
||||||
const tag = props.tags![id]
|
|
||||||
if (!!tag.category) {
|
if (!!tag.category) {
|
||||||
if (!value[tag.category]) {
|
if (!value[tag.category]) {
|
||||||
value[tag.category] = {}
|
value[tag.category] = []
|
||||||
}
|
}
|
||||||
value[tag.category][id] = tag.displayName || id
|
value[tag.category].push(tag)
|
||||||
} else {
|
} else {
|
||||||
value['NoCategory'][id] = tag.displayName || id
|
value['NoCategory'].push(tag)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -44,6 +45,7 @@ const onToggleTag = (event: Event, tagId: string) => {
|
||||||
tagsToggled.splice(index, 1)
|
tagsToggled.splice(index, 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
console.log(tagsToggled)
|
||||||
emits('toggledTagsChanged', tagsToggled)
|
emits('toggledTagsChanged', tagsToggled)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -53,6 +55,12 @@ const onToggleTag = (event: Event, tagId: string) => {
|
||||||
const resetTags = () => {
|
const resetTags = () => {
|
||||||
tagsToggled = []
|
tagsToggled = []
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
tagsLoaded.value = Array.isArray(props.tags)
|
||||||
|
? tagsLoaded.value = props.tags
|
||||||
|
: await fetchAndParseConfig<MediaEntryTag[]>(props.tags)
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template lang="pug">
|
<template lang="pug">
|
||||||
|
@ -66,17 +74,17 @@ const resetTags = () => {
|
||||||
v-if='category !== "NoCategory"'
|
v-if='category !== "NoCategory"'
|
||||||
) {{ category }}
|
) {{ category }}
|
||||||
.input.labeled-checkbox(
|
.input.labeled-checkbox(
|
||||||
v-for='(tagDisplayName, tagId) in tags'
|
v-for='tag in tags'
|
||||||
:id='tagId'
|
:id='tag.tagId'
|
||||||
)
|
)
|
||||||
label(
|
label(
|
||||||
:for='`${tagId}-toggle`'
|
:for='`${tag.tagId}-toggle`'
|
||||||
) {{ tagDisplayName }}
|
) {{ tag.displayName || tag.tagId }}
|
||||||
input(
|
input(
|
||||||
type='checkbox'
|
type='checkbox'
|
||||||
:name='`${tagId}-toggle`'
|
:name='`${tag.tagId}-toggle`'
|
||||||
:id='`${tagId}-toggle`'
|
:id='`${tag.tagId}-toggle`'
|
||||||
@input='onToggleTag($event, tagId)'
|
@input='onToggleTag($event, tag.tagId)'
|
||||||
)
|
)
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|
|
@ -21,7 +21,7 @@ export const createApp = ViteSSG(
|
||||||
// the root component
|
// the root component
|
||||||
main,
|
main,
|
||||||
// vue-router options
|
// vue-router options
|
||||||
{ routes: createRoutes(globals as SiteGlobals) },
|
{ routes: createRoutes(globals as unknown as SiteGlobals) },
|
||||||
// function to have custom setups
|
// function to have custom setups
|
||||||
async ({ app, router, routes, isClient, initialState }) => {
|
async ({ app, router, routes, isClient, initialState }) => {
|
||||||
const hljsResolved: HLJSApi = await hljs as any
|
const hljsResolved: HLJSApi = await hljs as any
|
||||||
|
@ -44,6 +44,6 @@ export const createApp = ViteSSG(
|
||||||
}
|
}
|
||||||
|
|
||||||
app.use(createPinia())
|
app.use(createPinia())
|
||||||
initializeRouteStore(routes, globals as SiteGlobals)
|
initializeRouteStore(routes, globals as unknown as SiteGlobals)
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, onMounted, ref } from 'vue'
|
import { computed, onMounted, useTemplateRef, ref, toRaw } from 'vue'
|
||||||
import type {
|
import type {
|
||||||
ConfigfulRouteDefinition,
|
ConfigfulRouteDefinition,
|
||||||
ListWithTags,
|
|
||||||
ListWithWarnings,
|
ListWithWarnings,
|
||||||
MediaEntry,
|
MediaEntry,
|
||||||
ResolvedListEntries,
|
ResolvedListEntries,
|
||||||
|
@ -16,7 +15,6 @@ import GalleryTile from '../gallery/gallery-tile.vue'
|
||||||
import ArticleTile from '../article/article-tile.vue'
|
import ArticleTile from '../article/article-tile.vue'
|
||||||
|
|
||||||
type MediaList =
|
type MediaList =
|
||||||
& ListWithTags<MediaEntry>
|
|
||||||
& ListWithWarnings<MediaEntry>
|
& ListWithWarnings<MediaEntry>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -41,6 +39,7 @@ const viewPath = computed(() => `${currentRoute.path}/view`)
|
||||||
|
|
||||||
const config = ref(null as MediaList | null)
|
const config = ref(null as MediaList | null)
|
||||||
const entries = ref({} as DisplayedEntries)
|
const entries = ref({} as DisplayedEntries)
|
||||||
|
const entriesRefs = useTemplateRef('tiles' as any)
|
||||||
const hasWarnings = ref(false)
|
const hasWarnings = ref(false)
|
||||||
const hideWarnings = defineModel('hideWarnings', { type: Boolean })
|
const hideWarnings = defineModel('hideWarnings', { type: Boolean })
|
||||||
const template = ref(routeConfig.template)
|
const template = ref(routeConfig.template)
|
||||||
|
@ -117,8 +116,8 @@ onMounted(async () => {
|
||||||
)
|
)
|
||||||
Transition
|
Transition
|
||||||
FilterPanel(
|
FilterPanel(
|
||||||
v-if='ready && !!config.tags'
|
v-if='ready && globalConfig.tags'
|
||||||
:tags='config.tags'
|
:tags='globalConfig.tags'
|
||||||
@toggledTagsChanged='onToggledTagsChanged($event)'
|
@toggledTagsChanged='onToggledTagsChanged($event)'
|
||||||
)
|
)
|
||||||
Transition
|
Transition
|
||||||
|
@ -130,7 +129,8 @@ onMounted(async () => {
|
||||||
v-for='(entry, id) in entries'
|
v-for='(entry, id) in entries'
|
||||||
)
|
)
|
||||||
GalleryTile(
|
GalleryTile(
|
||||||
v-if='template === "gallery-list" && (!entry.isHidden || !config.removeFromView)'
|
v-if='template === "gallery-list"'
|
||||||
|
ref='tiles'
|
||||||
:class='{ hidden: entry.isHidden && !config.removeFromView }'
|
:class='{ hidden: entry.isHidden && !config.removeFromView }'
|
||||||
:hideWarnings='hideWarnings'
|
:hideWarnings='hideWarnings'
|
||||||
:id='id'
|
:id='id'
|
||||||
|
@ -139,7 +139,8 @@ onMounted(async () => {
|
||||||
:entry='entry'
|
:entry='entry'
|
||||||
)
|
)
|
||||||
ArticleTile(
|
ArticleTile(
|
||||||
v-else-if='template === "article-list" && (!entry.isHidden || !config.removeFromView)'
|
v-else-if='template === "article-list"'
|
||||||
|
ref='tiles'
|
||||||
:class='{ hidden: entry.isHidden && !config.removeFromView }'
|
:class='{ hidden: entry.isHidden && !config.removeFromView }'
|
||||||
:hideWarnings='hideWarnings'
|
:hideWarnings='hideWarnings'
|
||||||
:id='id'
|
:id='id'
|
||||||
|
|
Loading…
Add table
Reference in a new issue