Files
hype404-static/src/pages/[slug]/index.astro

90 lines
2.5 KiB
Plaintext

---
import Layout from '../../layouts/Layout.astro';
import { site } from '../../lib/site.js';
import {
getAllPosts,
renderMarkdown,
getExcerpt,
getRelatedPosts,
extractFirstImage,
} from '../../lib/markdown.js';
export function getStaticPaths() {
const posts = getAllPosts();
const reserved = new Set(['about', 'rss.xml', 'robots.txt', 'sitemap.xml', 'sitemap-index.xml']);
return posts
.filter((p) => !reserved.has(p.slug))
.map((post) => ({ params: { slug: post.slug }, props: { post } }));
}
const { post } = Astro.props;
const allPosts = getAllPosts();
const html = renderMarkdown(post.body);
const related = getRelatedPosts(post, allPosts);
const excerpt = getExcerpt(post.body, 160);
const firstImage = extractFirstImage(html);
const fullImage = firstImage ? `${site.url}${firstImage}` : null;
const canonical = `${site.url}/${post.slug}/`;
const structuredData = JSON.stringify({
'@context': 'https://schema.org',
'@type': 'Article',
headline: post.title,
datePublished: post.date,
dateModified: post.date,
author: { '@type': 'Organization', name: site.name },
publisher: {
'@type': 'Organization',
name: site.name,
url: site.url,
},
description: excerpt,
...(fullImage ? { image: fullImage } : {}),
});
const formattedDate = new Date(post.date).toLocaleDateString('en-US', {
year: 'numeric',
month: 'long',
day: 'numeric',
});
---
<Layout
title={post.title}
description={excerpt}
canonical={canonical}
image={fullImage}
structuredData={structuredData}
>
<a href="/" class="back-link">← Back to Home</a>
<article class="post-content">
<h1>{post.title}</h1>
<div class="post-meta" style="margin-bottom: 30px;">
{formattedDate}
{post.tags.map((t) => (
<span class="tag"><a href={`/tag/${t}/`}>{t}</a></span>
))}
</div>
<div set:html={html} />
</article>
{related.length > 0 && (
<section class="related-posts">
<h3>Related Posts</h3>
<div class="related-posts-grid">
{related.map((r) => (
<div class="related-post-card">
<a href={`/${r.slug}/`} class="related-post-link">
<h4>{r.title}</h4>
<p class="related-post-excerpt">{r.rawExcerpt}</p>
<span class="related-post-date">
{new Date(r.date).toLocaleDateString('en-US', { year: 'numeric', month: 'short', day: 'numeric' })}
</span>
</a>
</div>
))}
</div>
</section>
)}
</Layout>