December 2, 2024

Marvel Saga Chronicles

Create a website that acts as a dynamic encyclopedia and storytelling hub for Marvel characters, comics, and events. The unique twist is allowing users to create custom storylines by combining different Marvel characters and events, visualized through an interactive timeline and map. Using Next JS 15, Shadcn, Tailwind CSS, Motion Framer, Marvel API, Firebase for Custom Comic creation and Authentication

Marvel Saga Chronicles

Pitch Details

1. Create Navbar

"use client";

import { motion } from "framer-motion";
import Image from "next/image";
import Link from "next/link";
import { useState } from "react";

export default function Navbar() {
  const [isOpen, setIsOpen] = useState(false);

  return (
    <motion.nav
      className="bg-gradient-to-r from-red-600 via-black to-black shadow-xl sticky top-0 z-50"
      initial={{ opacity: 0, y: -50 }}
      animate={{ opacity: 1, y: 0 }}
      transition={{ duration: 0.5 }}
    >
      <div className="max-w-full mx-auto px-6 sm:px-8 lg:px-12">
        <div className="flex items-center justify-between h-20">
          {/* Logo */}
          <motion.div
            className="flex items-center space-x-3"
            whileHover={{ scale: 1.1 }}
          >
            <Link href="/">
              <Image
                src="/Images/marvel-logo.png"
                alt="Marvel Saga Chronicles Logo"
                className="w-10 h-10 sm:w-12 sm:h-12 object-cover"
                width={48}
                height={48}
              />
            </Link>
            <Link href="/">
              <span className="text-white text-xl sm:text-2xl md:text-3xl font-extrabold tracking-wider hidden sm:block">
                Saga Chronicles
              </span>
            </Link>
          </motion.div>

          {/* Desktop Nav Links */}
          <div className="hidden md:flex space-x-6 text-white font-semibold text-lg">
            <Link
              href="#characters"
              className="hover:text-red-400 transition duration-300"
            >
              Characters
            </Link>
            <Link
              href="#comics"
              className="hover:text-red-400 transition duration-300"
            >
              Comics
            </Link>
            <Link
              href="#movies"
              className="hover:text-red-400 transition duration-300"
            >
              Movies
            </Link>
            <Link
              href="#about"
              className="hover:text-red-400 transition duration-300"
            >
              About
            </Link>
          </div>

          {/* Search and Buttons for Desktop */}
          <div className="hidden md:flex items-center space-x-4">
            {/* Search Bar */}
            <div className="relative">
              <input
                type="text"
                placeholder="Search..."
                className="px-4 py-2 w-44 sm:w-56 rounded-full bg-gray-800 text-white border-none focus:ring-2 focus:ring-red-500"
              />
              <button className="absolute right-2 top-1/2 transform -translate-y-1/2 px-4 py-1 bg-red-500 text-white rounded-full hover:bg-red-600 transition">
                Search
              </button>
            </div>
            {/* Login/Signup Buttons */}
            <div className="flex space-x-2">
              <button className="px-4 py-2 bg-gray-800 text-white rounded-full hover:bg-gray-700 transition">
                Login
              </button>
              <button className="px-4 py-2 bg-red-500 text-white rounded-full hover:bg-red-600 transition">
                Signup
              </button>
            </div>
          </div>

          {/* Hamburger Icon for Mobile */}
          <div className="md:hidden">
            <button
              onClick={() => setIsOpen(!isOpen)}
              className="text-white focus:outline-none"
            >
              <svg
                xmlns="http://www.w3.org/2000/svg"
                fill="none"
                viewBox="0 0 24 24"
                strokeWidth={1.5}
                stroke="currentColor"
                className="w-8 h-8"
              >
                <path
                  strokeLinecap="round"
                  strokeLinejoin="round"
                  d="M3.75 5.25h16.5m-16.5 6h16.5m-16.5 6h16.5"
                />
              </svg>
            </button>
          </div>
        </div>
      </div>

      {/* Mobile Menu */}
      {isOpen && (
        <motion.div
          className="md:hidden bg-black text-white space-y-6 px-6 py-6"
          initial={{ opacity: 0, y: -20 }}
          animate={{ opacity: 1, y: 0 }}
          transition={{ duration: 0.3 }}
        >
          <Link
            href="#characters"
            className="block hover:text-red-400 transition duration-300"
          >
            Characters
          </Link>
          <Link
            href="#comics"
            className="block hover:text-red-400 transition duration-300"
          >
            Comics
          </Link>
          <Link
            href="#movies"
            className="block hover:text-red-400 transition duration-300"
          >
            Movies
          </Link>
          <Link
            href="#about"
            className="block hover:text-red-400 transition duration-300"
          >
            About
          </Link>
          <div className="flex flex-col space-y-2">
            <button className="px-4 py-2 bg-gray-800 text-white rounded-full hover:bg-gray-700 transition">
              Login
            </button>
            <button className="px-4 py-2 bg-red-500 text-white rounded-full hover:bg-red-600 transition">
              Signup
            </button>
          </div>
        </motion.div>
      )}
    </motion.nav>
  );
}

