<template>
<div class="card-grid">
<div v-if="data.title" class="card-grid-title">{{ data.title }}</div>
<div class="card-grid-items">
<div
v-for="card in visibleCards"
:key="card.id"
class="card-grid-item"
tabindex="0"
@click="openCard(card)"
@keydown.enter="openCard(card)"
>
<img v-if="card.image" :src="card.image" class="card-image" alt="" loading="lazy" />
<div class="card-no-image" v-else>
<i class="ph ph-image"></i>
</div>
<div class="card-content">
<div class="card-title">{{ card.title }}</div>
<div v-if="card.subtitle" class="card-subtitle">{{ card.subtitle }}</div>
<div v-if="card.meta?.length" class="card-meta">
<span v-for="(m, i) in card.meta" :key="i" class="card-meta-item">
<span class="card-meta-label">{{ m.label }}</span>
<span class="card-meta-value">{{ m.value }}</span>
</span>
</div>
<div v-if="card.description" class="card-description">{{ card.description }}</div>
</div>
</div>
</div>
<div v-if="hasMore" class="card-grid-more">
<span class="card-grid-more-text">+{{ remainingCount }} more</span>
</div>
<Teleport to="body">
<div
v-if="activeCard"
class="card-modal-overlay"
@click.self="closeCard"
>
<div class="card-modal" role="dialog" aria-modal="true">
<button class="card-modal-close" @click="closeCard" aria-label="Close">
<i class="ph ph-x"></i>
</button>
<div class="card-modal-scroll">
<img v-if="activeCard.image" :src="activeCard.image" class="card-modal-image" alt="" />
<div class="card-modal-content">
<h2 class="card-modal-title">{{ activeCard.title }}</h2>
<div v-if="activeCard.subtitle" class="card-modal-subtitle">{{ activeCard.subtitle }}</div>
<div v-if="activeCard.meta?.length" class="card-modal-section">
<div class="card-modal-section-title">Overview</div>
<div class="card-modal-meta-grid">
<div v-for="(m, i) in activeCard.meta" :key="i" class="card-modal-meta-cell">
<div class="card-modal-meta-label">{{ m.label }}</div>
<div class="card-modal-meta-value">{{ m.value }}</div>
</div>
</div>
</div>
<div v-if="activeCard.description" class="card-modal-section">
<div class="card-modal-section-title">Description</div>
<p class="card-modal-text">{{ activeCard.description }}</p>
</div>
<div v-if="activeCard.details?.length" class="card-modal-section">
<div class="card-modal-section-title">Details</div>
<dl class="card-modal-details">
<div v-for="(d, i) in activeCard.details" :key="i" class="card-modal-detail-row">
<dt>{{ d.label }}</dt>
<dd>{{ d.value }}</dd>
</div>
</dl>
</div>
<div v-if="activeCard.actions?.length" class="card-modal-actions">
<a
v-for="(a, i) in activeCard.actions"
:key="i"
:href="a.url"
target="_blank"
rel="noopener noreferrer"
class="card-modal-action"
>
{{ a.label }}
</a>
</div>
</div>
</div>
</div>
</div>
</Teleport>
</div>
</template>
<script setup>
import { ref, computed } from 'vue'
const props = defineProps({
data: { type: Object, default: () => ({}) }
})
const DEFAULT_VISIBLE = 4
const activeCard = ref(null)
const cards = computed(() => Array.isArray(props.data.cards) ? props.data.cards : [])
const visibleCards = computed(() => cards.value.slice(0, DEFAULT_VISIBLE))
const hasMore = computed(() => cards.value.length > DEFAULT_VISIBLE)
const remainingCount = computed(() => cards.value.length - DEFAULT_VISIBLE)
function openCard(card) {
activeCard.value = card
}
function closeCard() {
activeCard.value = null
}
</script>
<style scoped>
.card-grid {
display: flex;
flex-direction: column;
gap: 12px;
}
.card-grid-title {
font-size: 1rem;
font-weight: 600;
color: var(--text, #cdd6f4);
}
.card-grid-items {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 12px;
}
@media (max-width: 640px) {
.card-grid-items {
grid-template-columns: 1fr;
}
}
.card-grid-item {
display: flex;
flex-direction: column;
background: var(--surface-elevated, #242438);
border: 1px solid var(--border, #2a2a3e);
border-radius: 8px;
overflow: hidden;
cursor: pointer;
transition: border-color 0.15s, transform 0.1s;
}
.card-grid-item:hover,
.card-grid-item:focus {
border-color: var(--accent, #4ec9b0);
outline: none;
}
.card-grid-item:active {
transform: scale(0.99);
}
.card-image {
width: 100%;
height: 140px;
object-fit: cover;
background: var(--surface, #1e1e2e);
}
.card-no-image {
width: 100%;
height: 140px;
display: flex;
align-items: center;
justify-content: center;
color: var(--text-muted, #6c7086);
font-size: 2rem;
background: var(--surface, #1e1e2e);
}
.card-content {
padding: 12px;
display: flex;
flex-direction: column;
gap: 8px;
flex: 1;
}
.card-title {
font-weight: 600;
font-size: 0.95rem;
color: var(--text, #cdd6f4);
line-height: 1.3;
}
.card-subtitle {
font-size: 0.8rem;
color: var(--text-muted, #6c7086);
}
.card-meta {
display: flex;
flex-wrap: wrap;
gap: 8px;
}
.card-meta-item {
display: flex;
gap: 4px;
font-size: 0.75rem;
}
.card-meta-label {
color: var(--text-muted, #6c7086);
}
.card-meta-value {
color: var(--text, #cdd6f4);
font-weight: 500;
}
.card-description {
font-size: 0.8rem;
color: var(--text, #cdd6f4);
line-height: 1.4;
opacity: 0.85;
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
overflow: hidden;
}
.card-grid-more {
text-align: center;
font-size: 0.8rem;
color: var(--text-muted, #6c7086);
}
.card-modal-overlay {
position: fixed;
inset: 0;
z-index: 1000;
background: rgba(0, 0, 0, 0.6);
display: flex;
align-items: center;
justify-content: center;
padding: 16px;
}
.card-modal {
position: relative;
width: 100%;
max-width: 560px;
max-height: 90vh;
background: var(--surface, #1e1e2e);
border: 1px solid var(--border, #2a2a3e);
border-radius: 12px;
overflow: hidden;
display: flex;
flex-direction: column;
}
.card-modal-close {
position: absolute;
top: 12px;
right: 12px;
width: 32px;
height: 32px;
display: flex;
align-items: center;
justify-content: center;
background: var(--surface-elevated, #242438);
border: 1px solid var(--border, #2a2a3e);
color: var(--text, #cdd6f4);
border-radius: 6px;
cursor: pointer;
z-index: 2;
}
.card-modal-close:hover {
border-color: var(--accent, #4ec9b0);
color: var(--accent, #4ec9b0);
}
.card-modal-scroll {
overflow-y: auto;
max-height: 90vh;
}
.card-modal-image {
width: 100%;
max-height: 320px;
object-fit: cover;
background: var(--surface, #1e1e2e);
}
.card-modal-content {
padding: 20px;
display: flex;
flex-direction: column;
gap: 18px;
}
.card-modal-title {
margin: 0;
font-size: 1.3rem;
font-weight: 600;
color: var(--text, #cdd6f4);
}
.card-modal-subtitle {
color: var(--text-muted, #6c7086);
font-size: 0.9rem;
}
.card-modal-section {
display: flex;
flex-direction: column;
gap: 10px;
}
.card-modal-section-title {
font-size: 0.75rem;
text-transform: uppercase;
letter-spacing: 0.05em;
color: var(--text-muted, #6c7086);
}
.card-modal-meta-grid {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 10px;
}
.card-modal-meta-cell {
background: var(--surface-elevated, #242438);
border: 1px solid var(--border, #2a2a3e);
padding: 10px;
border-radius: 6px;
}
.card-modal-meta-label {
font-size: 0.7rem;
color: var(--text-muted, #6c7086);
margin-bottom: 4px;
}
.card-modal-meta-value {
font-size: 0.9rem;
color: var(--text, #cdd6f4);
font-weight: 500;
}
.card-modal-text {
margin: 0;
line-height: 1.5;
color: var(--text, #cdd6f4);
}
.card-modal-details {
margin: 0;
display: grid;
gap: 8px;
}
.card-modal-detail-row {
display: flex;
justify-content: space-between;
gap: 12px;
padding: 8px 0;
border-bottom: 1px solid var(--border, #2a2a3e);
}
.card-modal-detail-row:last-child {
border-bottom: none;
}
.card-modal-detail-row dt {
color: var(--text-muted, #6c7086);
font-size: 0.85rem;
}
.card-modal-detail-row dd {
margin: 0;
color: var(--text, #cdd6f4);
font-size: 0.85rem;
text-align: right;
word-break: break-word;
}
.card-modal-actions {
display: flex;
gap: 10px;
flex-wrap: wrap;
}
.card-modal-action {
display: inline-flex;
align-items: center;
padding: 8px 14px;
background: var(--accent-muted, #313244);
color: var(--accent, #4ec9b0);
text-decoration: none;
border-radius: 6px;
font-size: 0.85rem;
font-weight: 500;
}
.card-modal-action:hover {
background: var(--border, #2a2a3e);
}
</style>