feat: complete MDX migration for blog, fix diagram fidelity and refactor styling architecture

This commit is contained in:
2026-02-17 21:36:59 +01:00
parent bff58e7cfa
commit cce6aa0935
75 changed files with 12282 additions and 12227 deletions

View File

@@ -1,268 +0,0 @@
import { NextRequest, NextResponse } from 'next/server';
import { FileExampleManager } from '../../../src/data/fileExamples';
// Simple ZIP creation without external dependencies
class SimpleZipCreator {
private files: Array<{ filename: string; content: string }> = [];
addFile(filename: string, content: string) {
this.files.push({ filename, content });
}
// Create a basic ZIP file structure
create(): number[] {
const encoder = new TextEncoder();
const chunks: number[][] = [];
let offset = 0;
const centralDirectory: Array<{
name: string;
offset: number;
size: number;
compressedSize: number;
}> = [];
// Process each file
for (const file of this.files) {
const contentBytes = Array.from(encoder.encode(file.content));
const filenameBytes = Array.from(encoder.encode(file.filename));
// Local file header
const localHeader: number[] = [];
// Local file header signature (little endian)
localHeader.push(0x50, 0x4b, 0x03, 0x04);
// Version needed to extract
localHeader.push(20, 0);
// General purpose bit flag
localHeader.push(0, 0);
// Compression method (0 = store)
localHeader.push(0, 0);
// Last modified time/date
localHeader.push(0, 0, 0, 0);
// CRC32 (0 for simplicity)
localHeader.push(0, 0, 0, 0);
// Compressed size
localHeader.push(...intToLittleEndian(contentBytes.length, 4));
// Uncompressed size
localHeader.push(...intToLittleEndian(contentBytes.length, 4));
// Filename length
localHeader.push(...intToLittleEndian(filenameBytes.length, 2));
// Extra field length
localHeader.push(0, 0);
// Add filename
localHeader.push(...filenameBytes);
chunks.push(localHeader);
chunks.push(contentBytes);
// Store info for central directory
centralDirectory.push({
name: file.filename,
offset: offset,
size: contentBytes.length,
compressedSize: contentBytes.length
});
offset += localHeader.length + contentBytes.length;
}
// Central directory
const centralDirectoryChunks: number[][] = [];
let centralDirectoryOffset = offset;
for (const entry of centralDirectory) {
const filenameBytes = Array.from(encoder.encode(entry.name));
const centralHeader: number[] = [];
// Central directory header signature
centralHeader.push(0x50, 0x4b, 0x01, 0x02);
// Version made by
centralHeader.push(20, 0);
// Version needed to extract
centralHeader.push(20, 0);
// General purpose bit flag
centralHeader.push(0, 0);
// Compression method
centralHeader.push(0, 0);
// Last modified time/date
centralHeader.push(0, 0, 0, 0);
// CRC32
centralHeader.push(0, 0, 0, 0);
// Compressed size
centralHeader.push(...intToLittleEndian(entry.compressedSize, 4));
// Uncompressed size
centralHeader.push(...intToLittleEndian(entry.size, 4));
// Filename length
centralHeader.push(...intToLittleEndian(filenameBytes.length, 2));
// Extra field length
centralHeader.push(0, 0);
// File comment length
centralHeader.push(0, 0);
// Disk number start
centralHeader.push(0, 0);
// Internal file attributes
centralHeader.push(0, 0);
// External file attributes
centralHeader.push(0, 0, 0, 0);
// Relative offset of local header
centralHeader.push(...intToLittleEndian(entry.offset, 4));
// Add filename
centralHeader.push(...filenameBytes);
centralDirectoryChunks.push(centralHeader);
}
const centralDirectorySize = centralDirectoryChunks.reduce((sum, chunk) => sum + chunk.length, 0);
// End of central directory
const endOfCentralDirectory: number[] = [];
// End of central directory signature
endOfCentralDirectory.push(0x50, 0x4b, 0x05, 0x06);
// Number of this disk
endOfCentralDirectory.push(0, 0);
// Number of the disk with the start of the central directory
endOfCentralDirectory.push(0, 0);
// Total number of entries on this disk
endOfCentralDirectory.push(...intToLittleEndian(centralDirectory.length, 2));
// Total number of entries
endOfCentralDirectory.push(...intToLittleEndian(centralDirectory.length, 2));
// Size of the central directory
endOfCentralDirectory.push(...intToLittleEndian(centralDirectorySize, 4));
// Offset of start of central directory
endOfCentralDirectory.push(...intToLittleEndian(centralDirectoryOffset, 4));
// ZIP file comment length
endOfCentralDirectory.push(0, 0);
// Combine all chunks
const result: number[] = [];
chunks.forEach(chunk => result.push(...chunk));
centralDirectoryChunks.forEach(chunk => result.push(...chunk));
result.push(...endOfCentralDirectory);
return result;
}
}
// Helper function to convert integer to little endian bytes
function intToLittleEndian(value: number, bytes: number): number[] {
const result: number[] = [];
for (let i = 0; i < bytes; i++) {
result.push((value >> (i * 8)) & 0xff);
}
return result;
}
export async function POST(request: NextRequest) {
try {
const body = await request.json();
const { fileIds } = body;
if (!Array.isArray(fileIds) || fileIds.length === 0) {
return NextResponse.json(
{ error: 'fileIds array is required and must not be empty' },
{ status: 400 }
);
}
// Get file contents
const files = await Promise.all(
fileIds.map(async (id) => {
const file = await FileExampleManager.getFileExample(id);
if (!file) {
throw new Error(`File with id ${id} not found`);
}
return file;
})
);
// Create ZIP
const zipCreator = new SimpleZipCreator();
files.forEach(file => {
zipCreator.addFile(file.filename, file.content);
});
const zipData = zipCreator.create();
const buffer = Buffer.from(new Uint8Array(zipData));
// Return ZIP file
return new Response(buffer, {
headers: {
'Content-Type': 'application/zip',
'Content-Disposition': `attachment; filename="code-examples-${Date.now()}.zip"`,
'Cache-Control': 'no-cache',
}
});
} catch (error) {
console.error('ZIP download error:', error);
const errorMessage = error instanceof Error ? error.message : 'Unknown error';
return NextResponse.json(
{ error: 'Failed to create zip file', details: errorMessage },
{ status: 500 }
);
}
}
export async function GET(request: NextRequest) {
try {
const { searchParams } = new URL(request.url);
const fileId = searchParams.get('id');
if (!fileId) {
return NextResponse.json(
{ error: 'id parameter is required' },
{ status: 400 }
);
}
const file = await FileExampleManager.getFileExample(fileId);
if (!file) {
return NextResponse.json(
{ error: 'File not found' },
{ status: 404 }
);
}
const encoder = new TextEncoder();
const content = encoder.encode(file.content);
const buffer = Buffer.from(content);
return new Response(buffer, {
headers: {
'Content-Type': getMimeType(file.language),
'Content-Disposition': `attachment; filename="${file.filename}"`,
'Cache-Control': 'no-cache',
}
});
} catch (error) {
console.error('File download error:', error);
return NextResponse.json(
{ error: 'Failed to download file' },
{ status: 500 }
);
}
}
// Helper function to get MIME type
function getMimeType(language: string): string {
const mimeTypes: Record<string, string> = {
'python': 'text/x-python',
'typescript': 'text/x-typescript',
'javascript': 'text/javascript',
'dockerfile': 'text/x-dockerfile',
'yaml': 'text/yaml',
'json': 'application/json',
'html': 'text/html',
'css': 'text/css',
'sql': 'text/x-sql',
'bash': 'text/x-shellscript',
'text': 'text/plain'
};
return mimeTypes[language] || 'text/plain';
}

