You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
160 lines
4.8 KiB
160 lines
4.8 KiB
<template>
|
|
<div class="card" @click="$emit('select', item.id)">
|
|
<div class="card-thumb" :class="typeBgClass">
|
|
<div class="card-thumb-inner">{{ typeEmoji }}</div>
|
|
<div class="type-badge" :class="typeBadgeClass">{{ typeLabel }}</div>
|
|
<div class="fav-btn" @click.stop="$emit('toggleStar', item.id)">
|
|
<svg width="13" height="13" viewBox="0 0 24 24" :fill="item.starred ? 'var(--accent)' : 'none'" :stroke="item.starred ? 'var(--accent)' : 'rgba(255,255,255,0.6)'" stroke-width="2">
|
|
<polygon points="12 2 15.09 8.26 22 9.27 17 14.14 18.18 21.02 12 17.77 5.82 21.02 7 14.14 2 9.27 8.91 8.26 12 2" />
|
|
</svg>
|
|
</div>
|
|
</div>
|
|
<div class="card-body">
|
|
<div class="card-source" v-if="item.sourceHost">
|
|
<div class="card-source-dot" :style="{ background: typeAccent + '22', color: typeAccent }">●</div>
|
|
{{ item.sourceHost }}
|
|
</div>
|
|
<div class="card-title-text">{{ item.title }}</div>
|
|
<div class="card-desc" v-if="item.description">{{ item.description }}</div>
|
|
<div class="card-meta">
|
|
<div class="card-tags">
|
|
<TagBadge v-for="t in item.tags?.slice(0, 2)" :key="t.id || t.name" :name="t.name" :color="t.color" />
|
|
</div>
|
|
<div class="card-date">{{ formatDate(item.createdAt) }}</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
const props = defineProps<{ item: any }>();
|
|
defineEmits<{ select: [id: number]; toggleStar: [id: number] }>();
|
|
|
|
const typeMap: Record<string, any> = {
|
|
web: { label: '网页', emoji: '🌐', bg: 'bg-web', badge: 'badge-web', accent: 'var(--blue)' },
|
|
text: { label: '文本', emoji: '📝', bg: 'bg-text', badge: 'badge-text', accent: 'var(--purple)' },
|
|
image: { label: '图片', emoji: '📷', bg: 'bg-img', badge: 'badge-img', accent: 'var(--teal)' },
|
|
video: { label: '视频', emoji: '▶️', bg: 'bg-vid', badge: 'badge-vid', accent: 'var(--red)' },
|
|
file: { label: '文档', emoji: '📄', bg: 'bg-doc', badge: 'badge-doc', accent: 'var(--accent)' },
|
|
};
|
|
|
|
const typeLabel = computed(() => typeMap[props.item.type]?.label || props.item.type);
|
|
const typeEmoji = computed(() => typeMap[props.item.type]?.emoji || '📌');
|
|
const typeBgClass = computed(() => typeMap[props.item.type]?.bg || 'bg-web');
|
|
const typeBadgeClass = computed(() => typeMap[props.item.type]?.badge || 'badge-web');
|
|
const typeAccent = computed(() => typeMap[props.item.type]?.accent || 'var(--blue)');
|
|
|
|
function formatDate(ts: number) {
|
|
if (!ts) return '';
|
|
const d = new Date(ts);
|
|
return `${String(d.getMonth() + 1).padStart(2, '0')}-${String(d.getDate()).padStart(2, '0')}`;
|
|
}
|
|
</script>
|
|
|
|
<style scoped>
|
|
.card {
|
|
background: var(--bg2);
|
|
border: 1px solid var(--border);
|
|
border-radius: var(--radius-lg);
|
|
overflow: hidden;
|
|
cursor: pointer;
|
|
transition: border-color 0.2s, transform 0.15s, box-shadow 0.2s;
|
|
animation: fadeUp 0.3s ease both;
|
|
}
|
|
.card:hover {
|
|
border-color: var(--border2);
|
|
transform: translateY(-2px);
|
|
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.3);
|
|
}
|
|
@keyframes fadeUp {
|
|
from { opacity: 0; transform: translateY(12px); }
|
|
to { opacity: 1; transform: translateY(0); }
|
|
}
|
|
.card-thumb {
|
|
height: 120px;
|
|
position: relative;
|
|
overflow: hidden;
|
|
}
|
|
.card-thumb-inner {
|
|
width: 100%;
|
|
height: 100%;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
font-size: 32px;
|
|
}
|
|
.type-badge {
|
|
position: absolute;
|
|
top: 8px;
|
|
left: 8px;
|
|
font-size: 10px;
|
|
font-weight: 500;
|
|
padding: 2px 7px;
|
|
border-radius: 4px;
|
|
backdrop-filter: blur(8px);
|
|
}
|
|
.fav-btn {
|
|
position: absolute;
|
|
top: 8px;
|
|
right: 8px;
|
|
width: 26px;
|
|
height: 26px;
|
|
border-radius: 6px;
|
|
background: rgba(0, 0, 0, 0.4);
|
|
backdrop-filter: blur(8px);
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
opacity: 0;
|
|
transition: opacity 0.15s;
|
|
}
|
|
.card:hover .fav-btn { opacity: 1; }
|
|
.card-body { padding: 12px 14px 14px; }
|
|
.card-title-text {
|
|
font-size: 14px;
|
|
font-weight: 500;
|
|
color: var(--text);
|
|
line-height: 1.4;
|
|
margin-bottom: 6px;
|
|
display: -webkit-box;
|
|
-webkit-line-clamp: 2;
|
|
-webkit-box-orient: vertical;
|
|
overflow: hidden;
|
|
}
|
|
.card-desc {
|
|
font-size: 12px;
|
|
color: var(--text2);
|
|
line-height: 1.5;
|
|
display: -webkit-box;
|
|
-webkit-line-clamp: 2;
|
|
-webkit-box-orient: vertical;
|
|
overflow: hidden;
|
|
margin-bottom: 10px;
|
|
}
|
|
.card-meta {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
gap: 8px;
|
|
}
|
|
.card-tags { display: flex; gap: 4px; flex-wrap: wrap; }
|
|
.card-date { font-size: 11px; color: var(--text3); font-family: var(--font-mono); white-space: nowrap; }
|
|
.card-source {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 4px;
|
|
font-size: 11px;
|
|
color: var(--text3);
|
|
margin-bottom: 8px;
|
|
}
|
|
.card-source-dot {
|
|
width: 12px;
|
|
height: 12px;
|
|
border-radius: 3px;
|
|
background: var(--bg4);
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
font-size: 8px;
|
|
}
|
|
</style>
|
|
|