Slide Card

Bertram Gilfoyle

System Architecture

Bertram Gilfoyle

Installation

1

Install the following packages if you do not have it.

npm install framer-motion lucide-react
2

Copy and paste the following code into your project.

"use client";
import * as React from "react";
import { AnimatePresence, motion } from "motion/react";
import { cn } from "@/lib/utils";

interface SlideCardProps {
  className?: string;
  imageSrc: string;
  name: string;
  role: string;
  bio: string;
  socialLinks?: {
    github?: string;
    linkedin?: string;
    twitter?: string;
  };
}

const ProfileContent = ({
    bio,
    socialLinks,
  }: {
    bio: string;
    socialLinks?: SlideCardProps["socialLinks"];
  }) => (
    <motion.div
      key="content"
      initial={{ opacity: 0, y: 50 }}
      animate={{ opacity: 1, y: 0 }}
      exit={{ opacity: 0, y: -50 }}
      className="px-6 text-center space-y-6"
    >
      <motion.p
        className="text-cyan-200/80 leading-relaxed"
        initial={{ opacity: 0, y: 20 }}
        animate={{ opacity: 1, y: 0 }}
        transition={{ delay: 0.2 }}
      >
        {bio}
      </motion.p>
  
      {socialLinks && (
        <motion.div
          className="space-y-4"
          initial={{ opacity: 0 }}
          animate={{ opacity: 1 }}
          transition={{ delay: 0.4 }}
        >
          <h4 className="text-sm font-semibold text-emerald-400/80">Connect</h4>
          <div className="flex justify-center gap-4">
            {socialLinks.github && (
              <SocialIcon
                href={socialLinks.github}
                icon={
                  <svg width="20" height="20" fill="currentColor" viewBox="0 0 24 24">
                    <path d="M12 0c-6.626 0-12 5.373-12 12 0 5.302 3.438 9.8 8.207 11.387.599.111.793-.261.793-.577v-2.234c-3.338.726-4.033-1.416-4.033-1.416-.546-1.387-1.333-1.756-1.333-1.756-1.089-.745.083-.729.083-.729 1.205.084 1.839 1.237 1.839 1.237 1.07 1.834 2.807 1.304 3.492.997.107-.775.418-1.305.762-1.604-2.665-.305-5.467-1.334-5.467-5.931 0-1.311.469-2.381 1.236-3.221-.124-.303-.535-1.524.117-3.176 0 0 1.008-.322 3.301 1.23.957-.266 1.983-.399 3.003-.404 1.02.005 2.047.138 3.006.404 2.291-1.552 3.297-1.23 3.297-1.23.653 1.653.242 2.874.118 3.176.77.84 1.235 1.911 1.235 3.221 0 4.609-2.807 5.624-5.479 5.921.43.372.823 1.102.823 2.222v3.293c0 .319.192.694.801.576 4.765-1.589 8.199-6.086 8.199-11.386 0-6.627-5.373-12-12-12z" />
                  </svg>
                }
              />
            )}
            {socialLinks.linkedin && (
              <SocialIcon
                href={socialLinks.linkedin}
                icon={
                  <svg width="20" height="20" fill="currentColor" viewBox="0 0 24 24">
                    <path d="M19 0h-14c-2.761 0-5 2.239-5 5v14c0 2.761 2.239 5 5 5h14c2.762 0 5-2.239 5-5v-14c0-2.761-2.238-5-5-5zm-11 19h-3v-11h3v11zm-1.5-12.268c-.966 0-1.75-.79-1.75-1.764s.784-1.764 1.75-1.764 1.75.79 1.75 1.764-.783 1.764-1.75 1.764zm13.5 12.268h-3v-5.604c0-3.368-4-3.113-4 0v5.604h-3v-11h3v1.765c1.396-2.586 7-2.777 7 2.476v6.759z" />
                  </svg>
                }
              />
            )}
            {socialLinks.twitter && (
              <SocialIcon
                href={socialLinks.twitter}
                icon={
                  <svg width="20" height="20" fill="currentColor" viewBox="0 0 24 24">
                    <path d="M24 4.557c-.883.392-1.832.656-2.828.775 1.017-.609 1.798-1.574 2.165-2.724-.951.564-2.005.974-3.127 1.195-.897-.957-2.178-1.555-3.594-1.555-3.179 0-5.515 2.966-4.797 6.045-4.091-.205-7.719-2.165-10.148-5.144-1.29 2.213-.669 5.108 1.523 6.574-.806-.026-1.566-.247-2.229-.616-.054 2.281 1.581 4.415 3.949 4.89-.693.188-1.452.232-2.224.084.626 1.956 2.444 3.379 4.6 3.419-2.07 1.623-4.678 2.348-7.29 2.04 2.179 1.397 4.768 2.212 7.548 2.212 9.142 0 14.307-7.721 13.995-14.646.962-.695 1.797-1.562 2.457-2.549z" />
                  </svg>
                }
              />
            )}
          </div>
        </motion.div>
      )}
    </motion.div>
  );
  
 
