Perks Grid

Why Developers Choose Us

Streamlined solutions for modern web development

Feature Card

Forever free and community-driven, backed by MIT license freedom.

Lightweight

90% smaller footprint than alternatives, without compromising power.

Enterprise-Grade

Battle-tested TypeScript codebase with comprehensive testing.

Maximum Performance

Seamlessly blends JS flexibility with GPU-powered acceleration.

Installation

1

Install the following packages if you do not have it.

npm i framer-motion lucide-react
2

Copy and paste the following code into your project.

import React from "react";
import { Check } from "lucide-react";

// Types remain the same...
type Card = {
  title: string;
  content: string;
  icon: JSX.Element;
  colorClass: string;
};

type FeatureCardsProps = {
  info: Card[];
};

type ColorConfig = {
  text: string;
  bg: string;
  bgOpacity: string;
  border: string;
  gradientFrom: string;
  gradientTo: string;
  dotBg: string;
  hoverBorder: string;
};

type ColorClassMap = {
  [key: string]: ColorConfig;
};

const colorClassMap: ColorClassMap = {
  'pink-500': {
    text: 'text-pink-500',
    bg: 'bg-pink-500',
    bgOpacity: 'bg-pink-500/10',
    border: 'border-pink-500/20',
    hoverBorder: 'group-hover:border-pink-500',
    gradientFrom: 'from-pink-500',
    gradientTo: 'to-pink-500',
    dotBg: 'bg-[radial-gradient(#EC4899_1px,transparent_1px)]',
  },
  'red-500': {
    text: 'text-red-500',
    bg: 'bg-red-500',
    bgOpacity: 'bg-red-500/10',
    border: 'border-red-500/20',
    hoverBorder: 'group-hover:border-red-500',
    gradientFrom: 'from-red-500',
    gradientTo: 'to-red-500',
    dotBg: 'bg-[radial-gradient(#EF4444_1px,transparent_1px)]',
  },
  'purple-500': {
    text: 'text-purple-500',
    bg: 'bg-purple-500',
    bgOpacity: 'bg-purple-500/10',
    border: 'border-purple-500/20',
    hoverBorder: 'group-hover:border-purple-500',
    gradientFrom: 'from-purple-500',
    gradientTo: 'to-purple-500',
    dotBg: 'bg-[radial-gradient(#A855F7_1px,transparent_1px)]',
  },
  'green-500': {
    text: 'text-green-500',
    bg: 'bg-green-500',
    bgOpacity: 'bg-green-500/10',
    border: 'border-green-500/20',
    hoverBorder: 'group-hover:border-green-500',
    gradientFrom: 'from-green-500',
    gradientTo: 'to-green-500',
    dotBg: 'bg-[radial-gradient(#22C55E_1px,transparent_1px)]',
  },
};

const FeatureCards: React.FC<FeatureCardsProps> = ({ info = [] }) => {
  const getHoverColorClass = (colorClass: string) => {
    switch (colorClass) {
      case 'pink-500':
        return 'group-hover:text-pink-500';
      case 'red-500':
        return 'group-hover:text-red-500';
      case 'purple-500':
        return 'group-hover:text-purple-500';
      case 'green-500':
        return 'group-hover:text-green-500';
      default:
        return '';
    }
  };

  return (
    <div className="w-full max-w-6xl mx-auto p-4 sm:p-6 space-y-12 relative">
      <div className="text-center space-y-4">
        <h2 className="text-3xl sm:text-4xl md:text-5xl font-bold bg-gradient-to-b from-gray-900 to-gray-700 dark:from-white dark:to-white/60 bg-clip-text text-transparent tracking-tight">
          Why Developers Choose Us
        </h2>
        <p className="text-gray-600 dark:text-gray-400/80 text-base sm:text-lg md:text-lg tracking-wide">
          Streamlined solutions for modern web development
        </p>
      </div>

      <div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-2 xl:grid-cols-2 gap-4 sm:gap-6 md:gap-8">
        {info.map((card, index) => {
          const colorClass = colorClassMap[card.colorClass];
          const hoverColorClass = getHoverColorClass(card.colorClass);
          
          return (
            <div key={index} className="group relative h-[200px] sm:h-[220px] md:h-[240px] transition-all duration-700">
              <div className={`absolute inset-0 bg-gradient-to-tr ${colorClass.gradientFrom}/5 via-transparent to-transparent dark:${colorClass.gradientFrom}/10 opacity-50 group-hover:opacity-100 transition-opacity duration-700`} />
              <div className={`relative h-full bg-white/80 dark:bg-black/60 backdrop-blur-sm rounded-xl p-4 sm:p-6 md:p-8 border border-md ${colorClass.border} ${colorClass.hoverBorder} transition-all duration-700 group-hover:bg-white/60 dark:group-hover:bg-black/40`}>
                <div className={`absolute inset-0 ${colorClass.dotBg} [background-size:18px_18px] [background-position:0_0] opacity-20 dark:opacity-25 transition-all duration-700 group-hover:opacity-30 dark:group-hover:opacity-40 group-hover:[background-size:14px_14px] group-hover:[background-position:7px_7px]`} />
                <div className="absolute inset-[1px] bg-gradient-to-b from-transparent via-transparent to-white/80 dark:to-black/80 rounded-xl" />

                <div className="relative z-10 h-full flex flex-col justify-between">
                  <div className={`h-8 w-8 sm:h-10 sm:w-10 md:h-12 md:w-12 rounded-full ${colorClass.bgOpacity} dark:${colorClass.bg}/20 flex items-center justify-center group-hover:rotate-[360deg] transition-all duration-700`}>

                  <div className={`h-4 w-4 sm:h-5 sm:w-5 md:h-6 md:w-6 ${colorClass.text}`}
                  >
                    {card.icon}
                  </div>
                  
                    {/* <Check
                      strokeWidth={2.5}
                    /> */}
                  </div>

                  <div className="space-y-2">
                    <h3 className={`text-lg sm:text-xl md:text-2xl font-semibold text-gray-900 dark:text-white/90 tracking-wide ${hoverColorClass} transition-colors`}>
                      {card.title}
                    </h3>
                    <p className="text-gray-600 dark:text-gray-400 leading-relaxed group-hover:text-gray-700 dark:group-hover:text-gray-300 transition-colors">
                      {card.content}
                    </p>
                  </div>
                </div>
              </div>
            </div>
          );
        })}
      </div>
    </div>
  );
};

export default FeatureCards;
3

Implement the code as demonstrated in the preview