- Infinite loop carousels for Similar Artworks & Trending rails
- Mouse wheel horizontal scrolling on both carousels
- Author avatar shown on hover in RailCard (similar + trending)
- Removed "View" badge from RailCard hover overlay
- Added `id` to Meilisearch filterable attributes
- Auto-prepend Scout prefix in meilisearch:configure-index command
- Added author name + avatar to Similar Artworks API response
- Added avatar_url to ArtworkListResource author object
- Added direct /art/{id}/{slug} URL to ArtworkListResource
- Fixed race condition: Similar Artworks no longer briefly shows trending items
- Fixed user_profiles eager load (user_id primary key, not id)
- Bumped /api/art/{id}/similar rate limit to 300/min
- Removed decorative heart icons from tag pills
- Moved ReactionBar under artwork description
71 lines
3.2 KiB
JavaScript
71 lines
3.2 KiB
JavaScript
import React from 'react'
|
|
|
|
const FALLBACK = 'https://files.skinbase.org/default/missing_md.webp'
|
|
const AVATAR_FALLBACK = 'https://files.skinbase.org/default/avatar_default.webp'
|
|
|
|
function ArtCard({ item }) {
|
|
const username = item.author_username ? `@${item.author_username}` : null
|
|
return (
|
|
<article>
|
|
<a
|
|
href={item.url}
|
|
className="group relative block overflow-hidden rounded-2xl ring-1 ring-white/5 bg-black/20 shadow-lg shadow-black/40 transition-all duration-200 ease-out hover:-translate-y-0.5 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-sky-300/70"
|
|
>
|
|
<div className="relative aspect-video overflow-hidden bg-neutral-900">
|
|
<div className="absolute inset-0 bg-gradient-to-br from-white/10 via-white/5 to-transparent pointer-events-none z-10" />
|
|
<img
|
|
src={item.thumb || FALLBACK}
|
|
alt={item.title}
|
|
className="h-full w-full object-cover transition-[transform,filter] duration-300 ease-out group-hover:scale-[1.04]"
|
|
loading="lazy"
|
|
decoding="async"
|
|
onError={(e) => { e.currentTarget.src = FALLBACK }}
|
|
/>
|
|
<div className="pointer-events-none absolute inset-x-0 bottom-0 z-20 bg-gradient-to-t from-black/80 via-black/40 to-transparent p-3 opacity-100 transition-opacity duration-200 md:opacity-0 md:group-hover:opacity-100">
|
|
<div className="truncate text-sm font-semibold text-white">{item.title}</div>
|
|
<div className="mt-1 flex items-center gap-2 text-xs text-white/80">
|
|
<img
|
|
src={item.author_avatar || AVATAR_FALLBACK}
|
|
alt={item.author}
|
|
className="w-5 h-5 rounded-full object-cover shrink-0"
|
|
onError={(e) => { e.currentTarget.src = AVATAR_FALLBACK }}
|
|
/>
|
|
<span className="truncate">{item.author}</span>
|
|
{username && <span className="text-white/50 shrink-0">{username}</span>}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<span className="sr-only">{item.title} by {item.author}</span>
|
|
</a>
|
|
</article>
|
|
)
|
|
}
|
|
|
|
/**
|
|
* Personalized trending: artworks matching user's top tags, sorted by trending score.
|
|
* Label and browse link adapt to the user's first top tag.
|
|
*/
|
|
export default function HomeTrendingForYou({ items, preferences }) {
|
|
if (!Array.isArray(items) || items.length === 0) return null
|
|
|
|
const topTag = preferences?.top_tags?.[0]
|
|
const heading = topTag ? `🎯 Trending in #${topTag}` : '🎯 Trending For You'
|
|
const link = topTag ? `/browse?tags=${encodeURIComponent(topTag)}&sort=trending` : '/discover/trending'
|
|
|
|
return (
|
|
<section className="mt-14 px-4 sm:px-6 lg:px-8">
|
|
<div className="mb-5 flex items-center justify-between">
|
|
<h2 className="text-xl font-bold text-white">{heading}</h2>
|
|
<a href={link} className="text-sm text-nova-300 hover:text-white transition">
|
|
See all →
|
|
</a>
|
|
</div>
|
|
<div className="grid grid-cols-2 gap-4 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5">
|
|
{items.slice(0, Math.floor(items.length / 5) * 5 || items.length).map((item) => (
|
|
<ArtCard key={item.id} item={item} />
|
|
))}
|
|
</div>
|
|
</section>
|
|
)
|
|
}
|