feat: sessão #3 — lib (db/auth/email/validations), API routes, NextAuth v5, middleware, páginas account/shelters/shelter-dashboard, Prisma v7 fix
This commit is contained in:
161
components/animals/AnimalCard.tsx
Normal file
161
components/animals/AnimalCard.tsx
Normal file
@@ -0,0 +1,161 @@
|
||||
'use client';
|
||||
|
||||
import Link from 'next/link';
|
||||
import Image from 'next/image';
|
||||
import { MapPin, ChevronRight, AlertTriangle } from 'lucide-react';
|
||||
import { Animal, formatAge } from '@/lib/mock-data';
|
||||
|
||||
interface AnimalCardProps {
|
||||
animal: Animal;
|
||||
index?: number;
|
||||
}
|
||||
|
||||
export default function AnimalCard({ animal, index = 0 }: AnimalCardProps) {
|
||||
const ageLabel = formatAge(animal.ageMonths);
|
||||
const sexSymbol = animal.sex === 'MALE' ? '♂' : '♀';
|
||||
const sterilizedLabel = animal.sterilized ? '· Esterilizado ✓' : '';
|
||||
const subline = `${animal.breed} · ${ageLabel} ${sterilizedLabel}`.trim();
|
||||
|
||||
return (
|
||||
<article
|
||||
className="animal-card"
|
||||
style={{ animationDelay: `${index * 80}ms` }}
|
||||
aria-label={`${animal.name}, ${animal.breed}, ${ageLabel}, disponível no ${animal.shelter.name}`}
|
||||
>
|
||||
{/* Fotografia */}
|
||||
<div style={{ position: 'relative', aspectRatio: '3/2', overflow: 'hidden' }}>
|
||||
<Image
|
||||
src={animal.photos[0]}
|
||||
alt={`${animal.name} — ${animal.breed} ${animal.sex === 'MALE' ? 'macho' : 'fêmea'}, ${ageLabel}`}
|
||||
fill
|
||||
sizes="(max-width: 640px) 100vw, (max-width: 1024px) 50vw, 33vw"
|
||||
style={{
|
||||
objectFit: 'cover',
|
||||
transition: 'transform 400ms ease',
|
||||
}}
|
||||
onMouseEnter={e => { (e.currentTarget as HTMLElement).style.transform = 'scale(1.04)'; }}
|
||||
onMouseLeave={e => { (e.currentTarget as HTMLElement).style.transform = 'scale(1)'; }}
|
||||
/>
|
||||
|
||||
{/* Gradiente base da foto */}
|
||||
<div
|
||||
aria-hidden="true"
|
||||
style={{
|
||||
position: 'absolute',
|
||||
inset: 0,
|
||||
background: 'linear-gradient(to top, rgba(35,20,8,0.35) 0%, transparent 50%)',
|
||||
pointerEvents: 'none',
|
||||
}}
|
||||
/>
|
||||
|
||||
{/* Badge urgente — top right */}
|
||||
{animal.urgent && (
|
||||
<div style={{ position: 'absolute', top: '12px', right: '12px' }}>
|
||||
<span className="badge-urgent">
|
||||
<AlertTriangle size={9} strokeWidth={2.5} />
|
||||
Urgente
|
||||
</span>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Corpo do card */}
|
||||
<div
|
||||
style={{
|
||||
padding: '16px',
|
||||
display: 'flex',
|
||||
flexDirection: 'column',
|
||||
gap: '8px',
|
||||
flex: 1,
|
||||
}}
|
||||
>
|
||||
{/* Nome + sexo */}
|
||||
<div style={{ display: 'flex', alignItems: 'baseline', justifyContent: 'space-between', gap: '8px' }}>
|
||||
<h3
|
||||
style={{
|
||||
fontFamily: 'var(--font-display)',
|
||||
fontWeight: 700,
|
||||
fontSize: '20px',
|
||||
color: 'var(--soil)',
|
||||
lineHeight: 1.1,
|
||||
letterSpacing: '-0.01em',
|
||||
}}
|
||||
>
|
||||
{animal.name}
|
||||
</h3>
|
||||
<span
|
||||
style={{
|
||||
fontFamily: 'var(--font-display)',
|
||||
fontWeight: 700,
|
||||
fontSize: '16px',
|
||||
color: 'var(--soil-mid)',
|
||||
flexShrink: 0,
|
||||
}}
|
||||
aria-label={animal.sex === 'MALE' ? 'Macho' : 'Fêmea'}
|
||||
>
|
||||
{sexSymbol}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
{/* Raça · Idade · Esterilizado */}
|
||||
<p
|
||||
style={{
|
||||
fontFamily: 'var(--font-accent)',
|
||||
fontSize: '10px',
|
||||
fontWeight: 400,
|
||||
color: 'var(--soil-faint)',
|
||||
letterSpacing: '0.08em',
|
||||
textTransform: 'uppercase',
|
||||
lineHeight: 1.4,
|
||||
}}
|
||||
>
|
||||
{subline}
|
||||
</p>
|
||||
|
||||
{/* Localização */}
|
||||
<div
|
||||
style={{
|
||||
display: 'flex',
|
||||
alignItems: 'center',
|
||||
gap: '5px',
|
||||
fontFamily: 'var(--font-body)',
|
||||
fontStyle: 'italic',
|
||||
fontSize: '14px',
|
||||
color: 'var(--soil-mid)',
|
||||
marginTop: '2px',
|
||||
}}
|
||||
>
|
||||
<MapPin size={12} style={{ color: 'var(--terra)', flexShrink: 0 }} aria-hidden="true" />
|
||||
<span>{animal.shelter.name}, {animal.shelter.district}</span>
|
||||
</div>
|
||||
|
||||
{/* Spacer */}
|
||||
<div style={{ flex: 1 }} />
|
||||
|
||||
{/* Separador */}
|
||||
<hr style={{ border: 'none', borderTop: '1px solid var(--parchment)', margin: '4px 0' }} />
|
||||
|
||||
{/* Acções */}
|
||||
<div style={{ display: 'flex', gap: '8px' }}>
|
||||
<Link
|
||||
href={`/main/animals/${animal.id}`}
|
||||
className="btn btn-secondary"
|
||||
style={{ flex: 1, justifyContent: 'center', padding: '10px 12px', fontSize: '11px', minHeight: '40px' }}
|
||||
aria-label={`Ver ficha completa de ${animal.name}`}
|
||||
>
|
||||
Ver mais
|
||||
</Link>
|
||||
<Link
|
||||
href={`/main/animals/${animal.id}#adoptar`}
|
||||
className="btn btn-primary"
|
||||
style={{ flex: 1, justifyContent: 'center', padding: '10px 12px', fontSize: '11px', minHeight: '40px', gap: '4px' }}
|
||||
aria-label={`Iniciar processo de adopção de ${animal.name}`}
|
||||
>
|
||||
Adoptar
|
||||
<ChevronRight size={12} strokeWidth={2.5} />
|
||||
</Link>
|
||||
</div>
|
||||
</div>
|
||||
</article>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user