My Code Chronicles #2: Creating a Dynamic Product Card Component in Next.js using Framer Motion

My Code Chronicles #2: Creating a Dynamic Product Card Component in Next.js using Framer Motion


Hover effects are more than just eye candy—they are essential for creating an engaging user experience on modern websites. When implemented thoughtfully, hover effects can provide subtle visual feedback, improve usability, and guide user interaction intuitively.

For e-commerce websites, like the one I was building for Glamour & Gourmandise, the product card is a critical component. It needs to be visually appealing, informative, and highly interactive. To achieve this, I implemented a sophisticated hover effect that reveals additional product details in a sleek and seamless way.



Code

"use client";

import { motion } from "framer-motion" 
// Imports...


export default function ProductCard(){

  // State
  const [isHovered, setIsHovered] = useState(false);

  return (
    <article
      tabIndex={0}
      className={cn(
        "rounded-special group/product relative flex h-fit cursor-pointer flex-col overflow-hidden",
        "aspect-[4/5] border-2 border-dashed border-primary/20 sm:aspect-square md:aspect-[4/5]",
      )}
      onMouseEnter={() => setIsHovered(true)}
      onMouseLeave={() => setIsHovered(false)}
    >
      {/* Product Image */}
      <Image
       src={image}
       alt={alt}
       className={cn(
         "h-auto w-full flex-none object-cover",
         "h-[58%] flex-none sm:aspect-auto",
       )}
      />

      {/* Product Details */}
      <div
        className={cn(
          "group relative h-[42%] w-full grow",
          isHovered ? "is-hovered" : "",
        )}
      >
        <motion.div
          layout
          transition={{
            layout: { duration: 0.1 },
            // default: { ease: "linear" },
          }}
          className={cn(
            "rounded-special absolute bottom-0 left-0 flex min-h-[105%] w-full flex-col justify-between space-y-2 bg-white p-2 group-[.is-hovered]:h-auto sm:space-y-4 sm:p-3 lg:p-4",
            "",
          )}
        >
          <div className="flex-none">
            <h3 className="line-clamp-2 font-title font-semibold !leading-[1.1] min-[480px]:text-lg min-[545px]:text-2xl sm:mb-1">
              {product.title}
            </h3>
            <div className="mb-1 hidden h-0 items-center space-x-1 overflow-hidden group-[.is-hovered]:flex group-[.is-hovered]:h-auto sm:mb-3">
              {product.tags.map(
                ({ tag }, i) =>
                  i < 2 && (
                    <Badge variant="product-tag" key={i} className="">
                      {tag}
                    </Badge>
                  ),
              )}
            </div>
            <p className="line-clamp-3 hidden text-xs leading-[1.1] text-muted-foreground group-[.is-hovered]:line-clamp-3 md:text-sm">
              {product.description}
            </p>
            {product.link && (
              <Link
                onClick={(e) => e.stopPropagation()}
                replace
                href={product.link}
                className="block w-fit text-xs font-semibold text-primary transition hover:text-secondary group-[.is-hovered]:block sm:hidden"
              >
                More info
              </Link>
            )}
          </div>

          <div className="flex flex-wrap items-center justify-between gap-2">
            <div className="">
              <p className="font-title text-sm font-semibold leading-none min-[400px]:text-xl">
                {product.formattedPrice.priceWithCurrency}
              </p>
              <span className="hidden text-right text-[0.6rem] leading-none text-secondary min-[545px]:block sm:text-xs">
                {product.formattedPrice.priceUnit}
              </span>
            </div>

            {/* Add To Cart Button */}
            <Button size="sm" disabled={true}>
                  <span className="block size-4 animate-spin rounded-full border border-current border-t-transparent" />
                </Button>
          </div>
        </motion.div>
      </div>
    </article>
  );
}
Enter fullscreen mode

Exit fullscreen mode



Implementation



The Wrapper

The article element is the foundation of this hover effect. It is responsible for:

  • Maintaining the aspect ratio (responsively adjusted based on screen size).
  • Housing all child components, including the image and product details.
  • Triggering hover state changes.
  • Aspect Ratio: Ensures consistent sizing of the card across different devices.
  • Hover State: Sets isHovered to true or false to control the animation.



Product Details Section

The product details are housed in an absolutely positioned div at the bottom of the wrapper. This positioning ensures that the expanding animation occurs within the wrapper, avoiding disruptions to the page layout.
Framer Motion’s layout={true} is the key to this hover effect. It:

  • Animates Height Changes: When the product details expand, layout={true} calculates the difference in height and interpolates the transition smoothly.
  • Direction Control: Because the details div is positioned absolutely at the bottom, the expansion happens upward into the card instead of downward.
  • User-Friendly Transitions: Without layout={true}, the height change would feel abrupt and jarring.



Conclusion

This hover effect showcases how to elevate user experience with smooth animations and thoughtful design. While the actual website for Glamour & Gourmandise opted for a simpler zoom-in effect on images (as per the owner’s preference), this implementation demonstrates the versatility and creative possibilities with Framer Motion and Next.js.
If you’re interested in learning more about layout animations or have tips to optimize this implementation further, let me know!



Source link
lol

By stp2y

Leave a Reply

Your email address will not be published. Required fields are marked *

No widgets found. Go to Widget page and add the widget in Offcanvas Sidebar Widget Area.