2. FeaturedSection

"use client";

import { motion } from "framer-motion";
import { useState, useEffect } from "react";
import Loading from "./Loading";

interface Character {
  id: number;
  name: string;
  thumbnail: {
    path: string;
    extension: string;
  };
}

interface FeaturedSectionProps {
  title: string;
  items: Character[];
}

export default function FeaturedSection({
  title,
  items,
}: FeaturedSectionProps) {
  const [loading, setLoading] = useState(true);

  // Update loading state based on items prop
  useEffect(() => {
    if (items && items.length > 0) {
      setLoading(false);
    }
  }, [items]);

  // Background animation
  const backgroundAnimation = {
    background: [
      "linear-gradient(90deg, rgba(0.0.0.0,1) 0%, rgba(220,38,38,1) 100%)",
      "linear-gradient(90deg, rgba(220,38,38,1) 0%, rgba(0.0.0.0,1) 100%)",
    ],
    transition: {
      duration: 5,
      repeat: Infinity,
      repeatType: "reverse" as const,
    },
  };

  return (
    <motion.section
      className="py-12 px-6 relative overflow-hidden bg-main"
      initial={{ opacity: 0 }}
      animate={{ opacity: 1 }}
      transition={{ duration: 1 }}
    >
      {/* Animated Background */}
      <motion.div
        className="absolute inset-0 z-0"
        animate={backgroundAnimation}
        style={{ backgroundSize: "300% 300%" }}
      />
      <div className="relative z-10">
        {/* Title */}
        <h2 className="text-4xl font-extrabold mb-8 text-center text-light text-shadow-lg">
          {title}
        </h2>

        {/* Loading Spinner */}
        {loading && (
          <div className="flex items-center justify-center">
            <Loading />
          </div>
        )}

        {/* Grid */}
        {!loading && items.length > 0 && (
          <motion.div
            className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-6"
            initial={{ opacity: 0 }}
            animate={{ opacity: 1 }}
            transition={{ duration: 1 }}
          >
            {items.map((item) => (
              <motion.div
                key={item.id}
                className="card relative group"
                whileHover={{ scale: 1.05, translateY: -5 }}
              >
                {/* Image */}
                <img
                  src={`${item.thumbnail.path}.${item.thumbnail.extension}`}
                  alt={item.name}
                  className="w-full h-64 object-cover group-hover:brightness-75 transition"
                />

                {/* Overlay */}
                <div className="absolute inset-0 bg-gradient-to-t from-black to-transparent opacity-0 group-hover:opacity-100 transition-opacity"></div>

                {/* Card Content */}
                <div className="p-4">
                  <h3 className="text-lg font-bold text-primary group-hover:text-yellow-400 transition">
                    {item.name}
                  </h3>
                  <p className="text-light mt-2 text-sm group-hover:text-gray-200 transition">
                    Explore the adventures of {item.name}.
                  </p>
                  <button className="btn btn-primary relative mt-4 overflow-hidden">
                    <span className="absolute inset-0 bg-white opacity-20 group-hover:scale-150 transition-transform"></span>
                    Learn More
                  </button>
                </div>
              </motion.div>
            ))}
          </motion.div>
        )}
      </div>
    </motion.section>
  );
}