Files
blog-nextjs/components/post-list-item.tsx
gbanyan 1b495d2d2d Remove next-view-transitions and use native View Transition API
- Remove external next-view-transitions dependency
- Use Next.js 16 native navigation and Safari 18+ native View Transition API
- Add ViewTransitionProvider for minimal wrapping with Safari 18+ detection
- Updated all Link imports from external package to next/link
- Removed link-wrapper.tsx and view-transitions-wrapper.tsx

This resolves Safari compatibility issues while maintaining transitions on modern browsers.
2026-03-14 23:00:21 +08:00

68 lines
3.1 KiB
TypeScript

import Link from 'next/link';
import Image from 'next/image';
import { Post } from 'contentlayer2/generated';
import { siteConfig } from '@/lib/config';
import { FiCalendar, FiTag } from 'react-icons/fi';
import { MetaItem } from './meta-item';
interface Props {
post: Post;
priority?: boolean;
}
export function PostListItem({ post, priority = false }: Props) {
const cover =
post.feature_image && post.feature_image.startsWith('../assets')
? post.feature_image.replace('../assets', '/assets')
: undefined;
const excerpt =
post.description || post.custom_excerpt || post.body?.raw?.slice(0, 120);
return (
<article className="motion-card group relative flex gap-4 rounded-2xl border border-white/40 bg-white/60 p-5 shadow-lg backdrop-blur-md transition-all hover:scale-[1.01] hover:shadow-xl dark:border-white/10 dark:bg-slate-900/60">
<div className="pointer-events-none absolute inset-x-0 top-0 h-0.5 origin-left scale-x-0 bg-gradient-to-r from-[rgba(124,58,237,0.9)] via-[rgba(167,139,250,0.9)] to-[rgba(14,165,233,0.8)] opacity-80 transition-transform duration-300 ease-out group-hover:scale-x-100" />
{cover && (
<div className="relative flex h-24 w-24 flex-none overflow-hidden rounded-md bg-slate-100 dark:bg-slate-800 sm:h-auto sm:w-40">
<Image
src={cover}
alt={post.title}
width={320}
height={240}
sizes="(max-width: 640px) 96px, 160px"
loading={priority ? undefined : 'lazy'}
priority={priority}
placeholder="blur"
blurDataURL="data:image/jpeg;base64,/9j/4AAQSkZJRgABAQAAAQABAAD/2wBDAAYEBQYFBAYGBQYHBwYIChAKCgkJChQODwwQFxQYGBcUFhYaHSUfGhsjHBYWICwgIyYnKSopGR8tMC0oMCUoKSj/2wBDAQcHBwoIChMKChMoGhYaKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCj/wAARCAAIAAoDASIAAhEBAxEB/8QAFQABAQAAAAAAAAAAAAAAAAAAAAv/xAAhEAACAQMDBQAAAAAAAAAAAAABAgMABAUGIWGRkqGx0f/EABUBAQEAAAAAAAAAAAAAAAAAAAMF/8QAGhEAAgIDAAAAAAAAAAAAAAAAAAECEgMRkf/aAAwDAQACEQMRAD8AltJagyeH0AthI5xdrLcNM91BF5pX2HaH9bcfaSXWGaRmknyJckliyjqTzSlT54b6bk+h0R//2Q=="
className="h-full w-full object-cover transition-transform duration-300 ease-out group-hover:scale-105"
/>
</div>
)}
<div className="flex-1 space-y-1.5">
<div className="flex flex-wrap gap-3 text-xs">
{post.published_at && (
<MetaItem icon={FiCalendar}>
{new Date(post.published_at).toLocaleDateString(
siteConfig.defaultLocale
)}
</MetaItem>
)}
{post.tags && post.tags.length > 0 && (
<MetaItem icon={FiTag} tone="muted">
{post.tags.slice(0, 3).join(', ')}
</MetaItem>
)}
</div>
<h2 className="type-body font-semibold leading-snug hover:text-accent sm:type-title">
<Link href={post.url}>{post.title}</Link>
</h2>
{excerpt && (
<p className="line-clamp-2 text-sm text-slate-600 dark:text-slate-300">
{excerpt}
</p>
)}
</div>
</article>
);
}