2024-02-04 10:31:23 +08:00

171 lines
3.9 KiB
Vue

<script lang="ts" setup>
import { usePageData, usePageFrontmatter } from 'vuepress/client'
import { computed } from 'vue'
import type {
PlumeThemePageData,
PlumeThemePostFrontmatter,
} from '../../shared/index.js'
import { useExtraBlogData, useReadingTime } from '../composables/index.js'
import IconBooks from './icons/IconBooks.vue'
import IconClock from './icons/IconClock.vue'
import IconTag from './icons/IconTag.vue'
const page = usePageData<PlumeThemePageData>()
const matter = usePageFrontmatter<PlumeThemePostFrontmatter>()
const readingTime = useReadingTime()
const extraData = useExtraBlogData()
const createTime = computed(() => {
if (matter.value.createTime)
return matter.value.createTime.split(' ')[0].replace(/\//g, '-')
return ''
})
const categoryList = computed(() => {
return page.value.categoryList ?? []
})
const tags = computed(() => {
if (matter.value.tags) {
return matter.value.tags.slice(0, 4).map(tag => ({
name: tag,
colors: extraData.value.tagsColorsPreset[extraData.value.tagsColors[tag]],
}))
}
return []
})
const hasMeta = computed(() => readingTime.value.times || tags.value.length || createTime.value)
</script>
<template>
<div
v-if="page.isBlogPost && categoryList.length"
class="page-category-wrapper"
>
<template
v-for="({ type, name }, index) in categoryList"
:key="`${index}-${type}`"
>
<span class="category">{{ name }}</span>
<span v-if="index !== categoryList.length - 1" class="dot">&rsaquo;</span>
</template>
</div>
<h2 v-if="page.isBlogPost" class="page-title" :class="{ padding: !hasMeta }">
{{ page.title }}
</h2>
<div v-if="hasMeta" class="page-meta-wrapper">
<p v-if="readingTime.times" class="reading-time">
<IconBooks class="icon" />
<span>{{ readingTime.words }}</span>
<span>{{ readingTime.times }}</span>
</p>
<p v-if="tags.length > 0">
<IconTag class="icon" />
<span
v-for="tag in tags"
:key="tag.name"
class="tag"
:style="{ '--vp-tag-color': tag.colors[0], '--vp-tag-bg-color': tag.colors[2] }"
>
{{ tag.name }}
</span>
</p>
<p v-if="createTime" class="create-time">
<IconClock class="icon" /><span>{{ createTime }}</span>
</p>
</div>
</template>
<style scoped>
.page-category-wrapper {
padding-left: 1rem;
margin-bottom: 2rem;
font-size: 16px;
font-weight: 400;
border-left: solid 4px var(--vp-c-brand-1);
transition: border-left var(--t-color);
}
.page-category-wrapper .category {
color: var(--vp-c-text-2);
transition: color var(--t-color);
}
.page-category-wrapper .category:hover {
color: var(--vp-c-brand-1);
}
.page-category-wrapper .dot {
margin: 0 0.2rem;
color: var(--vp-c-text-3);
}
.page-title {
margin-bottom: 1rem;
font-size: 28px;
font-weight: 600;
}
.page-title.padding {
padding-bottom: 4rem;
}
.page-meta-wrapper {
display: flex;
flex-wrap: wrap;
align-items: center;
justify-content: flex-start;
padding: 1rem 0 0.5rem;
margin-bottom: 2rem;
font-size: 14px;
color: var(--vp-c-text-3);
border-bottom: solid 1px var(--vp-c-divider);
transition: color var(--t-color), border-bottom var(--t-color);
}
.page-meta-wrapper p {
display: flex;
align-items: center;
margin-right: 1rem;
}
.page-meta-wrapper .icon {
width: 14px;
height: 14px;
margin-right: 0.3rem;
}
.page-meta-wrapper .tag {
display: inline-block;
padding: 3px 5px;
margin-right: 6px;
line-height: 1;
color: var(--vp-tag-color);
background-color: var(--vp-tag-bg-color);
border-radius: 3px;
}
.page-meta-wrapper .tag:last-of-type {
margin-right: 0;
}
.page-meta-wrapper .reading-time span {
margin-right: 8px;
}
.page-meta-wrapper .reading-time span:last-of-type {
margin-right: 0;
}
.page-meta-wrapper .create-time {
flex: 1;
justify-content: right;
min-width: 110px;
margin-right: 0;
text-align: right;
}
</style>