Hero 7

Centered hero with a word-by-word blur-in serif headline, a glass beta badge with a live dot, dual rounded CTAs, and animated beam line-art SVGs that trace the top, bottom, and both sides of the section. Respects reduced-motion.

Preview

Installation

npx shadcn@latest add https://hirael.com/r/hero-07.json

Code

components/blocks/hero-07.tsx
"use client";

import * as React from "react";
import { ArrowRight } from "lucide-react";
import { motion, useReducedMotion } from "motion/react";

import { Button } from "@/registry/hirael/ui/button";
import { cn } from "@/lib/utils";

const HEADLINE = "Build the page once, ship it everywhere";

function Headline() {
  const reduce = useReducedMotion();
  const words = HEADLINE.split(" ");

  return (
    <h1
      data-slot="hero-headline"
      className="max-w-3xl font-serif text-[42px] font-medium leading-[1.04] tracking-tight text-foreground sm:text-5xl md:text-7xl"
    >
      {words.map((word, i) => {
        const accent = i >= words.length - 2;
        return (
          <motion.span
            key={`${word}-${i}`}
            className={cn("inline-block", accent && "italic text-foreground")}
            initial={
              reduce ? false : { opacity: 0, y: 16, filter: "blur(8px)" }
            }
            whileInView={{ opacity: 1, y: 0, filter: "blur(0px)" }}
            viewport={{ once: true }}
            transition={{
              duration: 0.5,
              ease: "easeOut",
              delay: 0.3 + i * 0.12,
            }}
          >
            {word}
            {i < words.length - 1 ? " " : null}
          </motion.span>
        );
      })}
    </h1>
  );
}

function Beam({
  className,
  reduce,
}: {
  className?: string;
  reduce: boolean | null;
}) {
  const verticals = [
    "M141.338 232.625V5.075",
    "M200.338 232.625V5.075",
    "M259.338 231.625V4.075",
    "M318.338 230.625V3.075",
  ];
  const curves = [
    "M459.649 152.723 351.613 69.264a11 11 0 0 1-4.275-8.705V.074",
    "m.338 152.723 108.036-83.459a11 11 0 0 0 4.275-8.705V.074",
  ];

  return (
    <svg
      width={460}
      height={233}
      fill="none"
      viewBox="0 0 460 233"
      className={cn("text-foreground", className)}
      aria-hidden="true"
    >
      {[...verticals, ...curves].map((d, i) => (
        <path
          key={`base-${i}`}
          d={d}
          stroke="currentColor"
          strokeWidth="1"
          className="opacity-15"
        />
      ))}
      {curves.map((d, i) => (
        <motion.path
          key={`beam-${i}`}
          d={d}
          stroke="url(#hero07-beam)"
          strokeWidth="2.5"
          strokeLinecap="round"
          strokeDasharray="1000"
          initial={{ strokeDashoffset: 0 }}
          animate={reduce ? undefined : { strokeDashoffset: [0, -900, 0] }}
          transition={
            reduce
              ? undefined
              : {
                  duration: 5 + i * 1.2,
                  repeat: Infinity,
                  ease: "linear",
                }
          }
          style={{ willChange: "stroke-dashoffset" }}
        />
      ))}
      <defs>
        <linearGradient
          id="hero07-beam"
          x1="230"
          y1="233"
          x2="230"
          y2="0"
          gradientUnits="userSpaceOnUse"
        >
          <stop stopColor="transparent" offset="0%" />
          <stop
            stopColor="color-mix(in oklch, var(--primary) 60%, transparent)"
            offset="30%"
          />
          <stop stopColor="var(--foreground)" offset="50%" />
          <stop
            stopColor="color-mix(in oklch, var(--primary) 60%, transparent)"
            offset="70%"
          />
          <stop stopColor="transparent" offset="100%" />
        </linearGradient>
      </defs>
    </svg>
  );
}

