AugmentClaude

21st Registry

Publish React components to your team library or install shared ones.

Installation

  1. Make sure Claude is on your device and in your terminal.

    Skills load from ~/.claude/skills/ when Claude Code starts up — so you need it on your machine first. If you don't have it yet, install it once with the command below, then run claude in any terminal to verify.

    One-time setup
    npm i -g @anthropic-ai/claude-code

    Already have it? Skip ahead.

  2. Paste into Claude Code or into your terminal.

    This copies the whole skill folder into ~/.claude/skills/21st-registry-21st-dev/ — the SKILL.md plus any scripts, reference docs, or templates the skill ships with. Safe default: works for every skill.

    Faster alternative (instruction-only skills)

    Skips the clone and grabs only the SKILL.md file. Don't use this if the skill ships Python scripts, reference markdowns, or asset templates — they won't be downloaded and the skill will fail when it tries to load them.

    Quick install (SKILL.md only)
    Sign up to copy
  3. Restart Claude Code.

    Quit and reopen Claude Code (or any other agent that loads from ~/.claude/skills/). New skills are picked up on startup.

  4. Just ask Claude.

    Skills auto-activate when your request matches the skill's description — no slash command needed. Trigger phrases live in the skill's own frontmatter; you can read them in the “What this skill does” section above.

Prefer to read the source first? Open on GitHub.

When Claude uses it

Publish a React component to the user's team library on 21st.dev so teammates can install it with a single command, or install an existing one. Triggers when the user says things like "publish/share/upload this to 21st", "залей в наш регистр", "опубликуй компонент", "share with team", "make this reusable", "install our Button", "use the team button".

What this skill does

Publish & install components in the 21st team library

Use this skill in two directions: publishing a component for the team, and installing one the team already shared.

Pre-flight (always)

  1. Check auth: API_KEY_21ST env is set, or the registry CLI has saved credentials. If neither — tell the user to run npx @21st-dev/registry login once. Don't try to log them in yourself.
  2. The CLI is @21st-dev/registry. Don't reinvent — use it.

Publishing a component

Decide visibility — default to unlisted in a registry

User says…Visibility
"share with team", "залей нам", "publish internally", default for any unqualified askunlisted in the selected/default registry (no visibility flag needed)
"share publicly via link / on my profile, but don't list it in the library"--unlisted
"publish publicly", "make it public on 21st"--public
"restrict to the registry team", "private team draft"--private

Never use --public without explicit user instruction. Public components go through admin moderation and appear in the 21st library. --unlisted is the safe option when the user wants a shareable URL but doesn't want a library listing.

Standard publish

The CLI's positional file path triggers auto-detection — name from the default export, slug from the filename, tags from imports, demo auto-found or synthesised. So in 95% of cases this is enough:

npx @21st-dev/registry ./path/to/Component.tsx \
  --to default \
  --description "1-2 sentences about what it does and when to use it"

Flag reference

FlagWhen to use
--name "Display Name"Override the auto-detected name. Default is humanised version of the default-export name.
--description "…"Required. 10+ chars. Write a real description — what it does and when to use it. Never fabricate; if you don't know, ask the user.
--tags "form,input,validation"1-5 lowercase tags. Default: detected from imports (lucide-react → "icon", framer-motion → "animation", etc). Only override if the auto-detected ones miss the point.
--slug my-buttonOverride the URL slug. Default: kebab-case from name.
--demo ./Component.demo.tsxDemo file. Auto-detected by these patterns: {Component}.demo.tsx, demos/{slug}.tsx, demos/default.tsx. If none exist, the CLI synthesises a trivial <Component /> demo automatically — fine for v1, but a real demo gives a much better preview.
--preview ./preview.pngOptional. The team library uses a live iframe preview; a static image is only needed if you want a snappy thumbnail.
--to <registry-slug>Target a specific registry in the authenticated team (e.g. --to default). If omitted, the server uses the team's first/default registry.
--public / --unlisted / --privateOverride component visibility.

What the user gets back

✅ https://21st.dev/{username}/{slug}
Install in another project:
  npx @21st-dev/registry add @{username}/{slug}

Updating an existing component

Same command, same slug → upsert. The CLI prints "Updated" instead of "Published". No version flag needed; teammates always get the latest.

If the user says "I want a NEW component, not an update" but slug collides — confirm with the user before overwriting; suggest changing --slug.


Installing a component

Two address formats are accepted:

  • @team-slug/component-slug — install from a team registry (preferred for team-shared)
  • @username/component-slug — install from a user's personal/public components
npx @21st-dev/registry add @acme/animated-button
# or
npx @21st-dev/registry add @serjobas/animated-button

The CLI:

  1. Fetches the registry JSON and component file (server resolves @handle against team-slug first, then username)
  2. Writes it to the project (components/ui/{slug}.tsx by default)
  3. Runs pnpm/npm/yarn/bun add for any npm dependencies
  4. If it depends on other 21st components, prints them — install with add separately

For shadcn directly, public/unlisted components can use the plain registry URL:

npx shadcn@latest add "https://21st.dev/r/acme/animated-button"

For private components, pass the registry API key in the URL:

npx shadcn@latest add "https://21st.dev/r/acme/animated-button?api_key=$API_KEY_21ST"

Do not commit or share shadcn URLs that include api_key.

Flags:

  • --force — overwrite existing file
  • --no-install — skip npm install step (just write files)
  • --dir PATH — install into a different project directory

Searching the team library

When the user wants a component but doesn't know the exact name:

npx @21st-dev/registry search "<query>"

Default scope is team (your team's library). Use --scope mine for just your own, --scope public for the public library.

Always search before publishing if there's a chance a similar component already exists. Don't add duplicates to the team library.

npx @21st-dev/registry search "button" --scope team
# Prints list with @user/slug refs you can pass to `add`.

Hard rules for agents

  • Never use --public without an explicit "publish publicly" from the user.
  • Never fabricate a description. Ask the user, or read the code carefully.
  • Never include API keys, env values, or hardcoded internal URLs in the component file you publish.
  • Never publish a file with unsaved edits — flush first.
  • Always search before publishing if a similar component might already exist.
  • Always add a useful demo file with realistic props if you can; only fall back to the auto-synthesised one as a last resort.

Common mistakes to avoid

  1. Demo imports the component via a wrong path. The CLI auto-rewrites relative imports (e.g. import X from "../component") to @/components/ui/{slug} before upload — so write demos with relative imports to the user's source file, not aliases. The CLI will sort it out.
  2. Slug doesn't match between publishes. If the user renames the file, the auto-derived slug changes and you'll create a duplicate. Pass --slug explicitly when re-publishing under a stable name.
  3. Component lacks a default export. This will fail with a clear error — refactor the component to export default function ComponentName(...) first.

Related skills