Portfolio / frontend /src /components /Terminal.jsx
parthib07's picture
Update frontend/src/components/Terminal.jsx
9ebc7bb verified
import { useState, useEffect, useRef } from 'react'
import { motion, AnimatePresence } from 'framer-motion'
import { FiTerminal, FiX, FiMinus, FiMaximize2 } from 'react-icons/fi'
import resumePDF from '../assets/Resume.pdf'
const PROFILE = {
name: 'Parthib Karak',
title: 'Software & AI Engineer',
location: 'West Bengal, India',
college: 'IEM Kolkata (B.Tech CSE)',
cgpa: '8.9',
email: 'parthibkarak2004@gmail.com',
github: 'https://github.com/babaiii07',
linkedin: 'https://www.linkedin.com/in/parthib-karak',
leetcode: 'https://leetcode.com/u/Parthib_007',
}
const COMMANDS = {
help: {
desc: 'Show all available commands',
run: () => [
{ type: 'title', text: '📋 Available Commands' },
{ type: 'divider' },
{ type: 'cmd', text: 'whoami → Who is Parthib?' },
{ type: 'cmd', text: 'skills → Technical skill stack' },
{ type: 'cmd', text: 'projects → Featured AI projects' },
{ type: 'cmd', text: 'experience → Work experience' },
{ type: 'cmd', text: 'education → Academic background' },
{ type: 'cmd', text: 'contact → Get in touch' },
{ type: 'cmd', text: 'resume → Download resume' },
{ type: 'cmd', text: 'leetcode → DSA stats' },
{ type: 'cmd', text: 'hire → Why hire Parthib?' },
{ type: 'cmd', text: 'clear → Clear terminal' },
{ type: 'cmd', text: 'help → Show this menu' },
{ type: 'divider' },
{ type: 'info', text: '💡 Tip: Use ↑ ↓ arrow keys for command history' },
],
},
whoami: {
desc: 'Who is Parthib?',
run: () => [
{ type: 'title', text: '👤 Parthib Karak' },
{ type: 'divider' },
{ type: 'kv', key: 'Role ', value: 'Software & AI Engineer' },
{ type: 'kv', key: 'Location', value: 'West Bengal, India' },
{ type: 'kv', key: 'College ', value: 'IEM Kolkata — B.Tech CSE' },
{ type: 'kv', key: 'CGPA ', value: '8.9 / 10.0' },
{ type: 'kv', key: 'Status ', value: '🟢 Open to Opportunities' },
{ type: 'divider' },
{ type: 'text', text: 'Building production-grade AI systems that don\'t break in the' },
{ type: 'text', text: 'real world. Specialising in RAG, multi-agent systems, and LLM' },
{ type: 'text', text: 'engineering with LangChain, LangGraph, and Groq.' },
],
},
skills: {
desc: 'Technical skill stack',
run: () => [
{ type: 'title', text: '🛠️ Technical Skills' },
{ type: 'divider' },
{ type: 'category', text: 'AI / ML / LLMs' },
{ type: 'tags', items: ['LangChain', 'LangGraph', 'Groq', 'RAG', 'AutoGen', 'HuggingFace', 'Transformers'] },
{ type: 'category', text: 'Backend' },
{ type: 'tags', items: ['FastAPI', 'Django', 'Go / Gin', 'REST APIs', 'WebSockets'] },
{ type: 'category', text: 'Databases & Vector Stores' },
{ type: 'tags', items: ['MongoDB', 'MySQL', 'FAISS', 'ChromaDB', 'Pinecone'] },
{ type: 'category', text: 'Languages' },
{ type: 'tags', items: ['Python', 'JavaScript', 'TypeScript', 'Go', 'C++'] },
{ type: 'category', text: 'Tools & DevOps' },
{ type: 'tags', items: ['Docker', 'Git', 'GitHub', 'Postman', 'VS Code'] },
],
},
projects: {
desc: 'Featured AI projects',
run: () => [
{ type: 'title', text: '🚀 Featured Projects' },
{ type: 'divider' },
{ type: 'proj', num: '01', name: 'Healthcare Chatbot', tech: 'RAG · Pinecone · LangChain · Groq', url: 'https://parthib07-multiagent-healthcare-chatbot.hf.space' },
{ type: 'proj', num: '02', name: 'AI Virtual Dev POD', tech: 'Multi-Agent · LangGraph · ChromaDB', url: 'https://agnik28-ai-virtual-pod.hf.space/' },
{ type: 'proj', num: '03', name: 'Virtual Research Assistant', tech: 'AutoGen · Groq · MySQL', url: 'https://parthib07-virtual-research-paper-assistant.hf.space/' },
{ type: 'proj', num: '04', name: 'OwnGPT', tech: 'FastAPI · LangGraph · MongoDB · Google Auth', url: 'https://parthib07-owngpt-v2.hf.space' },
{ type: 'proj', num: '05', name: 'CodeFusion', tech: 'DeepSeek AI · FastAPI · React', url: 'https://codefusion-v2-parthib.onrender.com' },
{ type: 'proj', num: '06', name: 'AnySITE', tech: 'FastAPI · React · AI Models', url: 'https://anysite-vibecoding-parthib.onrender.com' },
{ type: 'divider' },
{ type: 'info', text: '🔗 Run "contact" to connect or visit GitHub for source code' },
],
},
experience: {
desc: 'Work experience',
run: () => [
{ type: 'title', text: '💼 Work Experience' },
{ type: 'divider' },
{ type: 'kv', key: 'Company ', value: 'BAAR Technologies' },
{ type: 'kv', key: 'Role ', value: 'Jr. Automation Engineer' },
{ type: 'kv', key: 'Type ', value: 'Full-time' },
{ type: 'kv', key: 'Period ', value: 'Feb 2026 – Mar 2026 · 2 mos' },
{ type: 'kv', key: 'Location', value: 'Kolkata, West Bengal · On-site' },
{ type: 'text', text: '' },
{ type: 'text', text: '→ Built automation-driven backend systems using Python & Django' },
{ type: 'text', text: '→ Managed high-performance databases with Microsoft SQL Server' },
{ type: 'text', text: '→ Developed scalable APIs and optimised SQL queries & procedures' },
{ type: 'text', text: '→ Collaborated cross-functionally to deliver reliable automation solutions' },
],
},
education: {
desc: 'Academic background',
run: () => [
{ type: 'title', text: '🎓 Education' },
{ type: 'divider' },
{ type: 'kv', key: 'Degree ', value: 'B.Tech — Computer Science & Engineering' },
{ type: 'kv', key: 'College ', value: 'Institute of Engineering & Management, Kolkata' },
{ type: 'kv', key: 'CGPA ', value: '8.9 / 10.0' },
{ type: 'kv', key: 'Year ', value: '2021 – 2025' },
{ type: 'divider' },
{ type: 'kv', key: 'Class XII', value: '93.6% — Kapsit High School (2021)' },
{ type: 'kv', key: 'Class X ', value: '84.5% — Kamarpukur RKM School (2019)' },
],
},
contact: {
desc: 'Get in touch',
run: () => [
{ type: 'title', text: '📬 Contact Parthib' },
{ type: 'divider' },
{ type: 'link', label: 'Email ', value: 'parthibkarak2004@gmail.com', href: 'mailto:parthibkarak2004@gmail.com' },
{ type: 'link', label: 'LinkedIn', value: 'linkedin.com/in/parthib-karak', href: 'https://www.linkedin.com/in/parthib-karak' },
{ type: 'link', label: 'GitHub ', value: 'github.com/babaiii07', href: 'https://github.com/babaiii07' },
{ type: 'link', label: 'LeetCode', value: 'leetcode.com/u/Parthib_007', href: 'https://leetcode.com/u/Parthib_007' },
{ type: 'divider' },
{ type: 'info', text: '💌 Available for freelance, internship & full-time roles' },
],
},
resume: {
desc: 'Download resume',
run: () => {
setTimeout(() => { window.open(resumePDF, '_blank') }, 300)
return [
{ type: 'title', text: '📄 Resume' },
{ type: 'divider' },
{ type: 'success', text: '✅ Opening resume in a new tab...' },
{ type: 'info', text: '📥 Download will start automatically' },
]
},
},
leetcode: {
desc: 'DSA stats',
run: () => [
{ type: 'title', text: '⚡ LeetCode Stats' },
{ type: 'divider' },
{ type: 'kv', key: 'Profile ', value: 'leetcode.com/u/Parthib_007' },
{ type: 'kv', key: 'Solved ', value: '500+ Problems' },
{ type: 'kv', key: 'Strength', value: 'Arrays · DP · Graphs · Trees · Strings' },
{ type: 'divider' },
{ type: 'text', text: '→ Strong DSA foundation for system design & optimization' },
{ type: 'text', text: '→ Active in competitive programming contests' },
],
},
hire: {
desc: 'Why hire Parthib?',
run: () => [
{ type: 'title', text: '🌟 Why Hire Parthib?' },
{ type: 'divider' },
{ type: 'text', text: '✦ Builds production-ready AI — not just demos' },
{ type: 'text', text: '✦ Specialised in RAG, Agents, LangChain & LangGraph' },
{ type: 'text', text: '✦ 500+ LeetCode — strong algorithmic foundations' },
{ type: 'text', text: '✦ CGPA 8.9 — academic excellence under real workload' },
{ type: 'text', text: '✦ End-to-end: from LLM to Docker to production deploy' },
{ type: 'text', text: '✦ Fast learner, self-driven, ships things that work' },
{ type: 'divider' },
{ type: 'success', text: '🚀 Ready to contribute from day one — run "contact" to connect!' },
],
},
clear: {
desc: 'Clear terminal',
run: () => null,
},
}
function renderLine(line, i) {
const base = { fontSize: '0.78rem', lineHeight: 1.7, fontFamily: 'var(--font-mono)' }
switch (line.type) {
case 'title':
return <div key={i} style={{ ...base, color: '#a78bfa', fontWeight: 700, marginTop: 4 }}>{line.text}</div>
case 'divider':
return <div key={i} style={{ ...base, color: 'rgba(99,102,241,0.3)', margin: '2px 0' }}>{'─'.repeat(44)}</div>
case 'kv':
return (
<div key={i} style={{ ...base, display: 'flex', gap: 8 }}>
<span style={{ color: '#64748b', minWidth: 80 }}>{line.key}</span>
<span style={{ color: '#e2e8f0' }}>{line.value}</span>
</div>
)
case 'text':
return <div key={i} style={{ ...base, color: '#94a3b8' }}>{line.text || '\u00A0'}</div>
case 'info':
return <div key={i} style={{ ...base, color: '#38bdf8' }}>{line.text}</div>
case 'success':
return <div key={i} style={{ ...base, color: '#4ade80', fontWeight: 600 }}>{line.text}</div>
case 'cmd':
return <div key={i} style={{ ...base, color: '#94a3b8' }}>{line.text}</div>
case 'category':
return <div key={i} style={{ ...base, color: '#06b6d4', fontWeight: 600, marginTop: 6 }}>{line.text}</div>
case 'tags':
return (
<div key={i} style={{ display: 'flex', flexWrap: 'wrap', gap: 4, marginBottom: 4 }}>
{line.items.map((t, j) => (
<span key={j} style={{
fontSize: '0.68rem', fontFamily: 'var(--font-mono)',
background: 'rgba(99,102,241,0.12)', border: '1px solid rgba(99,102,241,0.2)',
color: '#a78bfa', borderRadius: 4, padding: '1px 7px',
}}>{t}</span>
))}
</div>
)
case 'proj':
return (
<div key={i} style={{ ...base, display: 'flex', gap: 8, alignItems: 'baseline' }}>
<span style={{ color: '#6366f1', minWidth: 24 }}>{line.num}</span>
<span style={{ color: '#f0f4ff', fontWeight: 600, minWidth: 160 }}>{line.name}</span>
<span style={{ color: '#64748b', fontSize: '0.68rem' }}>{line.tech}</span>
{line.url && (
<a href={line.url} target="_blank" rel="noopener noreferrer"
style={{ color: '#38bdf8', fontSize: '0.68rem', textDecoration: 'none', marginLeft: 4 }}>↗ live</a>
)}
</div>
)
case 'link':
return (
<div key={i} style={{ ...base, display: 'flex', gap: 8 }}>
<span style={{ color: '#64748b', minWidth: 80 }}>{line.label}</span>
<a href={line.href} target="_blank" rel="noopener noreferrer"
style={{ color: '#38bdf8', textDecoration: 'none' }}
onMouseEnter={e => e.target.style.textDecoration = 'underline'}
onMouseLeave={e => e.target.style.textDecoration = 'none'}>
{line.value}
</a>
</div>
)
default:
return null
}
}
export default function Terminal() {
const [open, setOpen] = useState(false)
const [input, setInput] = useState('')
const [history, setHistory] = useState([])
const [cmdHistory, setCmdHistory] = useState([])
const [histIdx, setHistIdx] = useState(-1)
const inputRef = useRef(null)
const bottomRef = useRef(null)
// Welcome message on first open
useEffect(() => {
if (open && history.length === 0) {
setHistory([{
type: 'welcome',
lines: [
{ type: 'title', text: '⚡ Parthib\'s Portfolio Terminal' },
{ type: 'divider' },
{ type: 'text', text: 'Welcome, recruiter! Type "help" to see all commands.' },
{ type: 'text', text: 'Explore my profile interactively below.' },
{ type: 'divider' },
{ type: 'info', text: '→ Try: whoami · skills · projects · hire · resume' },
],
}])
}
}, [open])
useEffect(() => {
if (open) {
setTimeout(() => inputRef.current?.focus(), 80)
}
}, [open])
useEffect(() => {
bottomRef.current?.scrollIntoView({ behavior: 'smooth' })
}, [history])
const runCommand = (raw) => {
const cmd = raw.trim().toLowerCase()
if (!cmd) return
setCmdHistory(prev => [cmd, ...prev])
setHistIdx(-1)
if (cmd === 'clear') {
setHistory([])
return
}
const def = COMMANDS[cmd]
const outputLines = def ? def.run() : [
{ type: 'text', text: `bash: ${cmd}: command not found` },
{ type: 'info', text: 'Type "help" to see available commands.' },
]
setHistory(prev => [
...prev,
{ type: 'entry', cmd, lines: outputLines || [] },
])
}
const handleKey = (e) => {
if (e.key === 'Enter') {
runCommand(input)
setInput('')
} else if (e.key === 'ArrowUp') {
e.preventDefault()
const next = Math.min(histIdx + 1, cmdHistory.length - 1)
setHistIdx(next)
setInput(cmdHistory[next] ?? '')
} else if (e.key === 'ArrowDown') {
e.preventDefault()
const next = Math.max(histIdx - 1, -1)
setHistIdx(next)
setInput(next === -1 ? '' : cmdHistory[next] ?? '')
}
}
return (
<>
{/* Floating Toggle Button */}
<motion.button
onClick={() => setOpen(v => !v)}
style={{
position: 'fixed', bottom: 28, right: 28, zIndex: 9998,
width: 52, height: 52, borderRadius: '50%',
background: 'linear-gradient(135deg, #6366f1, #06b6d4)',
border: 'none', cursor: 'pointer',
display: 'flex', alignItems: 'center', justifyContent: 'center',
color: '#fff', fontSize: '1.25rem',
boxShadow: '0 8px 32px rgba(99,102,241,0.55), 0 2px 8px rgba(0,0,0,0.4)',
}}
whileHover={{ scale: 1.1, boxShadow: '0 12px 40px rgba(99,102,241,0.7)' }}
whileTap={{ scale: 0.95 }}
title="Open Portfolio Terminal"
aria-label="Open terminal"
>
<motion.span
animate={{ rotate: open ? 90 : 0 }}
transition={{ duration: 0.25 }}
style={{ display: 'flex' }}
>
{open ? <FiX size={22} /> : <FiTerminal size={22} />}
</motion.span>
</motion.button>
{/* Terminal Window */}
<AnimatePresence>
{open && (
<motion.div
initial={{ opacity: 0, y: 40, scale: 0.95 }}
animate={{ opacity: 1, y: 0, scale: 1 }}
exit={{ opacity: 0, y: 40, scale: 0.95 }}
transition={{ duration: 0.3, ease: [0.22, 1, 0.36, 1] }}
style={{
position: 'fixed', bottom: 90, right: 28, zIndex: 9997,
width: 'min(600px, calc(100vw - 32px))',
height: 'min(480px, 60vh)',
background: 'rgba(8,10,26,0.97)',
backdropFilter: 'blur(24px)',
borderRadius: 16,
border: '1px solid rgba(99,102,241,0.3)',
boxShadow: '0 24px 80px rgba(0,0,0,0.7), 0 0 0 1px rgba(99,102,241,0.1)',
display: 'flex', flexDirection: 'column',
overflow: 'hidden',
}}
onClick={() => inputRef.current?.focus()}
>
{/* Title Bar */}
<div style={{
display: 'flex', alignItems: 'center', gap: 8,
padding: '10px 16px',
background: 'rgba(99,102,241,0.06)',
borderBottom: '1px solid rgba(99,102,241,0.15)',
flexShrink: 0,
}}>
<div style={{ display: 'flex', gap: 6 }}>
{['#f43f5e', '#f59e0b', '#4ade80'].map((c, i) => (
<div key={i} style={{ width: 11, height: 11, borderRadius: '50%', background: c, opacity: 0.85, cursor: i === 0 ? 'pointer' : 'default' }}
onClick={i === 0 ? () => setOpen(false) : undefined} />
))}
</div>
<div style={{
flex: 1, textAlign: 'center',
fontSize: '0.72rem', fontFamily: 'var(--font-mono)',
color: '#475569', letterSpacing: '0.05em',
}}>
<FiTerminal size={11} style={{ display: 'inline', marginRight: 5, verticalAlign: 'middle' }} />
parthib@portfolio ~ zsh
</div>
<button onClick={() => setOpen(false)} style={{ background: 'none', border: 'none', cursor: 'pointer', color: '#475569', display: 'flex' }}>
<FiX size={14} />
</button>
</div>
{/* Output Area */}
<div style={{
flex: 1, overflowY: 'auto', padding: '12px 16px',
scrollbarWidth: 'thin', scrollbarColor: 'rgba(99,102,241,0.3) transparent',
}}>
{history.map((entry, ei) => (
<div key={ei} style={{ marginBottom: 10 }}>
{entry.type === 'entry' && (
<div style={{
display: 'flex', gap: 6, alignItems: 'baseline',
fontSize: '0.78rem', fontFamily: 'var(--font-mono)',
marginBottom: 4,
}}>
<span style={{ color: '#4ade80' }}>parthib</span>
<span style={{ color: '#475569' }}>@portfolio</span>
<span style={{ color: '#38bdf8' }}>~</span>
<span style={{ color: '#64748b' }}>$</span>
<span style={{ color: '#e2e8f0' }}>{entry.cmd}</span>
</div>
)}
{entry.lines?.map((line, li) => renderLine(line, li))}
</div>
))}
<div ref={bottomRef} />
</div>
{/* Input Row */}
<div style={{
display: 'flex', alignItems: 'center', gap: 8,
padding: '10px 16px',
borderTop: '1px solid rgba(99,102,241,0.12)',
background: 'rgba(0,0,0,0.2)',
flexShrink: 0,
}}>
<span style={{ fontSize: '0.78rem', fontFamily: 'var(--font-mono)', color: '#4ade80', flexShrink: 0 }}>parthib</span>
<span style={{ fontSize: '0.78rem', fontFamily: 'var(--font-mono)', color: '#475569', flexShrink: 0 }}>@portfolio</span>
<span style={{ fontSize: '0.78rem', fontFamily: 'var(--font-mono)', color: '#38bdf8', flexShrink: 0 }}>~</span>
<span style={{ fontSize: '0.78rem', fontFamily: 'var(--font-mono)', color: '#64748b', flexShrink: 0 }}>$</span>
<input
ref={inputRef}
value={input}
onChange={e => setInput(e.target.value)}
onKeyDown={handleKey}
placeholder="type a command..."
autoComplete="off"
spellCheck={false}
style={{
flex: 1, background: 'none', border: 'none', outline: 'none',
fontFamily: 'var(--font-mono)', fontSize: '0.78rem',
color: '#e2e8f0', caretColor: '#a78bfa',
}}
/>
<span style={{
width: 8, height: 15, background: '#6366f1', borderRadius: 1,
animation: 'blink-cursor 0.8s step-end infinite', flexShrink: 0,
}} />
</div>
</motion.div>
)}
</AnimatePresence>
</>
)
}