feat(travel): MasonryGrid 컴포넌트 — CSS columns Masonry + 무한스크롤

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-24 01:19:41 +09:00
parent 5efb9525d5
commit 4655e9ab3b
2 changed files with 255 additions and 0 deletions

View File

@@ -0,0 +1,138 @@
/* ── MasonryGrid ── */
.masonry-grid {
column-count: 4;
column-gap: 8px;
}
/* item */
.masonry-item {
break-inside: avoid;
margin-bottom: 8px;
position: relative;
border-radius: 4px;
overflow: hidden;
cursor: zoom-in;
/* scroll-reveal initial state */
opacity: 0;
transform: translateY(20px);
transition: opacity 0.45s ease, transform 0.45s ease;
}
.masonry-item--revealed {
opacity: 1;
transform: translateY(0);
}
.masonry-item__img {
display: block;
width: 100%;
height: auto;
transition: filter 0.25s ease;
}
.masonry-item:hover .masonry-item__img {
filter: brightness(1.08);
}
/* hover overlay */
.masonry-item__overlay {
position: absolute;
inset: 0;
display: flex;
align-items: flex-end;
padding: 8px 10px;
background: linear-gradient(transparent 60%, rgba(15, 12, 9, 0.7));
opacity: 0;
transition: opacity 0.25s ease;
pointer-events: none;
}
.masonry-item:hover .masonry-item__overlay {
opacity: 1;
}
.masonry-item__label {
font: 11px var(--tv-mono);
color: var(--tv-text);
letter-spacing: 0.04em;
}
/* sentinel */
.masonry-sentinel {
height: 1px;
column-span: all;
}
/* loading dots */
.masonry-loading {
column-span: all;
display: flex;
justify-content: center;
gap: 6px;
padding: 24px 0;
}
.masonry-dot {
width: 6px;
height: 6px;
border-radius: 50%;
background: var(--tv-muted);
animation: masonry-pulse 1.2s ease-in-out infinite;
}
.masonry-dot:nth-child(2) {
animation-delay: 0.15s;
}
.masonry-dot:nth-child(3) {
animation-delay: 0.3s;
}
@keyframes masonry-pulse {
0%, 80%, 100% { opacity: 0.25; transform: scale(0.8); }
40% { opacity: 1; transform: scale(1); }
}
/* end message */
.masonry-end {
column-span: all;
text-align: center;
font: 11px var(--tv-mono);
letter-spacing: 0.12em;
color: var(--tv-dim);
padding: 32px 0 16px;
margin: 0;
}
/* responsive */
@media (max-width: 1024px) {
.masonry-grid {
column-count: 3;
}
}
@media (max-width: 768px) {
.masonry-grid {
column-count: 2;
}
}
/* reduced motion */
@media (prefers-reduced-motion: reduce) {
.masonry-item {
opacity: 1;
transform: none;
transition: none;
}
.masonry-item__img,
.masonry-item__overlay {
transition: none;
}
.masonry-dot {
animation: none;
}
}