JSPM

  • ESM via JSPM
  • ES Module Entrypoint
  • Export Map
  • Keywords
  • License
  • Repository URL
  • TypeScript Types
  • README
  • Created
  • Published
  • Downloads 31
  • Score
    100M100P100Q85533F
  • License MIT

Framework-friendly embed cards for YouTube, X, Instagram, Reddit, Google Maps, Vimeo, TikTok, and custom providers.

Package Exports

  • embed-card
  • embed-card/manual
  • embed-card/next-themes
  • embed-card/web-component

Readme

embed-card

embed-card is a small frontend package for turning a URL into a polished embed card with one install. The monorepo docs and live demos live in apps/web.

Install

pnpm add embed-card

React usage

import { EmbedCard } from "embed-card"

export function ArticleEmbed() {
  return (
    <EmbedCard url="https://www.youtube.com/watch?v=dQw4w9WgXcQ" />
  )
}

EmbedCard renders a single styled surface — the embed content itself, with no surrounding title, description, or footer chrome. For iframe providers (YouTube, Vimeo, X, Google Maps) the root adopts the provider's native aspect ratio. Reddit threads render a client-side preview fetched from Reddit's public JSON API.

Theme the card

theme fields are optional, including shadow. The default is no box-shadow—only set theme.shadow when you want one. The default is exported as EMBED_CARD_DEFAULT_SHADOW if you need to compare or reset in app code.

import { EmbedCard } from "embed-card"

export function BrandedEmbed() {
  return (
    <EmbedCard
      url="https://vimeo.com/76979871"
      theme={{
        accentColor: "#e11d48",
        radius: 28,
      }}
    />
  )
}

Dark mode / appearance

Pass appearance to control whether the embed surface uses a light or dark palette.

Value Behaviour
"light" (default) Always use the light palette.
"dark" Always use the dark palette.
"system" Follow prefers-color-scheme at runtime; falls back to light on SSR.
<EmbedCard
  url="https://www.youtube.com/watch?v=dQw4w9WgXcQ"
  theme={{ appearance: "dark", accentColor: "#7c3aed" }}
/>

For the web component, set the appearance attribute:

<embed-card
  url="https://vimeo.com/76979871"
  appearance="dark"
></embed-card>

appearance="system" subscribes to prefers-color-scheme and re-renders automatically when the OS theme changes.

If the site uses next-themes (for example with Fumadocs), you can import the hydration-safe wrapper instead of wiring useTheme() yourself:

pnpm add next-themes
import { ThemedEmbedCard } from "embed-card/next-themes"

export function ArticleEmbed() {
  return <ThemedEmbedCard url="https://www.youtube.com/watch?v=dQw4w9WgXcQ" />
}

ctaLabel

For fallback link previews (URLs that don't match a built-in provider), ctaLabel overrides the default "Open original" call-to-action text shown inside the card surface. It has no effect on iframe or Reddit embeds.

<EmbedCard url="https://example.com" ctaLabel="Visit site" />

In the web component, use cta-label:

<embed-card url="https://example.com" cta-label="Visit site"></embed-card>

Web component usage

import { registerEmbedCard } from "embed-card/web-component"

registerEmbedCard()
<embed-card
  url="https://www.google.com/maps?q=Tokyo+Station&output=embed"
  accent-color="#0f766e"
></embed-card>

Shadow DOM parts

The custom element exposes a single CSS part for external styling:

Part Element
root The embed surface root (<figure>)

Breaking change from ≤0.1.x: The header, footer, and inner preview shadow parts no longer exist. Style ::part(root) instead, or use CSS variables for theming.

Low-level manual usage

import { resolveEmbed } from "embed-card/manual"

const embed = resolveEmbed("https://x.com/1chooo___/status/2028573993972969585")

Built-in providers

  • YouTube
  • X / Twitter
  • Reddit (browser JSON thread preview, not an iframe embed)
  • Google Maps
  • Vimeo
  • Fallback link preview

Custom providers

import {
  EmbedCard,
  createEmbedProvider,
  defaultProviders,
} from "embed-card"

const customProvider = createEmbedProvider({
  id: "docs",
  label: "Docs",
  accentColor: "#7c3aed",
  match: (url) => url.hostname === "docs.example.com",
  resolve: (url) => ({
    provider: "docs",
    providerLabel: "Docs",
    accentColor: "#7c3aed",
    title: "Documentation page",
    description: "Custom providers plug into the same normalized contract.",
    site: url.hostname,
    url: url.toString(),
    displayUrl: url.toString(),
    renderer: {
      type: "link",
      href: url.toString(),
      ctaLabel: "Open docs",
    },
  }),
})

<EmbedCard
  providers={[customProvider, ...defaultProviders]}
  url="https://docs.example.com/platform/embeds"
/>