[ PROMPT_NODE_27873 ]
Astro
[ SKILL_DOCUMENTATION ]
# Astro Web Framework
## Overview
Astro is a web framework designed for content-rich websites — blogs, docs, portfolios, marketing sites, and e-commerce. Its core innovation is the **Islands Architecture**: by default, Astro ships zero JavaScript to the browser. Interactive components are selectively hydrated as isolated "islands." Astro supports React, Vue, Svelte, Solid, and other UI frameworks simultaneously in the same project, letting you pick the right tool per component.
## When to Use This Skill
- Use when building a blog, documentation site, marketing page, or portfolio
- Use when performance and Core Web Vitals are the top priority
- Use when the project is content-heavy with Markdown or MDX files
- Use when you want SSG (static) output with optional SSR for dynamic routes
- Use when the user asks about `.astro` files, `Astro.props`, content collections, or `client:` directives
## How It Works
### Step 1: Project Setup
```bash
npm create astro@latest my-site
cd my-site
npm install
npm run dev
```
Add integrations as needed:
```bash
npx astro add tailwind # Tailwind CSS
npx astro add react # React component support
npx astro add mdx # MDX support
npx astro add sitemap # Auto sitemap.xml
npx astro add vercel # Vercel SSR adapter
```
Project structure:
```
src/
pages/ ← File-based routing (.astro, .md, .mdx)
layouts/ ← Reusable page shells
components/ ← UI components (.astro, .tsx, .vue, etc.)
content/ ← Type-safe content collections (Markdown/MDX)
styles/ ← Global CSS
public/ ← Static assets (copied as-is)
astro.config.mjs ← Framework config
```
### Step 2: Astro Component Syntax
`.astro` files have a code fence at the top (server-only) and a template below:
```astro
---
// src/components/Card.astro
// This block runs on the server ONLY — never in the browser
interface Props {
title: string;
href: string;
description: string;
}
const { title, href, description } = Astro.props;
---
/* Scoped to this component automatically */
.card { border: 1px solid #eee; padding: 1rem; }
```
### Step 3: File-Based Pages and Routing
```
src/pages/index.astro → /
src/pages/about.astro → /about
src/pages/blog/[slug].astro → /blog/:slug (dynamic)
src/pages/blog/[...path].astro → /blog/* (catch-all)
```
Dynamic route with `getStaticPaths`:
```astro
---
// src/pages/blog/[slug].astro
export async function getStaticPaths() {
const posts = await getCollection('blog');
return posts.map(post => ({
params: { slug: post.slug },
props: { post },
}));
}
const { post } = Astro.props;
const { Content } = await post.render();
---
{title}
```
```astro
---
// src/pages/about.astro
import BaseLayout from '../layouts/BaseLayout.astro';
---
{title}
{description}
{post.data.title}
``` ### Step 4: Content Collections Content collections give you type-safe access to Markdown and MDX files: ```typescript // src/content/config.ts import { z, defineCollection } from 'astro:content'; const blog = defineCollection({ type: 'content', schema: z.object({ title: z.string(), date: z.coerce.date(), tags: z.array(z.string()).default([]), draft: z.boolean().default(false), }), }); export const collections = { blog }; ``` ```astro --- // src/pages/blog/index.astro import { getCollection } from 'astro:content'; const posts = (await getCollection('blog')) .filter(p => !p.data.draft) .sort((a, b) => b.data.date.valueOf() - a.data.date.valueOf()); ----
{posts.map(post => (
- {post.data.title} ))}
About Us
Welcome to our company...
``` ### Step 7: SSR Mode (On-Demand Rendering) Enable SSR for dynamic pages by setting an adapter: ```javascript // astro.config.mjs import { defineConfig } from 'astro/config'; import vercel from '@astrojs/vercel/serverless'; export default defineConfig({ output: 'hybrid', // 'static' | 'server' | 'hybrid' adapter: vercel(), }); ``` Opt individual pages into SSR with `export const prerender = false`. ## Examples ### Example 1: Blog with RSS Feed ```typescript // src/pages/rss.xml.ts import rss from '@astrojs/rss'; import { getCollection } from 'astro:content'; export async function GET(context) { const posts = await getCollection('blog'); return rss({ title: 'My Blog', description: 'Latest posts', site: context.site, items: posts.map(post => ({ title: post.data.title, pubDate: post.data.date, link: `/blog/${post.slug}/`, })), }); } ``` ### Example 2: API Endpoint (SSR) ```typescript // src/pages/api/subscribe.ts import type { APIRoute } from 'astro'; export const POST: APIRoute = async ({ request }) => { const { email } = await request.json(); if (!email) { return new Response(JSON.stringify({ error: 'Email required' }), { status: 400, headers: { 'Content-Type': 'application/json' }, }); } await addToNewsletter(email); return new Response(JSON.stringify({ success: true }), { status: 200 }); }; ``` ### Example 3: React Component as Island ```tsx // src/components/SearchBox.tsx import { useState } from 'react'; export default function SearchBox() { const [query, setQuery] = useState(''); const [results, setResults] = useState([]); async function search(e: React.FormEvent) { e.preventDefault(); const data = await fetch(`/api/search?q=${query}`).then(r => r.json()); setResults(data); } return ( setQuery(e.target.value)} />- {results.map(r =>
- {r.title} )}
Source: claude-code-templates (MIT). See About Us for full credits.