mirror of
https://github.com/x1xhlol/system-prompts-and-models-of-ai-tools.git
synced 2025-12-16 05:25:11 +00:00
This commit adds a complete, production-ready web application for browsing and exploring AI tool system prompts. The interface provides an intuitive, responsive, and feature-rich experience for discovering AI tools. ## Web Application Features **Core Functionality:** - 🔍 Advanced search with real-time filtering - 📊 Interactive statistics dashboard with visualizations - 🔄 Side-by-side comparison of up to 4 tools - ⭐ Favorites system with local persistence - 📱 Fully responsive mobile-first design - 🎨 Dark/light mode with system preference detection - ⚡ Optimized performance with Next.js Server Components **Pages:** - Home: Hero section, features showcase, featured tools - Browse: Advanced filtering, grid/list views, category filters - Stats: Comprehensive analytics and visualizations - Compare: Side-by-side tool comparison - Tool Detail: In-depth tool information - About: Project information and tech stack **Components:** - Responsive navbar with mobile menu - Tool cards with interactive actions - Reusable UI components (Button, Card, Badge, Input) - Footer with quick links and stats - Theme provider for dark mode - Loading and error states ## Technical Stack **Framework & Libraries:** - Next.js 15 with App Router - React 19.0 with Server Components - TypeScript 5.6 for type safety - Tailwind CSS for styling - Zustand for state management - next-themes for theme switching - Lucide React for icons **Features:** - Server-side rendering (SSR) - Static site generation (SSG) for tool pages - Optimized bundle with automatic code splitting - SEO-friendly with metadata API - Accessibility best practices ## Project Structure web/ ├── app/ # Next.js pages ├── components/ # React components ├── lib/ # Utilities and data ├── data/ # Static data (index.json) ├── setup.sh # Setup script └── README.md # Documentation ## Developer Experience - TypeScript for type safety - ESLint for code quality - Hot module replacement - Fast refresh - Comprehensive documentation - Setup script for quick start ## Updated Documentation - Enhanced main README with web interface section - Created comprehensive web/README.md - Updated roadmap to mark completed features - Added Quick Start guide for web app ## Stats - 33 new files created - ~3,500 lines of TypeScript/TSX - Full responsive design (mobile, tablet, desktop) - Production-ready with build optimizations Users can now explore 32+ AI tools through an intuitive web interface instead of just command-line tools. Version: 2.0.0
171 lines
5.9 KiB
TypeScript
171 lines
5.9 KiB
TypeScript
'use client'
|
|
|
|
import Link from 'next/link'
|
|
import { Heart, GitCompare, ExternalLink, FileText, Code } from 'lucide-react'
|
|
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@/components/ui/card'
|
|
import { Badge } from '@/components/ui/badge'
|
|
import { Button } from '@/components/ui/button'
|
|
import { AITool } from '@/lib/types'
|
|
import { getCategoryIcon, getCategoryColor } from '@/lib/data'
|
|
import { useAppStore } from '@/lib/store'
|
|
import { formatNumber, slugify } from '@/lib/utils'
|
|
import { cn } from '@/lib/utils'
|
|
|
|
interface ToolCardProps {
|
|
tool: AITool
|
|
variant?: 'default' | 'compact'
|
|
}
|
|
|
|
export function ToolCard({ tool, variant = 'default' }: ToolCardProps) {
|
|
const { favorites, addFavorite, removeFavorite, isFavorite } = useAppStore()
|
|
const { comparison, addToComparison, removeFromComparison, isInComparison } = useAppStore()
|
|
|
|
const favorite = isFavorite(tool.directory)
|
|
const inComparison = isInComparison(tool.directory)
|
|
|
|
const toggleFavorite = (e: React.MouseEvent) => {
|
|
e.preventDefault()
|
|
if (favorite) {
|
|
removeFavorite(tool.directory)
|
|
} else {
|
|
addFavorite(tool.directory)
|
|
}
|
|
}
|
|
|
|
const toggleComparison = (e: React.MouseEvent) => {
|
|
e.preventDefault()
|
|
if (inComparison) {
|
|
removeFromComparison(tool.directory)
|
|
} else if (comparison.length < 4) {
|
|
addToComparison(tool.directory)
|
|
}
|
|
}
|
|
|
|
if (variant === 'compact') {
|
|
return (
|
|
<Link href={`/tool/${slugify(tool.directory)}`}>
|
|
<Card className="h-full hover:shadow-lg transition-all cursor-pointer group">
|
|
<CardHeader className="pb-3">
|
|
<div className="flex items-start justify-between gap-2">
|
|
<div className="flex-1 min-w-0">
|
|
<CardTitle className="text-lg line-clamp-1 group-hover:text-primary transition-colors">
|
|
{tool.name}
|
|
</CardTitle>
|
|
<CardDescription className="text-xs line-clamp-1 mt-1">
|
|
{tool.company}
|
|
</CardDescription>
|
|
</div>
|
|
<div className={cn('text-2xl flex-shrink-0', getCategoryColor(tool.category))}>
|
|
{getCategoryIcon(tool.category)}
|
|
</div>
|
|
</div>
|
|
</CardHeader>
|
|
<CardContent className="pb-3">
|
|
<div className="flex flex-wrap gap-1">
|
|
<Badge variant="secondary" className="text-xs">
|
|
{tool.category}
|
|
</Badge>
|
|
{tool.models.slice(0, 1).map((model) => (
|
|
<Badge key={model} variant="outline" className="text-xs">
|
|
{model}
|
|
</Badge>
|
|
))}
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
</Link>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<Card className="h-full flex flex-col hover:shadow-lg transition-all group">
|
|
<CardHeader>
|
|
<div className="flex items-start justify-between gap-4">
|
|
<div className="flex-1 min-w-0">
|
|
<div className="flex items-center gap-2 mb-2">
|
|
<div className="text-3xl">{getCategoryIcon(tool.category)}</div>
|
|
<Badge variant="secondary">{tool.category}</Badge>
|
|
</div>
|
|
<CardTitle className="text-xl line-clamp-1 group-hover:text-primary transition-colors">
|
|
{tool.name}
|
|
</CardTitle>
|
|
<CardDescription className="line-clamp-1 mt-1">
|
|
{tool.company}
|
|
</CardDescription>
|
|
</div>
|
|
<div className="flex gap-1 flex-shrink-0">
|
|
<Button
|
|
size="icon"
|
|
variant="ghost"
|
|
onClick={toggleFavorite}
|
|
className={cn('h-8 w-8', favorite && 'text-red-500')}
|
|
>
|
|
<Heart className={cn('w-4 h-4', favorite && 'fill-current')} />
|
|
</Button>
|
|
<Button
|
|
size="icon"
|
|
variant="ghost"
|
|
onClick={toggleComparison}
|
|
className={cn('h-8 w-8', inComparison && 'text-blue-500')}
|
|
disabled={!inComparison && comparison.length >= 4}
|
|
>
|
|
<GitCompare className={cn('w-4 h-4', inComparison && 'fill-current')} />
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</CardHeader>
|
|
|
|
<CardContent className="flex-1">
|
|
<p className="text-sm text-muted-foreground line-clamp-3 mb-4">
|
|
{tool.description}
|
|
</p>
|
|
|
|
<div className="space-y-2">
|
|
{/* Models */}
|
|
{tool.models.length > 0 && (
|
|
<div className="flex flex-wrap gap-1">
|
|
{tool.models.slice(0, 3).map((model) => (
|
|
<Badge key={model} variant="outline" className="text-xs">
|
|
{model}
|
|
</Badge>
|
|
))}
|
|
{tool.models.length > 3 && (
|
|
<Badge variant="outline" className="text-xs">
|
|
+{tool.models.length - 3} more
|
|
</Badge>
|
|
)}
|
|
</div>
|
|
)}
|
|
|
|
{/* Stats */}
|
|
<div className="flex items-center gap-4 text-xs text-muted-foreground">
|
|
<div className="flex items-center gap-1">
|
|
<FileText className="w-3 h-3" />
|
|
{tool.file_count} files
|
|
</div>
|
|
<div className="flex items-center gap-1">
|
|
<Code className="w-3 h-3" />
|
|
{formatNumber(tool.total_lines)} lines
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
|
|
<CardFooter className="flex gap-2">
|
|
<Button asChild className="flex-1">
|
|
<Link href={`/tool/${slugify(tool.directory)}`}>
|
|
View Details
|
|
</Link>
|
|
</Button>
|
|
{tool.website && (
|
|
<Button asChild variant="outline" size="icon">
|
|
<a href={tool.website} target="_blank" rel="noopener noreferrer">
|
|
<ExternalLink className="w-4 h-4" />
|
|
</a>
|
|
</Button>
|
|
)}
|
|
</CardFooter>
|
|
</Card>
|
|
)
|
|
}
|