const MovingGradient = () => (
  <div className="absolute inset-0 h-full w-full overflow-hidden">
    <div className="absolute inset-0 h-full w-full bg-gradient-to-r from-transparent via-emerald-500/10 to-transparent animate-[beam_3s_infinite]" />
    <div className="absolute inset-0 h-full w-full bg-gradient-to-b from-transparent via-cyan-500/5 to-transparent animate-[pulse_4s_infinite]" />
  </div>
);

const BorderGradient = () => (
  <>
    <div className="absolute top-0 left-0 w-[20%] h-[2px] bg-gradient-to-r from-transparent via-emerald-500 to-cyan-500 animate-[borderSlideRight_2s_infinite]" />
    <div className="absolute top-0 right-0 h-[20%] w-[2px] bg-gradient-to-b from-transparent via-cyan-500 to-emerald-500 animate-[borderSlideDown_2s_infinite]" />
    <div className="absolute bottom-0 right-0 w-[20%] h-[2px] bg-gradient-to-l from-transparent via-emerald-500 to-cyan-500 animate-[borderSlideLeft_2s_infinite]" />
    <div className="absolute bottom-0 left-0 h-[20%] w-[2px] bg-gradient-to-t from-transparent via-cyan-500 to-emerald-500 animate-[borderSlideUp_2s_infinite]" />
  </>
);

const SocialIcon = ({ href, icon }: { href: string; icon: React.ReactNode }) => (
  <motion.a
    href={href}
    target="_blank"
    rel="noopener noreferrer"
    whileHover={{ scale: 1.2, rotate: 360 }}
    transition={{ type: "spring", stiffness: 300 }}
    className="w-10 h-10 flex items-center justify-center rounded-full bg-gradient-to-br from-[#111827] to-[#0B1120] text-emerald-400 shadow-lg shadow-cyan-500/20 border border-cyan-950 hover:border-cyan-500/50"
  >
    {icon}
  </motion.a>
);

const SlideCard = React.forwardRef<HTMLDivElement, SlideCardProps>(
  ({ className, imageSrc, name, role, bio, socialLinks }, ref) => {
    const [isHovered, setHovered] = React.useState(false);

    return (
      <motion.div
        ref={ref}
        className={cn(
          "h-[500px] w-[280px] md:w-[300px]",
          "group relative flex flex-col items-center justify-between overflow-hidden rounded-2xl p-3",
          "bg-[#0B1120]",
          "before:absolute before:inset-0 before:rounded-2xl before:p-[2px]",
          "before:bg-[linear-gradient(45deg,transparent_25%,rgba(16,185,129,0.2)_50%,transparent_75%,transparent_100%)]",
          "before:bg-[length:250%_250%] before:animate-[borderGradient_8s_linear_infinite]",
          "after:absolute after:inset-[1px] after:rounded-2xl after:bg-[#0B1120]/90",
          "hover:before:bg-[linear-gradient(45deg,transparent_25%,rgba(6,182,212,0.3)_50%,transparent_75%,transparent_100%)]",
          className
        )}
        initial={{ y: 20, opacity: 0 }}
        animate={{ y: 0, opacity: 1 }}
        whileHover={{ scale: 1.02 }}
        onHoverStart={() => setHovered(true)}
        onHoverEnd={() => setHovered(false)}
      >
        <BorderGradient />
        <div className="relative z-10 w-full h-full flex flex-col">
          <Header
            name={name}
            role={role}
            imageSrc={imageSrc}
            isHovered={isHovered}
          />
          <Content
            isHovered={isHovered}
            imageSrc={imageSrc}
            name={name}
            bio={bio}
            socialLinks={socialLinks}
          />
        </div>
      </motion.div>
    );
  }
);

