add ai pages, cc usage
This commit is contained in:
		
							parent
							
								
									2f5b86111c
								
							
						
					
					
						commit
						7a9f907661
					
				
					 11 changed files with 1204 additions and 1 deletions
				
			
		
							
								
								
									
										73
									
								
								app/ai/components/AIStack.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								app/ai/components/AIStack.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,73 @@ | |||
| import { Bot } from 'lucide-react' | ||||
| import Link from '@/components/objects/Link' | ||||
| import type { AITool } from '../types' | ||||
| 
 | ||||
| interface AIStackProps { | ||||
|   tools: AITool[] | ||||
| } | ||||
| 
 | ||||
| export default function AIStack({ tools }: AIStackProps) { | ||||
|   const getStatusColor = (status: string) => { | ||||
|     switch(status) { | ||||
|       case 'primary': return 'text-green-400 border-green-400 bg-green-400/10' | ||||
|       case 'active': return 'text-blue-400 border-blue-400 bg-blue-400/10' | ||||
|       case 'occasional': return 'text-yellow-400 border-yellow-400 bg-yellow-400/10' | ||||
|       default: return 'text-gray-400 border-gray-400' | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const getStatusLabel = (status: string) => { | ||||
|     switch(status) { | ||||
|       case 'primary': return 'Primary' | ||||
|       case 'active': return 'Active Use' | ||||
|       case 'occasional': return 'Occasional Use' | ||||
|       default: return status | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     <section className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||
|       <h2 className="text-2xl font-semibold mb-6 text-gray-200 flex items-center gap-2"> | ||||
|         <Bot size={24} /> | ||||
|         My AI Stack | ||||
|       </h2> | ||||
|       <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4"> | ||||
|         {tools.map((tool, index) => ( | ||||
|           <div key={index} className="p-4 border border-gray-700 rounded-lg hover:border-gray-500 transition-all duration-300 flex flex-col"> | ||||
|             <div className="flex items-start justify-between mb-3 flex-1"> | ||||
|               <div className="flex items-center gap-3"> | ||||
|                 {tool.icon && <tool.icon className="text-2xl text-gray-300" />} | ||||
|                 {tool.svg && ( | ||||
|                   <div className="w-6 h-6 text-gray-300 fill-current"> | ||||
|                     {tool.svg} | ||||
|                   </div> | ||||
|                 )} | ||||
|                 <div> | ||||
|                   <h3 className="font-semibold text-gray-200">{tool.name}</h3> | ||||
|                   <p className="text-sm text-gray-400">{tool.description}</p> | ||||
|                 </div> | ||||
|               </div> | ||||
|             </div> | ||||
|             <div className="flex items-center justify-between mt-auto"> | ||||
|               <span className={`text-xs px-2 py-1 rounded-full border ${getStatusColor(tool.status)}`}> | ||||
|                 {getStatusLabel(tool.status)} | ||||
|               </span> | ||||
|               <span className="flex flex-row items-center gap-4"> | ||||
|                 {tool.link && ( | ||||
|                   <Link href={tool.link} className="text-blue-400 hover:text-blue-300 text-sm"> | ||||
|                     View → | ||||
|                   </Link> | ||||
|                 )} | ||||
|                 {tool.usage && ( | ||||
|                   <Link href={tool.usage} className="text-blue-400 hover:text-blue-300 text-sm"> | ||||
|                     Usage → | ||||
|                   </Link> | ||||
|                 )} | ||||
|               </span> | ||||
|             </div> | ||||
|           </div> | ||||
|         ))} | ||||
|       </div> | ||||
|     </section> | ||||
|   ) | ||||
| } | ||||
							
								
								
									
										39
									
								
								app/ai/components/FavoriteModels.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								app/ai/components/FavoriteModels.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,39 @@ | |||
| import { Star } from 'lucide-react' | ||||
| import type { FavoriteModel } from '../types' | ||||
| 
 | ||||
| interface FavoriteModelsProps { | ||||
|   models: FavoriteModel[] | ||||
| } | ||||
| 
 | ||||
| export default function FavoriteModels({ models }: FavoriteModelsProps) { | ||||
|   return ( | ||||
|     <section className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||
|       <h2 className="text-2xl font-semibold mb-6 text-gray-200 flex items-center gap-2"> | ||||
|         <Star size={24} /> | ||||
|         Favorite Models | ||||
|       </h2> | ||||
|       <div className="space-y-4"> | ||||
|         {models.map((model, index) => ( | ||||
|           <div key={index} className="p-4 bg-gray-800/50 rounded-lg"> | ||||
|             <div className="flex justify-between items-start mb-2"> | ||||
|               <div> | ||||
|                 <h3 className="font-semibold text-gray-200">{model.name}</h3> | ||||
|                 <p className="text-sm text-gray-400">{model.provider}</p> | ||||
|               </div> | ||||
|               <div className="flex gap-1"> | ||||
|                 {[...Array(5)].map((_, i) => ( | ||||
|                   <Star | ||||
|                     key={i} | ||||
|                     size={14} | ||||
|                     className={i < model.rating ? 'fill-yellow-400 text-yellow-400' : 'text-gray-600'} | ||||
|                   /> | ||||
|                 ))} | ||||
|               </div> | ||||
|             </div> | ||||
|             <p className="text-sm text-gray-300">{model.review}</p> | ||||
|           </div> | ||||
|         ))} | ||||
|       </div> | ||||
|     </section> | ||||
|   ) | ||||
| } | ||||
							
								
								
									
										54
									
								
								app/ai/components/ToolReviews.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								app/ai/components/ToolReviews.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,54 @@ | |||
| import { MessageSquare, Star } from 'lucide-react' | ||||
| import type { AIReview } from '../types' | ||||
| 
 | ||||
| interface ToolReviewsProps { | ||||
|   reviews: AIReview[] | ||||
| } | ||||
| 
 | ||||
| export default function ToolReviews({ reviews }: ToolReviewsProps) { | ||||
|   return ( | ||||
|     <section className="p-8 border-2 border-gray-700 rounded-lg hover:border-gray-600 transition-colors duration-300"> | ||||
|       <h2 className="text-2xl font-semibold mb-6 text-gray-200 flex items-center gap-2"> | ||||
|         <MessageSquare size={24} /> | ||||
|         Tool Reviews | ||||
|       </h2> | ||||
|       <div className="space-y-4"> | ||||
|         {reviews.map((review, index) => ( | ||||
|           <div key={index} className="p-4 bg-gray-800/50 rounded-lg"> | ||||
|             <div className="flex justify-between items-center mb-3"> | ||||
|               <h3 className="font-semibold text-gray-200">{review.tool}</h3> | ||||
|               <div className="flex gap-1"> | ||||
|                 {[...Array(5)].map((_, i) => ( | ||||
|                   <Star | ||||
|                     key={i} | ||||
|                     size={14} | ||||
|                     className={i < review.rating ? 'fill-yellow-400 text-yellow-400' : 'text-gray-600'} | ||||
|                   /> | ||||
|                 ))} | ||||
|               </div> | ||||
|             </div> | ||||
|             <div className="grid grid-cols-2 gap-2 mb-2 text-sm"> | ||||
|               <div> | ||||
|                 <p className="text-green-400 font-medium mb-1">Pros:</p> | ||||
|                 <ul className="text-gray-300 space-y-1"> | ||||
|                   {review.pros.map((pro, i) => ( | ||||
|                     <li key={i} className="text-xs">• {pro}</li> | ||||
|                   ))} | ||||
|                 </ul> | ||||
|               </div> | ||||
|               <div> | ||||
|                 <p className="text-red-400 font-medium mb-1">Cons:</p> | ||||
|                 <ul className="text-gray-300 space-y-1"> | ||||
|                   {review.cons.map((con, i) => ( | ||||
|                     <li key={i} className="text-xs">• {con}</li> | ||||
|                   ))} | ||||
|                 </ul> | ||||
|               </div> | ||||
|             </div> | ||||
|             <p className="text-sm text-blue-400 font-medium">{review.verdict}</p> | ||||
|           </div> | ||||
|         ))} | ||||
|       </div> | ||||
|     </section> | ||||
|   ) | ||||
| } | ||||
							
								
								
									
										49
									
								
								app/ai/components/TopPick.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								app/ai/components/TopPick.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,49 @@ | |||
| import { Trophy, ChevronRight } from 'lucide-react' | ||||
| import { SiClaude } from 'react-icons/si' | ||||
| import Link from '@/components/objects/Link' | ||||
| 
 | ||||
| export default function TopPick() { | ||||
|   return ( | ||||
|     <div className="px-4 mb-4"> | ||||
|       <h2 className="text-4xl font-semibold mb-6 text-gray-200 flex items-center gap-2"> | ||||
|         <Trophy size={32} /> | ||||
|         Top Pick of 2025 | ||||
|       </h2> | ||||
|       <div className="p-8 border-2 border-[#c15f3c] rounded-lg bg-orange-500/5"> | ||||
|         <div className="grid md:grid-cols-2 gap-6"> | ||||
|           <div className="flex items-center gap-4"> | ||||
|             <SiClaude className="text-6xl text-[#c15f3c]" /> | ||||
|             <div> | ||||
|               <h3 className="text-3xl font-bold text-gray-100">Claude</h3> | ||||
|               <p className="text-gray-400">by Anthropic</p> | ||||
|               <div className="flex items-center gap-2 mt-2"> | ||||
|                 <Link href="/ai/claude" className="text-blue-400 hover:text-blue-300 flex items-center gap-1"> | ||||
|                   View My Usage <ChevronRight size={16} /> | ||||
|                 </Link> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|           <div className="space-y-2"> | ||||
|             <p className="text-gray-300"> | ||||
|               Claude has become my go-to AI assistant for coding, writing, and learning very quickly. | ||||
|               I believe their Max 5x ($100/mo) is the best value for budget-conscious consumers like myself. | ||||
|             </p> | ||||
|             <div className='flex flex-col items-center gap-y-6 sm:flex-row sm:justify-between'> | ||||
|               <div className="flex gap-2 flex-wrap"> | ||||
|                 <span className="px-2 py-1 bg-gray-700 rounded text-xs text-gray-300">Claude Code</span> | ||||
|                 <span className="px-2 py-1 bg-gray-700 rounded text-xs text-gray-300">Best Tool Calling</span> | ||||
|                 <span className="px-2 py-1 bg-gray-700 rounded text-xs text-gray-300">High Value in Max Plan</span> | ||||
|                 <span className="px-2 py-1 bg-gray-700 rounded text-xs text-gray-300">Quite Fast Interface</span> | ||||
|               </div> | ||||
|               <div className="flex items-center justify-end"> | ||||
|                 <span className="px-3 py-1 bg-[#c15f3c]/20 text-[#c15f3c] rounded-full text-sm font-medium"> | ||||
|                   Top Overall Pick | ||||
|                 </span> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   ) | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue