skills/og-images/SKILL.md
Guides creation of OpenGraph and Twitter share images using next/og ImageResponse. Covers layout patterns, custom fonts, avatars, title case, and Satori rules. Use when building OG images, Twitter cards, or social previews.
npx skillsauth add richtabor/agent-skills og-imagesInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Generate dynamic OpenGraph (1200x630) and Twitter (1200x600) images using next/og ImageResponse.
app/opengraph-image.tsx): Best for static pages with known titles at build time. Export runtime, alt, size, contentType, and a default Image function.app/api/og/route.tsx): Best for dynamic content (blog posts, CMS). Accept slug and/or title as query params. Reference in metadata via generateMetadata().Use export const runtime = "edge" for both approaches.
| File | Purpose | Dimensions |
|------|---------|------------|
| opengraph-image.tsx | Facebook, LinkedIn, iMessage | 1200x630 |
| twitter-image.tsx | Twitter/X cards | 1200x600 |
| app/api/og/route.tsx | Dynamic API route | 1200x630 |
Place file-based routes in the relevant route directory (e.g., app/about/opengraph-image.tsx for /about).
flexDirection: "column" with justifyContent: "space-between" to separate content from branding#888)textWrap: "balance" and constrain width with maxWidthletterSpacing: "-0.02em" for tight, editorial feel at large sizes48px on all sides works well at 1200x630If the project has an avatar or logo, place it in the bottom-right corner using a flex container with justifyContent: "flex-end". Load images via fetch + arrayBuffer, convert to base64 data URI for the src. Use borderRadius: "50%" for circular avatars. Cache loaded assets in a Map to avoid refetching.
Load .ttf files from public/fonts/ using new URL("../../../public/fonts/YourFont.ttf", import.meta.url). Pass the ArrayBuffer to ImageResponse via the fonts option. Cache the font buffer after first load. Match the weight in the fonts config to the actual font file weight.
If titles come from a CMS, apply smart title case:
Reference the OG route in generateMetadata():
export function generateMetadata({ params }) {
return {
openGraph: {
images: [`/api/og?slug=${params.slug}`],
},
};
}
For static pages, pass the title directly: /api/og?title=About.
These are hard requirements of the next/og rendering engine (Satori):
display: "flex" — this is the only layout modergb(), hsl(), or CSS variablesgap on older versions — test before relying on it; fallback to margin| Issue | Solution |
|-------|----------|
| Text not rendering | Add display: "flex" to the text wrapper |
| Layout broken | Ensure all containers have display: "flex" |
| Colors wrong | Use hex colors, not CSS variables |
| Font not loading | Check the relative path from route file to public/fonts/ |
| Image not showing | Convert to base64 data URI, don't use relative paths |
Preview during development by visiting the route directly in the browser:
http://localhost:3000/api/og?title=Hello+World
After building, verify routes register as dynamic (f prefix) in the build output.
testing
Transforms notes into X (Twitter) posts. Triggers when user asks to create social content, draft tweets, or turn notes/ideas into posts.
tools
# WordPress Add Links Find and add internal and external links to a blog post draft, naturally woven into existing sentences. ## Trigger - "find links for this post" - "find internal links" - "add links to this post" - "link this draft" ## Environment Variables - `WORDPRESS_URL` — Blog base URL (e.g. `https://yourblog.com`) - `WORDPRESS_USERNAME` — WordPress account username - `WORDPRESS_APP_PASSWORD` — Application password (not your regular password) ## Process ### Phase 1: Load the Post
development
Writes technical blog posts about features being built. Triggers when user asks to write about development progress, implementations, or project updates.
testing
Reviews and validates agent skills against best practices. Triggers on "review this skill", "check my skill", "validate skill", "is this skill well-written", or when creating/editing skills.