const Header = ({ name, role, imageSrc, isHovered }: { 
  name: string;
  role: string;
  imageSrc: string;
  isHovered: boolean;
}) => (
  <motion.div
    className="w-full bg-[#111827] rounded-xl p-4 relative overflow-hidden border border-cyan-950"
    animate={{
      boxShadow: isHovered
        ? "0 0 25px 5px rgba(6, 182, 212, 0.2)"
        : "0 0 15px 2px rgba(6, 182, 212, 0.1)",
    }}
  >
    <MovingGradient />
    <div className="relative z-10 flex justify-between items-center">
      <div>
        <motion.h3
          className="text-xl font-bold bg-gradient-to-r from-emerald-400 via-cyan-300 to-emerald-400 bg-clip-text text-transparent"
          animate={{
            backgroundSize: isHovered ? "200% 200%" : "100% 100%",
          }}
          transition={{ duration: 2, repeat: Infinity }}
        >
          {name}
        </motion.h3>
        <motion.p
          className="text-sm text-cyan-200/60"
          animate={{
            color: isHovered
              ? "rgb(103 232 249 / 0.8)"
              : "rgb(103 232 249 / 0.6)",
          }}
        >
          {role}
        </motion.p>
      </div>
      <AnimatePresence>
        {isHovered && (
          <motion.div
            initial={{ opacity: 0, scale: 0.5, rotate: -180 }}
            animate={{ opacity: 1, scale: 1, rotate: 0 }}
            exit={{ opacity: 0, scale: 0.5, rotate: 180 }}
            transition={{ type: "spring", damping: 15 }}
            className="relative"
          >
            <div className="absolute inset-0 bg-gradient-to-r from-emerald-500 via-cyan-500 to-emerald-500 rounded-full animate-[spin_3s_linear_infinite]" />
            <img
              src={imageSrc}
              alt={name}
              className="w-16 h-16 rounded-full relative z-10 border-2 border-[#111827]"
            />
          </motion.div>
        )}
      </AnimatePresence>
    </div>
  </motion.div>
);

const Content = ({
  isHovered,
  imageSrc,
  name,
  bio,
  socialLinks,
}: {
  isHovered: boolean;
  imageSrc: string;
  name: string;
  bio: string;
  socialLinks?: SlideCardProps["socialLinks"];
}) => (
  <div className="flex-1 flex items-center justify-center py-6">
    <AnimatePresence mode="wait">
      {!isHovered ? (
        <ProfileImage imageSrc={imageSrc} name={name} />
      ) : (
        <ProfileContent bio={bio} socialLinks={socialLinks} />
      )}
    </AnimatePresence>
  </div>
);

const ProfileImage = ({ imageSrc, name }: { imageSrc: string; name: string }) => (
  <motion.div
    key="image"
    initial={{ opacity: 0, scale: 0.8 }}
    animate={{ opacity: 1, scale: 1 }}
    exit={{ opacity: 0, scale: 0.8, rotate: -180 }}
    transition={{ type: "spring", damping: 20 }}
    className="relative group"
  >
    <motion.div
      className="absolute inset-0 bg-gradient-to-r from-emerald-500 via-cyan-500 to-emerald-500 rounded-full blur-xl opacity-20"
      animate={{
        scale: [1, 1.2, 1],
        opacity: [0.2, 0.3, 0.2],
        rotate: [0, 180, 360],
      }}
      transition={{ duration: 8, repeat: Infinity }}
    />
    <div className="relative">
      <div className="absolute inset-0 rounded-full bg-gradient-to-r from-emerald-500 via-cyan-500 to-emerald-500 animate-[spin_8s_linear_infinite]" />
      <img
        src={imageSrc}
        alt={name}
        className="w-48 h-48 rounded-full relative z-10 border-4 border-[#0B1120] p-[2px]"
      />
    </div>
  </motion.div>
);

SlideCard.displayName = "SlideCard";

export { SlideCard };
export default SlideCard;
3

Add required animations in globals.css file


/* Add the following keyframe animations to your globals.css file */
@keyframes beam {
  0%, 100% { transform: translateX(-100%); }
  50% { transform: translateX(100%); }
}

@keyframes pulse {
  0%, 100% { transform: translateY(-100%); }
  50% { transform: translateY(100%); }
}

@keyframes spin {
  to { transform: rotate(360deg); }
}

@keyframes borderGradient {
  0% { background-position: 0% 0%; }
  100% { background-position: 100% 100%; }
}

@keyframes borderSlideRight {
  0% { transform: translateX(-100%); }
  100% { transform: translateX(500%); }
}

@keyframes borderSlideDown {
  0% { transform: translateY(-100%); }
  100% { transform: translateY(500%); }
}

@keyframes borderSlideLeft {
  0% { transform: translateX(100%); }
  100% { transform: translateX(-500%); }
}

@keyframes borderSlideUp {
  0% { transform: translateY(100%); }
  100% { transform: translateY(-500%); }
}
              
4

Implement the code as demonstrated in the preview