BadtzUI
    Beta
    Docs
    Hyperspace

    Hyperspace

    A dynamic and customizable background component that simulates a star-filled hyperspace effect.

    Loading...

    Installation

    Install dependencies

    npm install

    npm install clsx tailwind-merge

    Add utils file

    lib/utils.ts

    import { ClassValue, clsx } from "clsx";
    import { twMerge } from "tailwind-merge";
     
    export function cn(...inputs: ClassValue[]) {
      return twMerge(clsx(inputs));
    }

    Copy the source code

    hyperspace.tsx

    "use client";
     
    import * as React from "react";
    import { cn } from "@/lib/utils";
     
    interface HyperspaceProps extends React.HTMLAttributes<HTMLDivElement> {
      starTrailOpacity?: number;
      starSpeed?: number;
      starColor?: string;
      starSize?: number;
    }
     
    export default function Hyperspace({
      starTrailOpacity = 0.5,
      starSpeed = 1.01,
      starColor = "#FFFFFF",
      starSize = 0.5,
      className,
      ...props
    }: HyperspaceProps): JSX.Element {
      const canvasRef = React.useRef<HTMLCanvasElement | null>(null);
     
      function hexToRgb(hex: string): [number, number, number] {
        const cleanedHex = hex.replace("#", "");
        const bigint = parseInt(cleanedHex, 16);
        const r = (bigint >> 16) & 255;
        const g = (bigint >> 8) & 255;
        const b = bigint & 255;
        return [r, g, b];
      }
     
      const [r, g, b] = hexToRgb(starColor);
     
      React.useEffect(() => {
        if (typeof window === "undefined") return;
     
        const canvas = canvasRef.current;
        if (!canvas) return;
        const context = canvas.getContext("2d");
        if (!context) return;
     
        const resizeCanvas = () => {
          const container = canvas.parentElement;
          if (container) {
            canvas.width = container.offsetWidth;
            canvas.height = container.offsetHeight;
          }
        };
     
        resizeCanvas();
     
        const randomInRange = (max: number, min: number): number =>
          Math.floor(Math.random() * (max - min + 1)) + min;
     
        const sizeIncrement = 1.01;
        const radians = Math.PI / 180;
     
        class Star {
          state: {
            alpha: number;
            angle: number;
            x: number;
            vX: number;
            y: number;
            vY: number;
            size: number;
            active: boolean;
          } = {
            alpha: 0,
            angle: 0,
            x: 0,
            vX: 0,
            y: 0,
            vY: 0,
            size: starSize,
            active: true,
          };
     
          constructor() {
            this.reset();
          }
     
          reset() {
            const angle = randomInRange(0, 360) * radians;
            const vX = Math.cos(angle);
            const vY = Math.sin(angle);
     
            if (!canvas) return;
     
            const travelled =
              Math.random() > 0.5
                ? Math.random() * Math.max(canvas.width, canvas.height) +
                  Math.random() * (canvas.width * 0.24)
                : Math.random() * (canvas.width * 0.25);
     
            this.state = {
              alpha: Math.random(),
              angle: randomInRange(0, 360) * radians,
              x: Math.floor(vX * travelled) + canvas.width / 2,
              vX,
              y: Math.floor(vY * travelled) + canvas.height / 2,
              vY,
              size: starSize,
              active: true,
            };
          }
        }
     
        const stars = new Array(300).fill(null).map(() => new Star());
     
        let animationFrameId: number;
        const render = () => {
          const invertedOpacity = 1 - starTrailOpacity;
          context.fillStyle = `rgba(0, 0, 0, ${invertedOpacity})`;
          context.fillRect(0, 0, canvas.width, canvas.height);
     
          for (const star of stars) {
            const { x, y, size, vX, vY } = star.state;
     
            const newX = x + vX;
            const newY = y + vY;
     
            if (newX < 0 || newX > canvas.width || newY < 0 || newY > canvas.height) {
              star.reset();
            } else {
              star.state = {
                ...star.state,
                x: newX,
                vX: star.state.vX * starSpeed,
                y: newY,
                vY: star.state.vY * starSpeed,
                size: size * sizeIncrement,
              };
     
              context.strokeStyle = `rgba(${r}, ${g}, ${b}, ${star.state.alpha})`;
              context.lineWidth = size;
              context.beginPath();
              context.moveTo(x, y);
              context.lineTo(star.state.x, star.state.y);
              context.stroke();
            }
          }
     
          animationFrameId = requestAnimationFrame(render);
        };
     
        render();
     
        window.addEventListener("resize", resizeCanvas);
     
        return () => {
          cancelAnimationFrame(animationFrameId);
          window.removeEventListener("resize", resizeCanvas);
        };
      }, [starTrailOpacity, starSpeed, starColor, starSize]);
     
      return (
        <div className={cn("absolute inset-0 w-full h-full", className)} {...props}>
          <canvas ref={canvasRef} className="absolute inset-0 w-full h-full" />
        </div>
      );
    }

    Props

    PropTypeDescriptionDefault
    starTrailOpacityNumberControls the opacity of the star trails, where lower values make trails more visible.0.5
    starSpeedNumberAdjusts the speed of the stars, affecting how quickly they move across the screen.1.01
    starColorStringSets the color of the stars, specified as a hex code."#FFFFFF"
    starSizeNumberDefines the initial size of the stars.0.5
    classNameStringAdditional CSS classes to style the component.-

    Credits

  1. This component is heavily inspired by @ybensira