193 lines
5.5 KiB
Vue

<template>
<div class="mini-blog__div" v-if="items.length">
<h2 class="large-header">{{ title }}</h2>
<section class="mini-blog">
<div v-if="items.length" class="mini-blog__grid">
<article v-for="post in items" :key="post.path" class="card">
<RouterLink :to="post.path" class="card__media" aria-label="Zum Artikel">
<img v-if="post.cover" :src="post.cover" :alt="post.title" loading="lazy" />
<div v-else class="card__placeholder" aria-hidden="true"></div>
</RouterLink>
<div class="card__body">
<RouterLink :to="post.path" class="card__title">
{{ post.title }}
</RouterLink>
<div class="card__meta">
<time v-if="post.date" class="card__date">{{ formatDate(post.date) }}</time>
<ul v-if="post.tags?.length" class="card__tags">
<li v-for="t in post.tags" :key="t" class="card__tag">#{{ t }}</li>
</ul>
</div>
<p v-if="post.excerpt" class="card__excerpt" v-html="post.excerpt"></p>
<RouterLink :to="post.path" class="card__more">{{ readMoreLinkTitle }} →</RouterLink>
</div>
</article>
</div>
<div v-else class="mini-blog__empty">
<p>Keine Artikel im aktuellen Locale gefunden.</p>
</div>
<!-- <div class="mini-blog__footer">
<RouterLink :to="articleIndex" class="mini-blog__all">Alle Artikel ansehen</RouterLink>
</div> -->
<div class="center mini-blog__footer">
<a :href="articleIndex">
<Button>
{{ showAllPostsButtonTitle }}
</Button>
</a>
</div>
</section>
</div>
</template>
<script setup>
import { computed } from "vue";
import { usePageLang, useRouteLocale } from "vuepress/client";
import articles from "@temp/mini-blog.articles.json"; // kommt aus dem Build-Hook
const locale = useRouteLocale()
const lang = usePageLang().value || 'de-DE' // Ref<string>, e.g. "de-DE" or "en-US"
const props = defineProps({
title: {
type: String,
required: true
},
readMoreLinkTitle: {
type: String,
required: true
},
showAllPostsButtonTitle: {
type: String,
required: true
},
topArticlesCount: {
type: Number,
required: true
},
});
const FALLBACK_LOCALE = '/en/';
// Slug = path without locale prefix (e.g. "news/2025-07-05-release/")
const slugOf = (a) => a.path.replace(a.locale, '');
// Articles for current locale + EN fallback for missing ones
const items = computed(() => {
const loc = locale.value || "/";
const all = articles || [];
const localeArticles = all.filter(a => a.locale === loc);
const localeSlugs = new Set(localeArticles.map(slugOf));
const fallbacks = loc === FALLBACK_LOCALE
? []
: all.filter(a => a.locale === FALLBACK_LOCALE && !localeSlugs.has(slugOf(a)));
return [...localeArticles, ...fallbacks]
.sort((a, b) => (Date.parse(b.date) || 0) - (Date.parse(a.date) || 0))
.slice(0, props.topArticlesCount);
});
const articleIndex = computed(() =>
(locale.value === "/" ? "/article/" : `${locale.value}article/`)
);
const formatDate = (d) =>
d ? new Date(d).toLocaleDateString(lang, { year: "numeric", month: "short", day: "numeric" }) : "";
</script>
<style scoped>
/* Abschnitt */
.mini-blog {
max-width: 1100px;
margin: 2rem auto 0 auto;
}
.mini-blog__div {
padding-top: 0;
}
.mini-blog__div .large-header {
padding-top: 1em !important;
}
/* Grid */
.mini-blog__grid {
margin: auto 20px;
display: grid;
gap: 1.8rem 1.2rem;
grid-template-columns: 1fr;
}
@media (min-width: 720px) {
.mini-blog__grid { grid-template-columns: repeat(2, 1fr); }
}
/* Card */
.card {
display: flex;
flex-direction: column;
overflow: hidden;
border-radius: 12px;
border: 1px solid var(--vp-c-divider, rgba(0,0,0,.08));
background: var(--vp-c-bg, #fff);
box-shadow: var(--vp-shadow-1, 0 1px 2px rgba(0,0,0,.05));
transition: transform .2s ease, box-shadow .2s ease, border-color .2s ease;
}
.card:hover {
transform: translateY(-2px);
box-shadow: var(--vp-shadow-2, 0 8px 24px rgba(0,0,0,.12));
border-color: var(--vp-c-brand-1, rgba(0,0,0,.14));
}
/* Media */
.card__media {
display: block;
aspect-ratio: 16 / 9;
overflow: hidden;
background: linear-gradient(135deg, rgba(0,0,0,.06), rgba(0,0,0,.02));
}
.card__media img {
width: 100%; height: 100%; object-fit: cover; display: block;
}
.card__placeholder {
width: 100%; height: 100%;
background: repeating-linear-gradient(45deg, rgba(0,0,0,.06), rgba(0,0,0,.06) 10px, rgba(0,0,0,.03) 10px, rgba(0,0,0,.03) 20px);
}
/* Body */
.card__body { padding: .9rem 1rem 1rem; display: grid; gap: .5rem; }
.card__title {
font-weight: 700; line-height: 1.25; text-decoration: none;
color: var(--vp-c-text-1, inherit);
}
.card__title:hover { text-decoration: underline; }
/* Meta */
.card__meta { display: flex; align-items: center; gap: .5rem; flex-wrap: wrap; font-size: .875rem; }
.card__date { opacity: .75; }
.card__tags { display: flex; gap: .35rem; list-style: none; padding: 0; margin: 0; }
.card__tag {
padding: .1rem .45rem; border-radius: 999px; font-size: .75rem;
background: var(--vp-c-brand-soft, rgba(0,0,0,.06)); color: var(--vp-c-brand-1, inherit);
}
/* Excerpt */
.card__excerpt { margin: .25rem 0 0; opacity: .9; }
/* Footer */
.card__more { margin-top: .25rem; font-weight: 600; text-decoration: none; }
.card__more:hover { text-decoration: underline; }
.mini-blog__footer {
margin-top: 1.0rem;
}
/* Empty */
.mini-blog__empty { opacity: .8; }
</style>