View File

@@ -1,6 +1,6 @@
import { ImageResponse } from "next/og";
import { blogPosts } from "../../../src/data/blogPosts";
import { blogThumbnails } from "../../../src/data/blogThumbnails";
import { allPosts } from "contentlayer/generated";
import { blogThumbnails } from "../../../src/components/blog/blogThumbnails";
import { OGImageTemplate } from "../../../src/components/OGImageTemplate";
import { getOgFonts, OG_IMAGE_SIZE } from "../../../src/lib/og-helper";
@@ -14,7 +14,7 @@ export default async function Image({
params: Promise<{ slug: string }>;
}) {
const { slug } = await params;
const post = blogPosts.find((p) => p.slug === slug);
const post = allPosts.find((p) => p.slug === slug);
const thumbnail = blogThumbnails[slug];
const title = post?.title || "Marc Mintel";

View File

@@ -1,17 +1,18 @@
import * as React from "react";
import type { Metadata } from "next";
import { notFound } from "next/navigation";
import { blogPosts } from "../../../src/data/blogPosts";
import { allPosts } from "contentlayer/generated";
import { BlogPostHeader } from "../../../src/components/blog/BlogPostHeader";
import { Section } from "../../../src/components/Section";
import { Reveal } from "../../../src/components/Reveal";
import { BlogPostClient } from "../../../src/components/BlogPostClient";
import { PostComponents } from "../../../src/components/blog/posts";
import { TextSelectionShare } from "../../../src/components/TextSelectionShare";
import { BlogPostStickyBar } from "../../../src/components/blog/BlogPostStickyBar";
import { MDXRemote } from "next-mdx-remote/rsc";
import { MDXComponents } from "../../../mdx-components";
export async function generateStaticParams() {
return blogPosts.map((post) => ({
return allPosts.map((post) => ({
slug: post.slug,
}));
}
@@ -22,7 +23,7 @@ export async function generateMetadata({
params: Promise<{ slug: string }>;
}): Promise<Metadata> {
const { slug } = await params;
const post = blogPosts.find((p) => p.slug === slug);
const post = allPosts.find((p) => p.slug === slug);
if (!post) return {};
@@ -48,7 +49,7 @@ export default async function BlogPostPage({
params: Promise<{ slug: string }>;
}) {
const { slug } = await params;
const post = blogPosts.find((p) => p.slug === slug);
const post = allPosts.find((p) => p.slug === slug);
if (!post) {
notFound();
@@ -60,11 +61,9 @@ export default async function BlogPostPage({
year: "numeric",
});
const wordCount = post.description.split(/\s+/).length + 300; // Average post length
const wordCount = post.description.split(/\s+/).length + 300;
const readingTime = Math.max(1, Math.ceil(wordCount / 200));
const PostContent = PostComponents[slug];
return (
<div className="flex flex-col gap-8 md:gap-12 py-8 md:py-24 overflow-hidden">
<BlogPostClient readingTime={readingTime} title={post.title} />
@@ -78,7 +77,6 @@ export default async function BlogPostPage({
/>
<main id="post-content">
{/* Sticky Progress Bar */}
<BlogPostStickyBar
title={post.title}
url={`https://mintel.me/blog/${slug}`}
@@ -89,7 +87,7 @@ export default async function BlogPostPage({
<Reveal delay={0.4} width="100%">
{post.tags && post.tags.length > 0 && (
<div className="flex flex-wrap gap-2 mb-10 md:mb-12">
{post.tags.map((tag, index) => (
{post.tags?.map((tag) => (
<span
key={tag}
className="px-2.5 py-1 bg-slate-50 border border-slate-100 rounded text-[9px] md:text-[10px] font-mono text-slate-500 uppercase tracking-widest"
@@ -100,13 +98,12 @@ export default async function BlogPostPage({
</div>
)}
{PostContent ? (
<PostContent />
) : (
<div className="p-8 bg-slate-50 border border-slate-200 rounded-lg italic text-slate-500">
Inhalt wird bald veröffentlicht...
</div>
)}
<div className="max-w-none">
<MDXRemote
source={post.body.raw.replace(/^---[\s\S]*?\n---\s*/, '')}
components={MDXComponents}
/>
</div>
</Reveal>
</div>
</Section>

View File

@@ -4,7 +4,7 @@ import * as React from "react";
import { useState, useEffect } from "react";
import { MediumCard } from "../../src/components/MediumCard";
import { BlogCommandBar } from "../../src/components/blog/BlogCommandBar";
import { blogPosts } from "../../src/data/blogPosts";
import { allPosts as contentPosts } from "contentlayer/generated";
import { SectionHeader } from "../../src/components/SectionHeader";
import { Reveal } from "../../src/components/Reveal";
import { Section } from "../../src/components/Section";
@@ -19,7 +19,7 @@ export default function BlogPage() {
// Memoize allPosts
const allPosts = React.useMemo(
() =>
[...blogPosts].sort(
[...contentPosts].sort(
(a, b) => new Date(b.date).getTime() - new Date(a.date).getTime(),
),
[],

View File

@@ -1,5 +1,5 @@
import { MetadataRoute } from 'next';
import { blogPosts } from '../src/data/blogPosts';
import { allPosts } from 'contentlayer/generated';
import { technologies } from './technologies/[slug]/data';
/**
@@ -28,7 +28,7 @@ export default function sitemap(): MetadataRoute.Sitemap {
}));
// 2. Dynamic Blog Posts
const blogRoutes = blogPosts.map((post) => ({
const blogRoutes = allPosts.map((post) => ({
url: `${baseUrl}/blog/${post.slug}`,
lastModified: new Date(post.date),
changeFrequency: 'monthly' as const,
@@ -44,7 +44,7 @@ export default function sitemap(): MetadataRoute.Sitemap {
}));
// 4. Tag Pages
const allTags = [...new Set(blogPosts.flatMap((post) => post.tags))];
const allTags = [...new Set(allPosts.flatMap((post) => post.tags))];
const tagRoutes = allTags.map((tag) => ({
url: `${baseUrl}/tags/${tag}`,
lastModified: new Date(),

View File

@@ -1,12 +1,11 @@
import * as React from "react";
import Link from "next/link";
import { blogPosts } from "../../../src/data/blogPosts";
import { allPosts } from "contentlayer/generated";
import { MediumCard } from "../../../src/components/MediumCard";
import { Reveal } from "../../../src/components/Reveal";
export async function generateStaticParams() {
const allTags = Array.from(
new Set(blogPosts.flatMap((post) => post.tags || [])),
new Set(allPosts.flatMap((post) => post.tags || [])),
);
return allTags.map((tag) => ({
tag,
@@ -19,7 +18,7 @@ export default async function TagPage({
params: Promise<{ tag: string }>;
}) {
const { tag } = await params;
const posts = blogPosts.filter((post) => post.tags?.includes(tag));
const posts = allPosts.filter((post) => post.tags?.includes(tag));
return (
<div className="max-w-3xl mx-auto px-4 py-8">