feat(travel): AlbumCard 컴포넌트 — 대표사진 + 그라디언트 + 메타정보
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
116
src/pages/travel/AlbumCard.css
Normal file
116
src/pages/travel/AlbumCard.css
Normal file
@@ -0,0 +1,116 @@
|
||||
/* ── AlbumCard ── */
|
||||
|
||||
.album-card {
|
||||
position: relative;
|
||||
height: 240px;
|
||||
border-radius: 12px;
|
||||
border: 1px solid rgba(245, 230, 200, 0.08);
|
||||
overflow: hidden;
|
||||
cursor: pointer;
|
||||
outline: none;
|
||||
transition: transform 0.28s ease, box-shadow 0.28s ease;
|
||||
}
|
||||
|
||||
.album-card:hover,
|
||||
.album-card:focus-visible {
|
||||
transform: scale(1.03);
|
||||
box-shadow: 0 4px 24px color-mix(in srgb, var(--card-accent) 35%, transparent);
|
||||
}
|
||||
|
||||
/* cover image */
|
||||
.album-card__cover {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
object-fit: cover;
|
||||
transition: transform 0.35s ease;
|
||||
}
|
||||
|
||||
.album-card:hover .album-card__cover {
|
||||
transform: scale(1.06);
|
||||
}
|
||||
|
||||
/* gradient overlay */
|
||||
.album-card__gradient {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
background: linear-gradient(transparent 50%, rgba(15, 12, 9, 0.85));
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
/* meta */
|
||||
.album-card__meta {
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
padding: 16px;
|
||||
z-index: 2;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 4px;
|
||||
}
|
||||
|
||||
.album-card__region-badge {
|
||||
align-self: flex-start;
|
||||
font: 10px var(--tv-mono);
|
||||
letter-spacing: 0.08em;
|
||||
text-transform: uppercase;
|
||||
color: var(--card-accent);
|
||||
background: rgba(15, 12, 9, 0.6);
|
||||
padding: 2px 8px;
|
||||
border-radius: 4px;
|
||||
}
|
||||
|
||||
.album-card__name {
|
||||
margin: 0;
|
||||
font: 600 24px/1.15 var(--tv-serif);
|
||||
color: var(--tv-text);
|
||||
}
|
||||
|
||||
.album-card__count {
|
||||
font: 11px var(--tv-mono);
|
||||
letter-spacing: 0.06em;
|
||||
color: var(--tv-muted);
|
||||
background: rgba(15, 12, 9, 0.55);
|
||||
padding: 2px 8px;
|
||||
border-radius: 4px;
|
||||
align-self: flex-start;
|
||||
}
|
||||
|
||||
/* grid layout */
|
||||
.album-card-grid {
|
||||
display: grid;
|
||||
gap: 16px;
|
||||
grid-template-columns: repeat(3, 1fr);
|
||||
}
|
||||
|
||||
@media (max-width: 1024px) {
|
||||
.album-card-grid {
|
||||
grid-template-columns: repeat(2, 1fr);
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.album-card-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.album-card {
|
||||
height: 200px;
|
||||
}
|
||||
|
||||
.album-card__name {
|
||||
font-size: 18px;
|
||||
}
|
||||
}
|
||||
|
||||
/* reduced motion */
|
||||
@media (prefers-reduced-motion: reduce) {
|
||||
.album-card,
|
||||
.album-card__cover {
|
||||
transition: none;
|
||||
transform: none !important;
|
||||
}
|
||||
}
|
||||
55
src/pages/travel/AlbumCard.jsx
Normal file
55
src/pages/travel/AlbumCard.jsx
Normal file
@@ -0,0 +1,55 @@
|
||||
import React, { useRef, useCallback } from 'react';
|
||||
import { getRegionAccent } from './MiniMap';
|
||||
import './AlbumCard.css';
|
||||
|
||||
/* ─────────────────────────────────────────────
|
||||
AlbumCard — cover image + gradient + meta
|
||||
───────────────────────────────────────────── */
|
||||
export default function AlbumCard({ album, onClick }) {
|
||||
const cardRef = useRef(null);
|
||||
const accent = getRegionAccent(album.region || '');
|
||||
|
||||
const handleClick = useCallback(() => {
|
||||
if (!onClick) return;
|
||||
const rect = cardRef.current?.getBoundingClientRect();
|
||||
onClick(album, rect);
|
||||
}, [album, onClick]);
|
||||
|
||||
const handleKeyDown = useCallback(
|
||||
(e) => {
|
||||
if (e.key === 'Enter') handleClick();
|
||||
},
|
||||
[handleClick],
|
||||
);
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={cardRef}
|
||||
className="album-card"
|
||||
style={{ '--card-accent': accent }}
|
||||
role="button"
|
||||
tabIndex={0}
|
||||
onClick={handleClick}
|
||||
onKeyDown={handleKeyDown}
|
||||
>
|
||||
{/* cover */}
|
||||
<img
|
||||
className="album-card__cover"
|
||||
src={album.coverThumb}
|
||||
alt={album.name}
|
||||
loading="lazy"
|
||||
draggable={false}
|
||||
/>
|
||||
|
||||
{/* gradient overlay */}
|
||||
<div className="album-card__gradient" />
|
||||
|
||||
{/* meta */}
|
||||
<div className="album-card__meta">
|
||||
<span className="album-card__region-badge">{album.regionName}</span>
|
||||
<h3 className="album-card__name">{album.name}</h3>
|
||||
<span className="album-card__count">{album.photoCount} frames</span>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user