38 KiB
title, slug, published, order, layout, rightSidebar, aiChat, footer
| title | slug | published | order | layout | rightSidebar | aiChat | footer |
|---|---|---|---|---|---|---|---|
| Docs | docs | true | 0 | sidebar | true | true | true |
Getting Started
Reference documentation for setting up, customizing, and deploying this markdown framework.
How publishing works: Write posts in markdown, run npm run sync for development or npm run sync:prod for production, and they appear on your live site immediately. No rebuild or redeploy needed. Convex handles real-time data sync, so connected browsers update automatically.
Sync commands:
Development:
npm run sync- Sync markdown contentnpm run sync:discovery- Update discovery files (AGENTS.md, llms.txt)npm run sync:all- Sync content + discovery files together
Production:
npm run sync:prod- Sync markdown contentnpm run sync:discovery:prod- Update discovery filesnpm run sync:all:prod- Sync content + discovery files together
Quick start
git clone https://github.com/waynesutton/markdown-site.git
cd markdown-site
npm install
npx convex dev
npm run sync # development
npm run sync:prod # production
npm run dev
Open http://localhost:5173 to view locally.
Requirements
- Node.js 18+
- Convex account (free at convex.dev)
- Netlify account (free at netlify.com)
Project structure
markdown-site/
├── content/
│ ├── blog/ # Blog posts (.md)
│ └── pages/ # Static pages (.md)
├── convex/
│ ├── schema.ts # Database schema
│ ├── posts.ts # Post queries/mutations
│ ├── pages.ts # Page queries/mutations
│ ├── http.ts # API endpoints
│ └── rss.ts # RSS generation
├── netlify/
│ └── edge-functions/ # Netlify edge functions
│ ├── rss.ts # RSS proxy
│ ├── sitemap.ts # Sitemap proxy
│ ├── api.ts # API proxy
│ └── botMeta.ts # OG crawler detection
├── src/
│ ├── components/ # React components
│ ├── context/ # Theme context
│ ├── pages/ # Route components
│ └── styles/ # CSS
├── public/
│ ├── images/ # Static images
│ ├── raw/ # Generated raw markdown files
│ ├── robots.txt # Crawler rules
│ └── llms.txt # AI discovery
└── netlify.toml # Deployment config
Content
Blog posts
Create files in content/blog/ with frontmatter:
---
title: "Post Title"
description: "SEO description"
date: "2025-01-15"
slug: "url-path"
published: true
tags: ["tag1", "tag2"]
readTime: "5 min read"
image: "/images/og-image.png"
---
Content here...
| Field | Required | Description |
|---|---|---|
title |
Yes | Post title |
description |
Yes | SEO description |
date |
Yes | YYYY-MM-DD format |
slug |
Yes | URL path (unique) |
published |
Yes | true to show |
tags |
Yes | Array of strings |
readTime |
No | Display time estimate |
image |
No | OG image and featured card thumbnail. See Using Images in Blog Posts for markdown and HTML syntax |
excerpt |
No | Short text for card view |
featured |
No | true to show in featured section |
featuredOrder |
No | Order in featured (lower = first) |
authorName |
No | Author display name shown next to date |
authorImage |
No | Round author avatar image URL |
layout |
No | Set to "sidebar" for docs-style layout with TOC |
Static pages
Create files in content/pages/ with frontmatter:
---
title: "Page Title"
slug: "url-path"
published: true
order: 1
---
Content here...
| Field | Required | Description |
|---|---|---|
title |
Yes | Nav link text |
slug |
Yes | URL path |
published |
Yes | true to show |
order |
No | Nav order (lower = first) |
showInNav |
No | Show in navigation menu (default: true) |
excerpt |
No | Short text for card view |
image |
No | Thumbnail for featured card view |
featured |
No | true to show in featured section |
featuredOrder |
No | Order in featured (lower = first) |
authorName |
No | Author display name shown next to date |
authorImage |
No | Round author avatar image URL |
layout |
No | Set to "sidebar" for docs-style layout with TOC |
rightSidebar |
No | Enable right sidebar with CopyPageDropdown (opt-in, requires explicit true) |
Hide pages from navigation: Set showInNav: false to keep a page published and accessible via direct URL, but hidden from the navigation menu. Pages with showInNav: false remain searchable and available via API endpoints. Useful for pages you want to link directly but not show in the main nav.
Sidebar layout
Posts and pages can use a docs-style layout with a table of contents sidebar. Add layout: "sidebar" to the frontmatter:
---
title: "Documentation"
slug: "docs"
published: true
layout: "sidebar"
---
# Introduction
## Section One
### Subsection
## Section Two
Features:
- Left sidebar displays table of contents extracted from H1, H2, H3 headings
- Two-column layout: 220px sidebar + flexible content area
- Sidebar only appears if headings exist in the content
- Active heading highlighting as you scroll
- Smooth scroll navigation when clicking TOC links
- Mobile responsive: stacks to single column below 1024px
- Works for both blog posts and static pages
The sidebar extracts headings automatically from your markdown content. No manual TOC needed.
Right sidebar
When enabled in siteConfig.rightSidebar.enabled, posts and pages can display a right sidebar containing the CopyPageDropdown at 1135px+ viewport width.
Configuration:
Enable globally in src/config/siteConfig.ts:
rightSidebar: {
enabled: true, // Set to false to disable right sidebar globally
minWidth: 1135, // Minimum viewport width to show sidebar
},
Control per post/page with frontmatter:
---
title: "My Post"
rightSidebar: true # Enable right sidebar for this post
---
Features:
- Right sidebar appears at 1135px+ viewport width
- Contains CopyPageDropdown with all sharing options
- Three-column layout: left sidebar (TOC), main content, right sidebar
- CopyPageDropdown automatically moves from nav to right sidebar when enabled
- Hidden below 1135px breakpoint, CopyPageDropdown returns to nav
- Per-post/page control via
rightSidebar: truefrontmatter field - Opt-in only: right sidebar only appears when explicitly enabled in frontmatter
Use cases:
- Keep CopyPageDropdown accessible on wide screens without cluttering the nav
- Provide quick access to sharing options while reading long content
- Works alongside left sidebar TOC for comprehensive navigation
Example for blog post:
---
title: "My Tutorial"
description: "A detailed guide"
date: "2025-01-20"
slug: "my-tutorial"
published: true
tags: ["tutorial"]
layout: "sidebar"
---
# Introduction
## Getting Started
### Prerequisites
## Advanced Topics
How frontmatter works
Frontmatter is the YAML metadata at the top of each markdown file between --- markers. Here is how it flows through the system:
Content directories:
content/blog/*.mdcontains blog posts with frontmattercontent/pages/*.mdcontains static pages with frontmatter
Processing flow:
- Markdown files in
content/blog/andcontent/pages/contain YAML frontmatter scripts/sync-posts.tsusesgray-matterto parse frontmatter and validate required fields- Parsed data is sent to Convex mutations (
api.posts.syncPostsPublic,api.pages.syncPagesPublic) convex/schema.tsdefines the database structure for storing the data
Adding a new frontmatter field:
To add a custom frontmatter field, update these files:
- The interface in
scripts/sync-posts.ts(PostFrontmatterorPageFrontmatter) - The parsing logic in
parseMarkdownFile()orparsePageFile()functions - The schema in
convex/schema.ts - The sync mutation in
convex/posts.tsorconvex/pages.ts
Syncing content
Development:
npm run sync # Sync markdown content
npm run sync:discovery # Update discovery files (AGENTS.md, llms.txt)
npm run sync:all # Sync content + discovery files together
Production:
npm run sync:prod # Sync markdown content
npm run sync:discovery:prod # Update discovery files
npm run sync:all:prod # Sync content + discovery files together
Sync everything together:
npm run sync:all # Development: content + discovery
npm run sync:all:prod # Production: content + discovery
When to sync vs deploy
| What you're changing | Command | Timing |
|---|---|---|
Blog posts in content/blog/ |
npm run sync |
Instant (no rebuild) |
Pages in content/pages/ |
npm run sync |
Instant (no rebuild) |
| Featured items (via frontmatter) | npm run sync |
Instant (no rebuild) |
| Site config changes | npm run sync:discovery |
Updates discovery files |
| Import external URL | npm run import then sync |
Instant (no rebuild) |
Images in public/images/ |
Git commit + push | Requires rebuild |
siteConfig in Home.tsx |
Redeploy | Requires rebuild |
| Logo gallery config | Redeploy | Requires rebuild |
| React components/styles | Redeploy | Requires rebuild |
Markdown content syncs instantly to Convex. Images and source code require pushing to GitHub for Netlify to rebuild.
Configuration
Fork configuration
After forking, you have two options to configure your site:
Option 1: Automated (Recommended)
cp fork-config.json.example fork-config.json
# Edit fork-config.json with your site information
npm run configure
This updates all 11 configuration files in one command. See FORK_CONFIG.md for the full JSON schema and options.
Option 2: Manual
Follow the step-by-step guide in FORK_CONFIG.md to update each file manually.
Files updated by configuration
| File | What to update |
|---|---|
src/config/siteConfig.ts |
Site name, title, intro, bio, blog page, logo gallery, GitHub contributions, right sidebar configuration |
src/pages/Home.tsx |
Intro paragraph text, footer links |
convex/http.ts |
SITE_URL, SITE_NAME, description strings (3 locations) |
convex/rss.ts |
SITE_URL, SITE_TITLE, SITE_DESCRIPTION (RSS feeds) |
src/pages/Post.tsx |
SITE_URL, SITE_NAME, DEFAULT_OG_IMAGE (OG tags) |
index.html |
Title, meta description, OG tags, JSON-LD |
public/llms.txt |
Site name, URL, description, topics |
public/robots.txt |
Sitemap URL and header comment |
public/openapi.yaml |
API title, server URL, site name in examples |
public/.well-known/ai-plugin.json |
Site name, descriptions |
src/context/ThemeContext.tsx |
Default theme |
Site title and description metadata
These files contain the main site description text. Update them with your own tagline:
| File | What to change |
|---|---|
index.html |
meta description, og:description, twitter:description, JSON-LD |
README.md |
Main description at top of file |
src/config/siteConfig.ts |
name, title, and bio fields |
src/pages/Home.tsx |
Intro paragraph (hardcoded JSX with links) |
convex/http.ts |
SITE_NAME constant and description strings (3 locations) |
convex/rss.ts |
SITE_TITLE and SITE_DESCRIPTION constants |
public/llms.txt |
Header quote, Name, and Description fields |
public/openapi.yaml |
API title and example site name |
AGENTS.md |
Project overview section |
content/blog/about-this-blog.md |
Title, description, excerpt, and opening paragraph |
content/pages/about.md |
excerpt field and opening paragraph |
content/pages/docs.md |
Opening description paragraph |
Backend constants (convex/http.ts and convex/rss.ts):
// convex/http.ts
const SITE_URL = "https://your-site.netlify.app";
const SITE_NAME = "Your Site Name";
// convex/rss.ts
const SITE_URL = "https://your-site.netlify.app";
const SITE_TITLE = "Your Site Name";
const SITE_DESCRIPTION = "Your site description for RSS feeds.";
Post page constants (src/pages/Post.tsx):
const SITE_URL = "https://your-site.netlify.app";
const SITE_NAME = "Your Site Name";
const DEFAULT_OG_IMAGE = "/images/og-default.svg";
These constants affect RSS feeds, API responses, sitemaps, and social sharing metadata.
Site settings
Edit src/config/siteConfig.ts:
export default {
name: "Site Name",
title: "Tagline",
logo: "/images/logo.svg", // null to hide homepage logo
intro: "Introduction text...",
bio: "Bio text...",
// Blog page configuration
blogPage: {
enabled: true, // Enable /blog route
showInNav: true, // Show in navigation
title: "Blog", // Nav link and page title
order: 0, // Nav order (lower = first)
},
// Hardcoded navigation items for React routes
hardcodedNavItems: [
{
slug: "stats",
title: "Stats",
order: 10,
showInNav: true, // Set to false to hide from nav
},
{
slug: "write",
title: "Write",
order: 20,
showInNav: true,
},
],
// Inner page logo configuration
innerPageLogo: {
enabled: true, // Set to false to hide logo on inner pages
size: 28, // Logo height in pixels (keeps aspect ratio)
},
// Featured section
featuredViewMode: "list", // 'list' or 'cards'
showViewToggle: true,
// Logo gallery (static grid or scrolling marquee)
logoGallery: {
enabled: true, // false to hide
images: [{ src: "/images/logos/logo.svg", href: "https://example.com" }],
position: "above-footer",
speed: 30,
title: "Built with",
scrolling: false, // false = static grid, true = scrolling marquee
maxItems: 4, // Number of logos when scrolling is false
},
links: {
docs: "/docs",
convex: "https://convex.dev",
},
};
Logo configuration:
logo: Homepage logo path (set tonullto hide). Usespublic/images/logo.svgby default.innerPageLogo: Logo shown on blog page, posts, and static pages. Desktop: top left. Mobile: top right. Setenabled: falseto hide on inner pages while keeping homepage logo.
Navigation structure:
Navigation combines three sources sorted by order:
- Blog link (if
blogPage.enabledandblogPage.showInNavare true) - Hardcoded nav items (React routes from
hardcodedNavItems) - Markdown pages (from
content/pages/withshowInNav: true)
All items sort by order (lower first), then alphabetically by title.
Featured items
Posts and pages appear in the featured section when marked with featured: true in frontmatter.
Add to featured section:
# In any post or page frontmatter
featured: true
featuredOrder: 1
excerpt: "Short description for card view."
image: "/images/thumbnail.png"
Then run npm run sync or npm run sync:all. No redeploy needed.
| Field | Description |
|---|---|
featured |
Set true to show in featured section |
featuredOrder |
Order in featured section (lower = first) |
excerpt |
Short text shown on card view |
image |
Thumbnail for card view (displays as square) |
Thumbnail images: In card view, the image field displays as a square thumbnail above the title. Non-square images are automatically cropped to center. Square thumbnails: 400x400px minimum (800x800px for retina).
Posts without images: Cards display without the image area. The card shows just the title and excerpt with adjusted padding.
Ordering: Items with featuredOrder appear first (lower numbers first). Items without featuredOrder appear after, sorted by creation time.
Display options (in siteConfig):
// In src/pages/Home.tsx
const siteConfig = {
featuredViewMode: "list", // 'list' or 'cards'
showViewToggle: true, // Let users switch views
};
GitHub contributions graph
Display your GitHub contribution activity on the homepage. Configure in siteConfig:
gitHubContributions: {
enabled: true, // Set to false to hide
username: "yourusername", // Your GitHub username
showYearNavigation: true, // Show arrows to navigate between years
linkToProfile: true, // Click graph to open GitHub profile
title: "GitHub Activity", // Optional title above the graph
},
| Option | Description |
|---|---|
enabled |
true to show, false to hide |
username |
Your GitHub username |
showYearNavigation |
Show prev/next year navigation |
linkToProfile |
Click graph to visit GitHub profile |
title |
Text above graph (undefined to hide) |
Theme-aware colors match each site theme. Uses public API (no GitHub token required).
Visitor map
Display real-time visitor locations on a world map on the stats page. Uses Netlify's built-in geo detection (no third-party API needed). Privacy friendly: only stores city, country, and coordinates. No IP addresses stored.
visitorMap: {
enabled: true, // Set to false to hide
title: "Live Visitors", // Optional title above the map
},
| Option | Description |
|---|---|
enabled |
true to show, false to hide |
title |
Text above map (undefined to hide) |
The map displays with theme-aware colors. Visitor dots pulse to indicate live sessions. Location data comes from Netlify's automatic geo headers at the edge.
Logo gallery
The homepage includes a logo gallery that can scroll infinitely or display as a static grid. Each logo can link to a URL.
// In src/config/siteConfig.ts
logoGallery: {
enabled: true, // false to hide
images: [
{ src: "/images/logos/logo1.svg", href: "https://example.com" },
{ src: "/images/logos/logo2.svg", href: "https://another.com" },
],
position: "above-footer", // or 'below-featured'
speed: 30, // Seconds for one scroll cycle
title: "Built with", // undefined to hide
scrolling: false, // false = static grid, true = scrolling marquee
maxItems: 4, // Number of logos when scrolling is false
},
| Option | Description |
|---|---|
enabled |
true to show, false to hide |
images |
Array of { src, href } objects |
position |
'above-footer' or 'below-featured' |
speed |
Seconds for one scroll cycle (lower = faster) |
title |
Text above gallery (undefined to hide) |
scrolling |
true for infinite scroll, false for static grid |
maxItems |
Max logos to show when scrolling is false (default: 4) |
Display modes:
scrolling: true: Infinite horizontal scroll with all logosscrolling: false: Static centered grid showing firstmaxItemslogos
To add logos:
- Add SVG/PNG files to
public/images/logos/ - Update the
imagesarray withsrcpaths andhrefURLs - Push to GitHub (requires rebuild)
To disable: Set enabled: false
To remove samples: Delete files from public/images/logos/ or clear the images array.
Blog page
The site supports a dedicated blog page at /blog with two view modes: list view (year-grouped posts) and card view (thumbnail grid). Configure in src/config/siteConfig.ts:
blogPage: {
enabled: true, // Enable /blog route
showInNav: true, // Show in navigation
title: "Blog", // Nav link and page title
order: 0, // Nav order (lower = first)
viewMode: "list", // Default view: "list" or "cards"
showViewToggle: true, // Show toggle button to switch views
},
displayOnHomepage: true, // Show posts on homepage
| Option | Description |
|---|---|
enabled |
Enable the /blog route |
showInNav |
Show Blog link in navigation |
title |
Text for nav link and page heading |
order |
Position in navigation (lower = first) |
viewMode |
Default view: "list" or "cards" |
showViewToggle |
Show toggle button to switch views |
displayOnHomepage |
Show post list on homepage |
View modes:
- List view: Year-grouped posts with titles, read time, and dates
- Card view: Grid of cards showing thumbnails, titles, excerpts, and metadata
Card view details:
Cards display post thumbnails (from image frontmatter field), titles, excerpts (or descriptions), read time, and dates. Posts without images show cards without thumbnail areas. Grid is responsive: 3 columns on desktop, 2 on tablet, 1 on mobile.
Display options:
- Homepage only:
displayOnHomepage: true,blogPage.enabled: false - Blog page only:
displayOnHomepage: false,blogPage.enabled: true - Both:
displayOnHomepage: true,blogPage.enabled: true
Navigation order: The Blog link merges with page links and sorts by order. Pages use the order field in frontmatter. Set blogPage.order: 5 to position Blog after pages with order 0-4.
View preference: User's view mode choice is saved to localStorage and persists across page visits.
Scroll-to-top button
A scroll-to-top button appears after scrolling down. Configure in src/components/Layout.tsx:
const scrollToTopConfig: Partial<ScrollToTopConfig> = {
enabled: true, // Set to false to disable
threshold: 300, // Show after scrolling 300px
smooth: true, // Smooth scroll animation
};
| Option | Description |
|---|---|
enabled |
true to show, false to hide |
threshold |
Pixels scrolled before button appears |
smooth |
true for smooth scroll, false for jump |
Uses Phosphor ArrowUp icon and works with all themes.
Theme
Default: tan. Options: dark, light, tan, cloud.
Edit src/context/ThemeContext.tsx:
const DEFAULT_THEME: Theme = "tan";
Font
Configure the font in src/config/siteConfig.ts:
export const siteConfig: SiteConfig = {
// ... other config
fontFamily: "serif", // Options: "serif", "sans", or "monospace"
};
Or edit src/styles/global.css directly:
body {
/* Sans-serif */
font-family:
-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
/* Serif (default) */
font-family: "New York", ui-serif, Georgia, serif;
/* Monospace */
font-family: "IBM Plex Mono", "Liberation Mono", ui-monospace, monospace;
}
Available options: serif (default), sans, or monospace.
Font Sizes
All font sizes use CSS variables in :root. Customize by editing:
:root {
--font-size-base: 16px;
--font-size-sm: 13px;
--font-size-lg: 17px;
--font-size-blog-content: 17px;
--font-size-post-title: 32px;
}
Mobile sizes defined in @media (max-width: 768px) block.
Images
| Image | Location | Size |
|---|---|---|
| Favicon | public/favicon.svg |
512x512 |
| Site logo | public/images/logo.svg |
512x512 |
| Default OG image | public/images/og-default.svg |
1200x630 |
| Post images | public/images/ |
Any |
Images require git deploy. Images are served as static files from your repository, not synced to Convex. After adding images to public/images/:
- Commit the image files to git
- Push to GitHub
- Wait for Netlify to rebuild
The npm run sync command only syncs markdown text content. Images are deployed when Netlify builds your site. Use npm run sync:discovery to update discovery files (AGENTS.md, llms.txt) when site configuration changes.
Adding images to posts: You can add images using markdown syntax  or HTML <img> tags. The site uses rehypeRaw and rehypeSanitize to safely render HTML in markdown content. See Using Images in Blog Posts for complete examples and best practices.
Logo options:
- Homepage logo: Configured via
logoinsiteConfig.ts. Set tonullto hide. - Inner page logo: Configured via
innerPageLogoinsiteConfig.ts. Shows on blog page, posts, and static pages. Desktop: top left corner. Mobile: top right corner (smaller). Setenabled: falseto hide on inner pages while keeping homepage logo.
Search
Press Command+K (Mac) or Ctrl+K (Windows/Linux) to open the search modal. Click the search icon in the nav or use the keyboard shortcut.
Features:
- Real-time results as you type
- Keyboard navigation (arrow keys, Enter, Escape)
- Result snippets with context around matches
- Distinguishes between posts and pages
- Works with all four themes
Search uses Convex full text search indexes. No configuration needed.
Mobile menu
On mobile and tablet screens, a hamburger menu provides navigation. The menu slides out from the left with:
- Keyboard navigation (Escape to close)
- Focus trap for accessibility
- Auto-close on route change
The menu appears automatically on screens under 768px wide.
Copy Page dropdown
Each post and page includes a share dropdown with options:
| Option | Description |
|---|---|
| Copy page | Copies formatted markdown to clipboard |
| Open in ChatGPT | Opens ChatGPT with raw markdown URL |
| Open in Claude | Opens Claude with raw markdown URL |
| Open in Perplexity | Opens Perplexity with raw markdown URL |
| View as Markdown | Opens raw .md file in new tab |
| Download as SKILL.md | Downloads skill file for AI agent training |
Raw markdown URLs: AI service links use GitHub raw URLs to fetch markdown content. This bypasses Netlify edge functions and provides reliable access for AI services.
Git push required for AI links: The "Open in ChatGPT," "Open in Claude," and "Open in Perplexity" options use GitHub raw URLs. For these to work, you must push your content to GitHub with git push. The npm run sync command syncs content to Convex for your live site, but AI services fetch directly from GitHub.
| What you want | Command needed |
|---|---|
| Content visible on your site | npm run sync or sync:prod |
| Discovery files updated | npm run sync:discovery or sync:discovery:prod |
| AI links (ChatGPT/Claude/Perplexity) | git push to GitHub |
| Both content and discovery | npm run sync:all or sync:all:prod |
Download as SKILL.md: Downloads the content formatted as an Anthropic Agent Skills file with metadata, triggers, and instructions sections.
Real-time stats
The /stats page displays real-time analytics:
- Active visitors (with per-page breakdown)
- Total page views
- Unique visitors
- Views by page (sorted by count)
All stats update automatically via Convex subscriptions.
API endpoints
| Endpoint | Description |
|---|---|
/stats |
Real-time analytics |
/rss.xml |
RSS feed (descriptions) |
/rss-full.xml |
RSS feed (full content) |
/sitemap.xml |
XML sitemap |
/api/posts |
JSON post list |
/api/post?slug=xxx |
Single post (JSON) |
/api/post?slug=xxx&format=md |
Single post (markdown) |
/api/export |
All posts with full content |
/raw/{slug}.md |
Static raw markdown file |
/.well-known/ai-plugin.json |
AI plugin manifest |
/openapi.yaml |
OpenAPI 3.0 specification |
/llms.txt |
AI agent discovery |
Raw markdown files
When you run npm run sync (development) or npm run sync:prod (production), static .md files are generated in public/raw/ for each published post and page. Use npm run sync:all or npm run sync:all:prod to sync content and update discovery files together.
Access pattern: /raw/{slug}.md
Examples:
/raw/setup-guide.md/raw/about.md
These files include a metadata header with type, date, reading time, and tags. Access via the "View as Markdown" option in the Copy Page dropdown.
Markdown tables
Tables render with GitHub-style formatting:
- Clean borders across all themes
- Mobile responsive with horizontal scroll
- Theme-aware alternating row colors
- Hover states for readability
Example:
| Feature | Status |
|---|---|
| Borders | Clean |
| Mobile | Scroll |
| Themes | All |
Collapsible sections
Create expandable/collapsible content using HTML <details> and <summary> tags:
<details>
<summary>Click to expand</summary>
Hidden content here. Supports markdown: - Lists - **Bold** and _italic_ - Code
blocks
</details>
Expanded by default: Add the open attribute:
<details open>
<summary>Already expanded</summary>
This section starts open.
</details>
Nested sections: You can nest <details> inside other <details> for multi-level collapsible content.
Collapsible sections work with all four themes and are styled to match the site design.
Import external content
Use Firecrawl to import articles from external URLs:
npm run import https://example.com/article
Setup:
- Get an API key from firecrawl.dev
- Add
FIRECRAWL_API_KEY=fc-xxxto.env.local
The import command creates local markdown files only. It does not interact with Convex directly.
After importing:
npm run syncto push to developmentnpm run sync:prodto push to production- Use
npm run sync:allornpm run sync:all:prodto sync content and update discovery files together
There is no npm run import:prod because import creates local files and sync handles the target environment.
Imported posts are drafts (published: false). Review, edit, set published: true, then sync.
Deployment
Netlify setup
- Connect GitHub repo to Netlify
- Build command:
npm ci --include=dev && npx convex deploy --cmd 'npm run build' - Publish directory:
dist - Add env variables:
CONVEX_DEPLOY_KEY(from Convex Dashboard > Project Settings > Deploy Key)VITE_CONVEX_URL(your production Convex URL, e.g.,https://your-deployment.convex.cloud)
Both are required: deploy key for builds, URL for edge function runtime.
Convex production
npx convex deploy
Edge functions
RSS, sitemap, and API routes are handled by Netlify Edge Functions in netlify/edge-functions/. They dynamically read VITE_CONVEX_URL from the environment. No manual URL configuration needed.
Convex schema
// convex/schema.ts
export default defineSchema({
posts: defineTable({
slug: v.string(),
title: v.string(),
description: v.string(),
content: v.string(),
date: v.string(),
published: v.boolean(),
tags: v.array(v.string()),
readTime: v.optional(v.string()),
image: v.optional(v.string()),
excerpt: v.optional(v.string()), // For card view
featured: v.optional(v.boolean()), // Show in featured section
featuredOrder: v.optional(v.number()), // Order in featured (lower = first)
authorName: v.optional(v.string()), // Author display name
authorImage: v.optional(v.string()), // Author avatar image URL
lastSyncedAt: v.number(),
})
.index("by_slug", ["slug"])
.index("by_published", ["published"])
.index("by_featured", ["featured"]),
pages: defineTable({
slug: v.string(),
title: v.string(),
content: v.string(),
published: v.boolean(),
order: v.optional(v.number()),
excerpt: v.optional(v.string()), // For card view
image: v.optional(v.string()), // Thumbnail for featured cards
featured: v.optional(v.boolean()), // Show in featured section
featuredOrder: v.optional(v.number()), // Order in featured (lower = first)
authorName: v.optional(v.string()), // Author display name
authorImage: v.optional(v.string()), // Author avatar image URL
lastSyncedAt: v.number(),
})
.index("by_slug", ["slug"])
.index("by_published", ["published"])
.index("by_featured", ["featured"]),
});
Troubleshooting
Posts not appearing
- Check
published: truein frontmatter - Run
npm run syncfor development - Run
npm run sync:prodfor production - Use
npm run sync:allornpm run sync:all:prodto sync content and update discovery files together - Verify in Convex dashboard
RSS/Sitemap errors
- Verify
VITE_CONVEX_URLis set in Netlify - Test Convex HTTP URL:
https://your-deployment.convex.site/rss.xml - Check edge functions in
netlify/edge-functions/
Build failures
- Verify
CONVEX_DEPLOY_KEYis set in Netlify - Ensure
@types/nodeis in devDependencies - Build command must include
--include=dev - Check Node.js version (18+)