Components
Testimonials

Testimonials

Join our cutting-edge Testimonials component! This uses Shadcn for UI, Framer Motion for animations, and Supabase for the database. Sign up for early access and updates!

Testimonials

Submit Your Testimonial

Prerequisites

Before you start, ensure you have the following:

Install the Required Dependencies in your project:

  npm install motion/react lucide-react @supabase/supabase-js 
  npx shadcn-ui@latest add avatar carousel

Create a new Project in superbse for reference check superbase check the video

Get Your API keys API URL, anon from supabase.

NEXT_PUBLIC_SUPABASE_URL=REPLACE_WITH_YOUR_KEY
NEXT_PUBLIC_SUPABASE_ANON_KEY=REPLACE_WITH_YOUR_KEY
You're almost done just few Steps left.

Installation

Create a file named testimonials.tsx and copy and paste the following code:

"use client";
 
import { useEffect, useRef, useState } from 'react';
import { createClient } from '@supabase/supabase-js';
import 'tailwindcss/tailwind.css';
import { useRouter } from 'next/navigation';
import { motion } from 'motion/react';
import { ChevronLeft, ChevronLeftIcon, ChevronRight, ChevronRightIcon, Star } from 'lucide-react';
import { toast } from '@/components/ui/use-toast';
import clsx, { ClassValue } from "clsx";
import { twMerge } from "tailwind-merge";
import { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar';
import { Carousel, CarouselContent, CarouselItem, CarouselPrevious, CarouselNext } from '@/components/ui/carousel';
 
 
const supabaseUrl = process.env.NEXT_PUBLIC_SUPABASE_URL!;
const supabaseAnonKey = process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!;
const supabase = createClient(supabaseUrl, supabaseAnonKey);
 
export function cn(...inputs: ClassValue[]) {
  return twMerge(clsx(inputs));
}
 
interface TestimonialFeedbackFormProps {
  initialName?: string;
  initialFeedback?: string;
  initialRating?: number;
  onSubmitSuccess?: () => void;
  onSubmitError?: (error: any) => void;
  redirectUrl?: string;
  title?: string;
  nameLabel?: string;
  feedbackLabel?: string;
  ratingLabel?: string;
  submitButtonText?: string;
}
 
const TestimonialFeedbackForm: React.FC<TestimonialFeedbackFormProps> = ({
  initialName = '',
  initialFeedback = '',
  initialRating = 0,
  onSubmitSuccess,
  onSubmitError,
  redirectUrl = 'https://www.cedzlabs.com',
  title = 'Testimonials',
  nameLabel = 'Name',
  feedbackLabel = 'Feedback',
  ratingLabel = 'Rating',
  submitButtonText = 'Submit',
}) => {
  const [name, setName] = useState(initialName);
  const [feedback, setFeedback] = useState(initialFeedback);
  const [rating, setRating] = useState(initialRating);
  const router = useRouter();
 
  const handleSubmit = async (e: React.FormEvent) => {
    e.preventDefault();
    const { data, error } = await supabase.from('testimonials').insert([{ name, feedback, rating }]);
    if (error) {
      console.error('Error adding testimonial:', error);
      onSubmitError && onSubmitError(error);
    } else {
      setName('');
      setFeedback('');
      setRating(0);
      toast({
        description: "Successfully added your testimonial!",
      });
      onSubmitSuccess && onSubmitSuccess();
      router.push(redirectUrl); // Redirect after successful submission (optional)
    }
  };
 
  const handleRatingChange = (newRating: number) => {
    setRating(newRating);
  };
 
  return (
    <div className="container mx-auto p-4">
      <h1 className="text-3xl font-bold mb-4">{title}</h1>
      <form onSubmit={handleSubmit} className="mb-8">
        <div className="mb-4">
          <label className="block text-sm font-medium text-gray-700 dark:text-gray-300">{nameLabel}</label>
          <input
            type="text"
            value={name}
            onChange={(e) => setName(e.target.value)}
            className={cn("mt-1 block w-full px-3 py-2 border border-gray-300 rounded-md shadow-sm focus:outline-none focus:ring-indigo-500 focus:border-indigo-500 sm:text-sm dark:bg-gray-800 dark:border-gray-600 dark:text-gray-300")}
            required
          />
        </div>
        <div className="mb-4">
          <label className="block text-sm font-medium text-gray-700 dark:text-gray-300">{feedbackLabel}</label>
          <textarea
            value={feedback}
            onChange={(e) => setFeedback(e.target.value)}
            className={cn("mt-1 block w-full px-3 py-2 border border-gray-300 rounded-md shadow-sm focus:outline-none focus:ring-indigo-500 focus:border-indigo-500 sm:text-sm dark:bg-gray-800 dark:border-gray-600 dark:text-gray-300")}
            required
          />
        </div>
        <div className="mb-4">
          <label className="block text-sm font-medium text-gray-700 dark:text-gray-300">{ratingLabel}</label>
          <div className="flex items-center">
            {[1, 2, 3, 4, 5].map((star) => (
              <motion.div
                key={star}
                whileHover={{ scale: 1.2 }}
                whileTap={{ scale: 0.9 }}
                onClick={() => handleRatingChange(star)}
                className="cursor-pointer"
              >
                <Star
                  className={cn("w-8 h-8", star <= rating ? 'text-yellow-500' : 'text-gray-300')}
                  fill={star <= rating ? 'currentColor' : 'none'}
                />
              </motion.div>
            ))}
          </div>
        </div>
        <button
          type="submit"
          className={cn("inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-md shadow-sm text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500")}
        >
          {submitButtonText}
        </button>
      </form>
    </div>
  );
};
 
interface Testimonial {
  id: number;
  name: string;
  feedback: string;
  rating: number;
}
 
const TestimonialsSlider: React.FC = () => {
  const [testimonials, setTestimonials] = useState<Testimonial[]>([]);
  
  useEffect(() => {
    const fetchTestimonials = async () => {
      const { data, error } = await supabase.from('testimonials').select('*');
      if (error) {
        console.error('Error fetching testimonials:', error);
      } else {
        setTestimonials(data);
      }
    };
 
    fetchTestimonials();
  }, []);
 
  return (
    <section className="w-full max-w-4xl mx-auto py-12 md:py-16 lg:py-20">
      <div className="flex flex-col items-center gap-6">
        <h2 className="text-3xl font-bold tracking-tighter sm:text-4xl md:text-5xl">What Our Customers Say</h2>
        <Carousel className="w-full">
          <CarouselContent>
            {testimonials.map((testimonial) => (
              <CarouselItem key={testimonial.id}>
                <div className="grid gap-4 md:grid-cols-2 md:gap-8">
                  <div className="flex justify-center items-center gap-4">
                    <Avatar className="w-16 h-16 border">
                      <AvatarImage src="/placeholder-user.jpg" alt={testimonial.name} />
                      <AvatarFallback>{testimonial.name.charAt(0)}</AvatarFallback>
                    </Avatar>
                    <div>
                      <h3 className="text-lg font-semibold">{testimonial.name}</h3>
                      <p className="text-muted-foreground">Rating: {testimonial.rating}</p>
                    </div>
                  </div>
                  <blockquote className="text-lg leading-relaxed md:text-xl">
                    &ldquo;{testimonial.feedback}&rdquo;
                  </blockquote>
                </div>
              </CarouselItem>
            ))}
          </CarouselContent>
          <CarouselPrevious className="absolute left-4 top-1/2 -translate-y-1/2 rounded-full bg-background p-2 shadow-md transition-all hover:bg-accent hover:text-accent-foreground focus:outline-none focus-visible:ring-1 focus-visible:ring-ring">
            <ChevronLeftIcon className="h-6 w-6" />
          </CarouselPrevious>
          <CarouselNext className="absolute right-4 top-1/2 -translate-y-1/2 rounded-full bg-background p-2 shadow-md transition-all hover:bg-accent hover:text-accent-foreground focus:outline-none focus-visible:ring-1 focus-visible:ring-ring">
            <ChevronRightIcon className="h-6 w-6" />
          </CarouselNext>
        </Carousel>
      </div>
    </section>
  );
};
 
 
interface MarqueeTestimonialsProps {
  duration?: number;
  ease?: string;
  repeat?: number;
  bgColor?: string;
  textColor?: string;
  borderColor?: string;
  starColor?: string;
  starEmptyColor?: string;
}
 
const MarqueeTestimonials: React.FC<MarqueeTestimonialsProps> = ({
  duration = 20,
  ease = 'linear',
  repeat = Infinity,
  bgColor = 'bg-white dark:bg-gray-800',
  textColor = 'text-gray-700 dark:text-gray-300',
  borderColor = 'border-gray-300 dark:border-gray-600',
  starColor = 'text-yellow-500',
  starEmptyColor = 'text-gray-300',
}) => {
  const [testimonials, setTestimonials] = useState<Testimonial[]>([]);
 
  useEffect(() => {
    const fetchTestimonials = async () => {
      const { data, error } = await supabase.from('testimonials').select('*');
      if (error) {
        console.error('Error fetching testimonials:', error);
      } else {
        setTestimonials(data);
      }
    };
 
    fetchTestimonials();
  }, []);
 
  return (
    <div className="flex overflow-x-hidden whitespace-nowrap relative myGradient">
      <motion.div
        className="flex"
        animate={{ x: ['0','-100%'] }}
        transition={{ repeat, duration, ease }}
      >
        {[...testimonials, ...testimonials].map((testimonial, index) => (
          <div
            key={`${testimonial.id}-${index}`}
            className={cn(bgColor, borderColor, "border rounded-lg shadow-lg p-4 m-2 w-80 flex-shrink-0")}
          >
            <h3 className="text-xl font-bold text-center">{testimonial.name}</h3>
            <p className={cn(textColor, "mt-2 line-clamp-3")}>
              {testimonial.feedback}
            </p>
            <div className="flex justify-center mt-4">
              {[1, 2, 3, 4, 5].map((star) => (
                <Star
                  key={star}
                  className={cn("w-6 h-6", star <= testimonial.rating ? starColor : starEmptyColor)}
                  fill={star <= testimonial.rating ? 'currentColor' : 'none'}
                />
              ))}
            </div>
          </div>
        ))}
      </motion.div>
    </div>
  );
};
 
export { TestimonialFeedbackForm, TestimonialsSlider, MarqueeTestimonials };

Create the following .env variables:

  NEXT_PUBLIC_SUPABASE_URL=REPLACE_WITH_YOUR_KEY
  NEXT_PUBLIC_SUPABASE_ANON_KEY=REPLACE_WITH_YOUR_KEY

Run the code in the SQLeditor in supabase.

  CREATE TABLE testimonials (
      id SERIAL PRIMARY KEY,
      name VARCHAR(255) NOT NULL,
      feedback TEXT NOT NULL,
      rating INT CHECK (rating >= 0 AND rating <= 5) NOT NULL
  );
Congrats! You have successfully created a Testimonials component for your project! 🎉

Props

Form Feedback

Prop nameTypeDefaultDescription
initialNamestring''The initial value for the name input field.
initialFeedbackstring''The initial value for the feedback textarea.
initialRatingnumber0The initial value for the rating (number of stars).
onSubmitSuccessfunctionundefinedCallback function to be called on successful form submission.
onSubmitErrorfunctionundefinedCallback function to be called on form submission error.
redirectUrlstring'https://www.cedzlabs.com'URL to redirect to after successful form submission.
titlestring'Testimonials'The title of the form.
nameLabelstring'Name'The label for the name input field.
feedbackLabelstring'Feedback'The label for the feedback textarea.
ratingLabelstring'Rating'The label for the rating section.
submitButtonTextstring'Submit'The text to display on the submit button.

Examples

Example 1: Marquee

Testimonials

Props for marquee testominals

Prop nameTypeDefaultDescription
durationnumber20Controls the duration of the marquee animation.
easestring'linear'Controls the easing function of the animation.
repeatnumberInfinityControls how many times the animation should repeat.
bgColorstring'bg-white dark:bg-gray-800'Controls the background color of the testimonial cards.
textColorstring'text-gray-700 dark:text-gray-300'Controls the text color of the feedback.
borderColorstring'border-gray-300 dark:border-gray-600'Controls the border color of the testimonial cards.
starColorstring'text-yellow-500'Controls the color of the filled stars.
starEmptyColorstring'text-gray-300'Controls the color of the empty stars.

Example 2: slider

What Our Customers Say

Credits

NameCredit ForLink
YakkshitComponent Developeryakkshit
Cedz LabsVideo TutorialCedz Labs
snadcn uiUI componentssnadcn ui
supabaseDB, Documentationsupabase

Suggestions Or Book

NameCredit ForContact
YakkshitComponent Developer