function SideBeam({
  className,
  reduce,
}: {
  className?: string;
  reduce: boolean | null;
}) {
  const paths = [
    "M0 442.957L209.048 442.957C212.366 442.957 215.508 441.458 217.596 438.879L321.802 310.196",
    "M0 1.19531L209.048 1.19557C212.366 1.19558 215.508 2.69391 217.596 5.27302L321.802 133.956",
  ];

  return (
    <svg
      width={200}
      height={444}
      viewBox="0 0 323 444"
      fill="none"
      className={cn("text-foreground", className)}
      aria-hidden="true"
    >
      {paths.map((d, i) => (
        <path
          key={`base-${i}`}
          d={d}
          stroke="currentColor"
          strokeWidth="1"
          className="opacity-15"
        />
      ))}
      {paths.map((d, i) => (
        <motion.path
          key={`beam-${i}`}
          d={d}
          stroke="url(#hero07-side)"
          strokeWidth="3"
          strokeLinecap="round"
          strokeDasharray="1200"
          initial={{ strokeDashoffset: 0 }}
          animate={reduce ? undefined : { strokeDashoffset: [0, -900, 0] }}
          transition={
            reduce
              ? undefined
              : {
                  duration: 6 + i * 1.5,
                  repeat: Infinity,
                  ease: "linear",
                }
          }
          style={{ willChange: "stroke-dashoffset" }}
        />
      ))}
      <defs>
        <linearGradient
          id="hero07-side"
          x1="160.901"
          y1="444"
          x2="160.901"
          y2="0"
          gradientUnits="userSpaceOnUse"
        >
          <stop stopColor="transparent" offset="0%" />
          <stop
            stopColor="color-mix(in oklch, var(--primary) 60%, transparent)"
            offset="30%"
          />
          <stop stopColor="var(--foreground)" offset="50%" />
          <stop
            stopColor="color-mix(in oklch, var(--primary) 60%, transparent)"
            offset="70%"
          />
          <stop stopColor="transparent" offset="100%" />
        </linearGradient>
      </defs>
    </svg>
  );
}

export default function Hero07() {
  const reduce = useReducedMotion();

  return (
    <section
      data-slot="hero"
      className="relative isolate grid min-h-[640px] place-items-center overflow-hidden bg-background px-6 py-28 text-center text-foreground md:px-10 md:py-32"
    >
      <div
        aria-hidden
        data-slot="hero-backdrop"
        className="pointer-events-none absolute inset-0 [mask-image:radial-gradient(ellipse_at_center,black_10%,transparent_72%)]"
        style={{
          background:
            "radial-gradient(120% 120% at 50% 30%, color-mix(in oklch, var(--primary) 8%, transparent), transparent 60%)",
        }}
      />

      <Beam
        reduce={reduce}
        className="pointer-events-none absolute -top-8 start-1/2 -translate-x-1/2 scale-[0.6] rtl:translate-x-1/2 sm:top-0 sm:scale-90"
      />
      <Beam
        reduce={reduce}
        className="pointer-events-none absolute -bottom-8 start-1/2 -translate-x-1/2 -scale-[0.6] rtl:translate-x-1/2 sm:bottom-0 sm:-scale-90"
      />
      <SideBeam
        reduce={reduce}
        className="pointer-events-none absolute top-1/2 start-0 hidden -translate-y-1/2 scale-90 rtl:-scale-x-90 md:block"
      />
      <SideBeam
        reduce={reduce}
        className="pointer-events-none absolute top-1/2 end-0 hidden -translate-y-1/2 -scale-x-90 rtl:scale-90 md:block"
      />

      <motion.div
        className="relative z-10 mx-auto flex w-full max-w-3xl flex-col items-center gap-5"
        initial={reduce ? false : { opacity: 0 }}
        whileInView={{ opacity: 1 }}
        viewport={{ once: true }}
        transition={{ duration: 0.8, ease: "easeOut" }}
      >
        <span
          data-slot="hero-badge"
          className="inline-flex w-fit items-center gap-2 rounded-full border border-border bg-card/70 px-4 py-1.5 font-mono text-[10px] uppercase tracking-[0.14em] text-muted-foreground backdrop-blur-sm"
        >
          <span className="relative flex size-1.5">
            <span
              className="absolute inline-flex size-full animate-ping rounded-full opacity-75"
              style={{ background: "var(--accent-cool)" }}
            />
            <span
              className="relative inline-flex size-1.5 rounded-full"
              style={{ background: "var(--accent-cool)" }}
            />
          </span>
          Now in beta
        </span>

        <Headline />

        <p
          data-slot="hero-subhead"
          className="mt-2 max-w-xl text-muted-foreground sm:mt-4"
        >
          One source of truth for your interface. Design it, theme it, and reuse
          it across every project without rewriting the same components.
        </p>

        <div className="mt-4 flex flex-col gap-3 sm:flex-row">
          <Button asChild size="lg" className="group rounded-full px-7">
            <a href="#">
              <span>Get started</span>
              <ArrowRight className="size-4 -rotate-45 transition-transform duration-150 ease-out group-hover:translate-x-0.5 rtl:rotate-[135deg] rtl:group-hover:-translate-x-0.5" />
            </a>
          </Button>
          <Button
            asChild
            size="lg"
            variant="outline"
            className="rounded-full px-7"
          >
            <a href="#">
              <span>Read the docs</span>
            </a>
          </Button>
        </div>
      </motion.div>
    </section>
  );
}

Dependencies

shadcn registry

button

npm

motionlucide-react