190 lines
6.9 KiB
TypeScript
190 lines
6.9 KiB
TypeScript
"use client";
|
|
|
|
import { useState, useEffect } from 'react';
|
|
import { motion } from 'framer-motion';
|
|
import { ExternalLink, Github, Calendar } from 'lucide-react';
|
|
import Link from 'next/link';
|
|
|
|
interface Project {
|
|
id: number;
|
|
title: string;
|
|
description: string;
|
|
content: string;
|
|
tags: string[];
|
|
featured: boolean;
|
|
category: string;
|
|
date: string;
|
|
github?: string;
|
|
live?: string;
|
|
}
|
|
|
|
const Projects = () => {
|
|
const [mounted, setMounted] = useState(false);
|
|
|
|
useEffect(() => {
|
|
setMounted(true);
|
|
}, []);
|
|
|
|
const [projects, setProjects] = useState<Project[]>([]);
|
|
|
|
// Load projects from API
|
|
useEffect(() => {
|
|
const loadProjects = async () => {
|
|
try {
|
|
const response = await fetch('/api/projects?featured=true&published=true&limit=6');
|
|
if (response.ok) {
|
|
const data = await response.json();
|
|
setProjects(data.projects || []);
|
|
} else {
|
|
console.error('Failed to fetch projects from API');
|
|
}
|
|
} catch (error) {
|
|
console.error('Error loading projects:', error);
|
|
}
|
|
};
|
|
|
|
loadProjects();
|
|
}, []);
|
|
|
|
if (!mounted) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<section id="projects" className="py-20 px-4 relative">
|
|
<div className="max-w-7xl mx-auto">
|
|
<motion.div
|
|
initial={{ opacity: 0, y: 30 }}
|
|
whileInView={{ opacity: 1, y: 0 }}
|
|
viewport={{ once: true }}
|
|
transition={{ duration: 0.8 }}
|
|
className="text-center mb-16"
|
|
>
|
|
<h2 className="text-4xl md:text-5xl font-bold mb-6 gradient-text">
|
|
Featured Projects
|
|
</h2>
|
|
<p className="text-xl text-gray-400 max-w-2xl mx-auto">
|
|
Here are some of my recent projects that showcase my skills and passion for creating innovative solutions.
|
|
</p>
|
|
</motion.div>
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-8">
|
|
{projects.map((project, index) => (
|
|
<motion.div
|
|
key={project.id}
|
|
initial={{ opacity: 0, y: 30 }}
|
|
whileInView={{ opacity: 1, y: 0 }}
|
|
viewport={{ once: true }}
|
|
transition={{ duration: 0.6, delay: index * 0.1 }}
|
|
whileHover={{ y: -10 }}
|
|
className={`group relative overflow-hidden rounded-2xl glass-card card-hover ${
|
|
project.featured ? 'ring-2 ring-blue-500/50' : ''
|
|
}`}
|
|
>
|
|
<div className="relative h-48 overflow-hidden">
|
|
<div className="absolute inset-0 bg-gradient-to-br from-blue-500/20 to-purple-500/20" />
|
|
<div className="absolute inset-0 bg-gray-800/50 flex flex-col items-center justify-center p-4">
|
|
<div className="w-16 h-16 bg-gradient-to-br from-blue-500 to-purple-500 rounded-full flex items-center justify-center mb-2">
|
|
<span className="text-2xl font-bold text-white">
|
|
{project.title.split(' ').map(word => word[0]).join('').toUpperCase()}
|
|
</span>
|
|
</div>
|
|
<span className="text-sm font-medium text-gray-400 text-center leading-tight">
|
|
{project.title}
|
|
</span>
|
|
</div>
|
|
|
|
{project.featured && (
|
|
<div className="absolute top-4 right-4 px-3 py-1 bg-gradient-to-r from-blue-500 to-purple-500 text-white text-xs font-semibold rounded-full">
|
|
Featured
|
|
</div>
|
|
)}
|
|
|
|
<div className="absolute inset-0 bg-black/60 opacity-0 group-hover:opacity-100 transition-opacity duration-300 flex items-center justify-center space-x-4">
|
|
{project.github && project.github.trim() !== '' && project.github !== '#' && (
|
|
<motion.a
|
|
href={project.github}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
whileHover={{ scale: 1.1 }}
|
|
whileTap={{ scale: 0.95 }}
|
|
className="p-3 bg-gray-800/80 rounded-lg text-white hover:bg-gray-700/80 transition-colors"
|
|
>
|
|
<Github size={20} />
|
|
</motion.a>
|
|
)}
|
|
{project.live && project.live.trim() !== '' && project.live !== '#' && (
|
|
<motion.a
|
|
href={project.live}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
whileHover={{ scale: 1.1 }}
|
|
whileTap={{ scale: 0.95 }}
|
|
className="p-3 bg-blue-600/80 rounded-lg text-white hover:bg-blue-500/80 transition-colors"
|
|
>
|
|
<ExternalLink size={20} />
|
|
</motion.a>
|
|
)}
|
|
</div>
|
|
</div>
|
|
|
|
<div className="p-6">
|
|
<div className="flex items-center justify-between mb-3">
|
|
<h3 className="text-xl font-bold text-white group-hover:text-blue-400 transition-colors">
|
|
{project.title}
|
|
</h3>
|
|
<div className="flex items-center space-x-2 text-gray-400">
|
|
<Calendar size={16} />
|
|
<span className="text-sm">{project.date}</span>
|
|
</div>
|
|
</div>
|
|
|
|
<p className="text-gray-300 mb-4 leading-relaxed">
|
|
{project.description}
|
|
</p>
|
|
|
|
<div className="flex flex-wrap gap-2 mb-4">
|
|
{project.tags.map((tag) => (
|
|
<span
|
|
key={tag}
|
|
className="px-3 py-1 bg-gray-800/50 text-gray-300 text-sm rounded-full border border-gray-700"
|
|
>
|
|
{tag}
|
|
</span>
|
|
))}
|
|
</div>
|
|
|
|
<Link
|
|
href={`/projects/${project.title.toLowerCase().replace(/[^a-z0-9]+/g, '-')}`}
|
|
className="inline-flex items-center space-x-2 text-blue-400 hover:text-blue-300 transition-colors font-medium"
|
|
>
|
|
<span>View Project</span>
|
|
<ExternalLink size={16} />
|
|
</Link>
|
|
</div>
|
|
</motion.div>
|
|
))}
|
|
</div>
|
|
|
|
<motion.div
|
|
initial={{ opacity: 0, y: 30 }}
|
|
whileInView={{ opacity: 1, y: 0 }}
|
|
viewport={{ once: true }}
|
|
transition={{ duration: 0.8, delay: 0.4 }}
|
|
className="text-center mt-12"
|
|
>
|
|
<Link
|
|
href="/projects"
|
|
className="btn-primary px-8 py-4 text-lg font-semibold inline-flex items-center space-x-2"
|
|
>
|
|
<span>View All Projects</span>
|
|
<ExternalLink size={20} />
|
|
</Link>
|
|
</motion.div>
|
|
</div>
|
|
</section>
|
|
);
|
|
};
|
|
|
|
export default Projects;
|