Files
collective-governance-ai/src/pages/index.astro
Nathan Schneider 5899006717 Configure site for /cg-ai subdirectory deployment
- Set base path to '/cg-ai' in astro.config.mjs
- Update favicon to use metagov.png with dynamic base path
- Add deployment documentation for subdirectory configuration
- Rebuild dist with correct asset paths

CSS and other assets will now load correctly when deployed
to example.com/cg-ai/

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-18 15:07:50 -07:00

553 lines
21 KiB
Plaintext

---
import { readFile } from 'node:fs/promises';
import { join } from 'node:path';
import { marked } from 'marked';
// Import and parse the markdown content
const markdownPath = join(process.cwd(), 'src/data/zine.md');
const markdownContent = await readFile(markdownPath, 'utf-8');
// Split content into sections based on ## headers
const sections = markdownContent.split(/(?=^## )/m).filter(s => s.trim());
// Extract section titles for the stack TOC
const sectionTitles = sections.map(section => {
const match = section.match(/^##?\s+(.+)$/m);
return match ? match[1] : 'Untitled';
});
// Parse each section
const parsedSections = sections.map(section => {
return marked.parse(section);
});
---
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="icon" type="image/png" href={`${import.meta.env.BASE_URL}/metagov.png`} />
<meta name="viewport" content="width=device-width" />
<meta name="generator" content={Astro.generator} />
<title>Digital Zine</title>
<style is:global>
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
body {
font-family: 'Courier New', 'Courier', monospace;
line-height: 1.5;
background: #ffffff;
color: #000000;
overflow-x: hidden;
}
/* Sidebar stack navigation */
.stack-nav {
position: fixed;
left: 2rem;
top: 2rem;
bottom: 2rem;
z-index: 100;
width: 220px;
overflow-y: auto;
overflow-x: hidden;
padding-right: 0.5rem;
}
/* Scrollbar styling */
.stack-nav::-webkit-scrollbar {
width: 8px;
}
.stack-nav::-webkit-scrollbar-track {
background: #ffffff;
border: 2px solid #000000;
}
.stack-nav::-webkit-scrollbar-thumb {
background: #000000;
border: 1px solid #ffffff;
}
.stack-item {
background: #ffffff;
border: 3px solid #000000;
padding: 0.65rem 0.85rem;
margin-bottom: -3px;
font-size: 0.75rem;
font-weight: bold;
text-transform: uppercase;
cursor: pointer;
transition: all 0.2s ease;
position: relative;
box-shadow: 4px 4px 0 #000000;
line-height: 1.3;
}
.stack-item:hover {
background: #000000;
color: #ffffff;
transform: translateX(4px);
}
.stack-item.active {
background: #000000;
color: #ffffff;
box-shadow: 6px 6px 0 #666666;
}
.stack-item:first-child {
transform: skewY(-1deg);
margin-bottom: 0.5rem;
}
.stack-item:last-child {
transform: skewY(1deg);
margin-bottom: 0;
margin-top: 0.5rem;
}
/* Connection line */
.connection-line {
position: fixed;
pointer-events: none;
z-index: 50;
}
.connection-line line {
stroke: #000000;
stroke-width: 2;
stroke-dasharray: 8, 4;
}
/* Main content area */
.zine-container {
margin-left: 300px;
margin-right: 4rem;
max-width: 1200px;
padding: 2rem 4rem;
}
section {
min-height: 100vh;
display: flex;
flex-direction: column;
justify-content: center;
padding: 3rem 2rem;
border-left: 4px solid #000000;
margin-bottom: 2rem;
position: relative;
opacity: 0;
transform: translateX(-30px);
transition: opacity 0.6s ease-out, transform 0.6s ease-out;
}
section.visible {
opacity: 1;
transform: translateX(0);
}
section::before {
content: '';
position: absolute;
left: -4px;
top: 0;
width: 4px;
height: 0;
background: #000000;
transition: height 0.6s ease-out;
}
section.visible::before {
height: 100%;
}
/* ASCII background patterns for each section */
section::after {
content: attr(data-pattern);
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
font-family: 'Courier New', monospace;
font-size: 1.5rem;
line-height: 2rem;
color: rgba(0, 0, 0, 0.08);
word-wrap: break-word;
overflow: hidden;
pointer-events: none;
z-index: -1;
white-space: pre-wrap;
mask-image: linear-gradient(to bottom,
rgba(0,0,0,0.9) 0%,
rgba(0,0,0,1) 20%,
rgba(0,0,0,1) 80%,
rgba(0,0,0,0.3) 100%);
-webkit-mask-image: linear-gradient(to bottom,
rgba(0,0,0,0.9) 0%,
rgba(0,0,0,1) 20%,
rgba(0,0,0,1) 80%,
rgba(0,0,0,0.3) 100%);
}
section:nth-child(1)::after { content: '▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ ▸ '; }
section:nth-child(2)::after { content: '◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ ◆ ◇ '; }
section:nth-child(3)::after { content: '▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ ▓ ░ '; }
section:nth-child(4)::after { content: '● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ ● ○ '; }
section:nth-child(5)::after { content: '▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ ▲ △ '; }
section:nth-child(6)::after { content: '■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ ■ □ '; }
section:nth-child(7)::after { content: '◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ ◉ ◎ '; }
section:nth-child(8)::after { content: '※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ ※ ✱ '; }
section:nth-child(9)::after { content: '⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ ⬡ ⬢ '; }
section:nth-child(10)::after { content: '◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ ◈ ◇ '; }
section:nth-child(11)::after { content: '⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ ⊕ ⊖ '; }
section:nth-child(12)::after { content: '◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ ◐ ◑ '; }
section:nth-child(13)::after { content: '▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ ▣ ▢ '; }
section:nth-child(14)::after { content: '◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ ◕ ◔ '; }
section:nth-child(15)::after { content: '★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ ★ ☆ '; }
section:nth-child(16)::after { content: '◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ ◬ ◭ '; }
h1 {
font-size: clamp(2rem, 4vw, 3.5rem);
margin-bottom: 1.5rem;
font-weight: 900;
text-transform: uppercase;
letter-spacing: -0.02em;
border-bottom: 6px solid #000000;
padding-bottom: 1rem;
display: inline-block;
}
h2 {
font-size: clamp(1.5rem, 3vw, 2.5rem);
margin: 2rem 0 1rem;
font-weight: 900;
text-transform: uppercase;
text-decoration: underline;
text-decoration-thickness: 4px;
text-underline-offset: 8px;
}
p {
font-size: clamp(1rem, 1.5vw, 1.2rem);
margin: 1rem 0;
line-height: 1.7;
max-width: 70ch;
}
ul {
list-style: none;
margin: 1.5rem 0;
padding-left: 0;
max-width: 70ch;
}
li {
font-size: clamp(1rem, 1.5vw, 1.2rem);
line-height: 1.7;
margin-bottom: 1rem;
padding-left: 2rem;
position: relative;
}
li::before {
content: '▸';
position: absolute;
left: 0;
font-weight: bold;
font-size: 1.3em;
}
li:last-child {
margin-bottom: 0;
}
/* Image styling */
img {
vertical-align: text-bottom;
max-width: 100%;
height: auto;
}
p img {
display: inline-block;
vertical-align: text-bottom;
margin: 0 0.5rem;
max-height: 1.5em;
width: auto;
}
/* Zine texture overlay */
body::before {
content: '';
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-image:
repeating-linear-gradient(
0deg,
rgba(0,0,0,.03) 0px,
transparent 1px,
transparent 2px,
rgba(0,0,0,.03) 3px
);
pointer-events: none;
z-index: 999;
}
/* Rough edges effect */
section:nth-child(odd) {
background:
linear-gradient(90deg, #000 0%, transparent 0.5%) left,
linear-gradient(90deg, transparent 99.5%, #000 100%) right;
background-size: 4px 100%;
background-repeat: no-repeat;
}
/* Mobile menu button */
.menu-toggle {
display: none;
position: fixed;
top: 1rem;
right: 1rem;
z-index: 1000;
background: #ffffff;
border: 3px solid #000000;
padding: 0.75rem;
cursor: pointer;
box-shadow: 4px 4px 0 #000000;
}
.menu-toggle:active {
box-shadow: 2px 2px 0 #000000;
transform: translate(2px, 2px);
}
.hamburger {
width: 24px;
height: 18px;
position: relative;
display: flex;
flex-direction: column;
justify-content: space-between;
}
.hamburger span {
display: block;
height: 3px;
width: 100%;
background: #000000;
transition: all 0.3s ease;
}
.menu-toggle.active .hamburger span:nth-child(1) {
transform: rotate(45deg) translateY(7px);
}
.menu-toggle.active .hamburger span:nth-child(2) {
opacity: 0;
}
.menu-toggle.active .hamburger span:nth-child(3) {
transform: rotate(-45deg) translateY(-7px);
}
@media (max-width: 1024px) {
.menu-toggle {
display: block;
}
.stack-nav {
position: fixed;
left: 0;
top: 0;
transform: translateX(-100%);
width: 80%;
max-width: 300px;
height: 100vh;
padding: 4rem 1rem 1rem;
background: #ffffff;
border-right: 4px solid #000000;
display: flex;
flex-direction: column;
gap: 0;
overflow-y: auto;
transition: transform 0.3s ease;
z-index: 999;
}
.stack-nav.active {
transform: translateX(0);
}
.stack-item {
margin-bottom: -3px;
font-size: 0.75rem;
padding: 0.75rem;
transform: none !important;
margin-top: 0 !important;
}
.zine-container {
margin-left: 0;
margin-right: 0;
padding: 1rem;
}
.connection-line {
display: none;
}
}
</style>
</head>
<body>
<!-- Mobile Menu Toggle -->
<button class="menu-toggle" aria-label="Toggle menu">
<div class="hamburger">
<span></span>
<span></span>
<span></span>
</div>
</button>
<!-- Stack Navigation -->
<nav class="stack-nav">
{sectionTitles.map((title, index) => (
<div class="stack-item" data-section={index}>
{title}
</div>
))}
</nav>
<!-- Connection Line SVG -->
<svg class="connection-line" width="100%" height="100%">
<line id="connector" x1="0" y1="0" x2="0" y2="0" />
</svg>
<!-- Main Content -->
<div class="zine-container">
{parsedSections.map((html, index) => (
<section data-section-index={index} set:html={html} />
))}
</div>
<script>
// Intersection Observer for scroll animations
const observerOptions = {
threshold: 0.15,
rootMargin: '-10% 0px -10% 0px'
};
let activeSection = 0;
const observer = new IntersectionObserver((entries) => {
entries.forEach(entry => {
if (entry.isIntersecting) {
entry.target.classList.add('visible');
// Update active stack item
const sectionIndex = parseInt(entry.target.dataset.sectionIndex);
activeSection = sectionIndex;
document.querySelectorAll('.stack-item').forEach((item, index) => {
if (index === sectionIndex) {
item.classList.add('active');
} else {
item.classList.remove('active');
}
});
// Update connection line
updateConnectionLine();
}
});
}, observerOptions);
// Observe all sections
const allSections = document.querySelectorAll('section');
allSections.forEach(section => {
observer.observe(section);
});
// Make first two sections visible immediately
if (allSections.length > 0) allSections[0].classList.add('visible');
if (allSections.length > 1) allSections[1].classList.add('visible');
// Set first TOC item as active on page load
const firstStackItem = document.querySelector('.stack-item');
if (firstStackItem) {
firstStackItem.classList.add('active');
setTimeout(updateConnectionLine, 100);
}
// Update connection line between stack and content
function updateConnectionLine() {
const activeStackItem = document.querySelector('.stack-item.active');
const activeContentSection = document.querySelector(`section[data-section-index="${activeSection}"]`);
if (activeStackItem && activeContentSection && window.innerWidth > 1024) {
const stackRect = activeStackItem.getBoundingClientRect();
const contentRect = activeContentSection.getBoundingClientRect();
const line = document.getElementById('connector');
const x1 = stackRect.right;
const y1 = stackRect.top + stackRect.height / 2;
const x2 = contentRect.left;
const y2 = contentRect.top + contentRect.height / 2;
line.setAttribute('x1', x1);
line.setAttribute('y1', y1);
line.setAttribute('x2', x2);
line.setAttribute('y2', y2);
}
}
// Mobile menu toggle
const menuToggle = document.querySelector('.menu-toggle');
const stackNav = document.querySelector('.stack-nav');
if (menuToggle) {
menuToggle.addEventListener('click', () => {
menuToggle.classList.toggle('active');
stackNav.classList.toggle('active');
});
}
// Click to scroll to section
document.querySelectorAll('.stack-item').forEach((item) => {
item.addEventListener('click', () => {
const sectionIndex = item.getAttribute('data-section');
const section = document.querySelector(`section[data-section-index="${sectionIndex}"]`);
if (section) {
// Make section visible before scrolling to avoid position shift
section.classList.add('visible');
// Close mobile menu if open
if (window.innerWidth <= 1024) {
menuToggle?.classList.remove('active');
stackNav.classList.remove('active');
}
// Small delay to ensure transform completes
setTimeout(() => {
section.scrollIntoView({ behavior: 'smooth', block: 'start' });
}, 50);
}
});
});
// Update line on scroll and resize
window.addEventListener('scroll', updateConnectionLine);
window.addEventListener('resize', updateConnectionLine);
// Initial line draw
setTimeout(updateConnectionLine, 100);
</script>
</body>
</html>