Files
TeyvatGuide/src/components/app/t-postcard.vue
BTMuli f98b1913f7 💄 微调
2025-09-18 18:39:48 +08:00

560 lines
15 KiB
Vue

<template>
<div
v-if="card"
:id="`post-card-${card.postId}`"
class="tpc-card"
:class="{ 'select-mode': props.selectMode }"
@click="trySelect()"
>
<div class="tpc-top">
<div class="tpc-cover" @click="toPost()">
<TMiImg :src="card.cover" alt="cover" :ori="true" v-if="card.cover !== ''" />
<img src="/source/UI/defaultCover.webp" alt="cover" v-else />
<div v-if="card.status" class="tpc-act">
<div class="tpc-status">{{ card.status?.label }}</div>
<div class="tpc-time">
<v-icon>mdi-clock-time-four-outline</v-icon>
<span>{{ card.subtitle }}</span>
</div>
</div>
<div
v-else-if="props.modelValue.post.images.length > 1"
class="tpc-image-cnt"
:title="`图片数:${props.modelValue.post.images.length}`"
>
<v-icon size="10">mdi-folder-multiple-image</v-icon>
<span>{{ props.modelValue.post.images.length }}</span>
</div>
</div>
<div class="tpc-title" :title="card.title" @click="shareCard()">{{ card.title }}</div>
</div>
<div class="tpc-mid" v-if="card.user !== null">
<TpAvatar
:data="card.user"
position="left"
:style="{ cursor: props.userClick ? 'pointer' : 'default' }"
@click="onUserClick()"
/>
</div>
<div class="tpc-bottom">
<div class="tpc-tags">
<div v-for="(reason, idx) in card.reasons" :key="idx" class="tpc-reason" title="推荐理由">
<v-icon size="10">mdi-lightbulb-on</v-icon>
<span>{{ reason.text }}</span>
</div>
<TpcTag
v-for="topic in card.topics"
:key="topic.id"
@click="toTopic(topic)"
:tag="topic.name"
/>
</div>
<div class="tpc-data" v-if="card.data !== null">
<div class="tpc-info-item" :title="`浏览数:${card.data.view}`">
<v-icon size="12">mdi-eye</v-icon>
<span>{{ card.data.view }}</span>
</div>
<div class="tpc-info-item" :title="`收藏数:${card.data.mark}`">
<v-icon size="12">mdi-star</v-icon>
<span>{{ card.data.mark }}</span>
</div>
<div class="tpc-info-item" :title="`回复数:${card.data.reply}`">
<v-icon size="12">mdi-comment</v-icon>
<span>{{ card.data.reply }}</span>
</div>
<div class="tpc-info-item" :title="`点赞数:${card.data.like}`">
<v-icon size="12">mdi-thumb-up</v-icon>
<span>{{ card.data.like }}</span>
</div>
<div class="tpc-info-item" :title="`转发数:${card.data.forward}`">
<v-icon size="12">mdi-share-variant</v-icon>
<span>{{ card.data.forward }}</span>
</div>
</div>
<div class="tpc-data">
<div class="tpc-info-item" :title="`创建时间: ${card.meta.create_time}`">
<v-icon size="12">mdi-calendar-clock</v-icon>
<span>{{ card.meta.create_time }}</span>
</div>
<div
v-if="card.meta.update_time"
class="tpc-info-item"
:title="`更新时间: ${card.meta.update_time}`"
>
<v-icon size="12">mdi-calendar-edit</v-icon>
<span>{{ card.meta.update_time }}</span>
</div>
</div>
</div>
<div
class="tpc-forum"
:style="{
background: str2Color(`${card.forum.id}${card.forum.name}`, -60),
}"
v-if="card.forum !== null && card.forum.name !== ''"
:title="`频道: ${card.forum.name}`"
@click="toForum(card.forum)"
>
<img v-if="card.forum.icon !== ''" :src="card.forum.icon" :alt="card.forum.name" />
<span>{{ card.forum.name }}</span>
</div>
<v-checkbox-btn
v-model="isSelected"
v-if="props.selectMode"
class="tpc-select"
@click.stop="trySelect()"
data-html2canvas-ignore
/>
<div
class="tpc-info-id"
:style="{
background: str2Color(`${props.modelValue.post.post_id}`, 0),
}"
v-else
>
{{ props.modelValue.post.post_id }}
</div>
</div>
</template>
<script lang="ts" setup>
import TMiImg from "@comp/app/t-mi-img.vue";
import showSnackbar from "@comp/func/snackbar.js";
import TpAvatar from "@comp/viewPost/tp-avatar.vue";
import TpcTag from "@comp/viewPost/tpc-tag.vue";
import { emit } from "@tauri-apps/api/event";
import { generateShareImg } from "@utils/TGShare.js";
import { createPost } from "@utils/TGWindow.js";
import { str2Color, timestampToDate } from "@utils/toolFunc.js";
import { computed, onMounted, ref, shallowRef, watch } from "vue";
import { useRoute, useRouter } from "vue-router";
type TPostCardProps = {
modelValue: TGApp.BBS.Post.FullData;
selectMode?: boolean;
userClick?: boolean;
};
type TPostCardEmits = {
(e: "onSelected", v: string): void;
(e: "onUserClick", u: TGApp.BBS.Post.User, g: number): void;
};
type RenderForum = { name: string; icon: string; id: number };
type RenderStatus = { stat: number; label: string; color: string };
type RenderData = { mark: number; forward: number; like: number; reply: number; view: number };
type RenderMeta = { create_time: string; update_time?: string };
export type RenderCard = {
title: string;
cover: string;
postId: number;
subtitle: string;
user: TGApp.BBS.Post.User | null;
forum: RenderForum | null;
data: RenderData | null;
meta: RenderMeta;
status?: RenderStatus;
topics: Array<TGApp.BBS.Post.Topic>;
reasons: Array<TGApp.BBS.Post.RecommendTags>;
};
const stats: Readonly<Array<RenderStatus>> = [
{ stat: 0, label: "未知", color: "var(--tgc-od-red)" },
{ stat: 1, label: "进行中", color: "var(--tgc-od-green)" },
{ stat: 2, label: "评选中", color: "var(--tgc-od-orange)" },
{ stat: 3, label: "已结束", color: "var(--tgc-od-white)" },
];
const route = useRoute();
const router = useRouter();
const props = withDefaults(defineProps<TPostCardProps>(), { selectMode: false });
const isSelected = ref<boolean>(false);
const emits = defineEmits<TPostCardEmits>();
const card = shallowRef<RenderCard>();
const cardBg = computed<string>(() => {
if (card.value && card.value.status) return card.value.status.color;
return "none";
});
onMounted(async () => (card.value = getPostCard(props.modelValue)));
watch(
() => props.modelValue,
async () => (card.value = getPostCard(props.modelValue)),
);
function trySelect(): void {
if (props.selectMode) emits("onSelected", props.modelValue.post.post_id);
isSelected.value = !isSelected.value;
}
async function toPost(): Promise<void> {
if (props.selectMode) return;
if (!card.value) return;
if (route.name !== "帖子详情") {
await createPost(card.value);
return;
}
if (route.params.post_id.toString() === card.value.postId.toString()) {
showSnackbar.warn("当前已在该帖子详情页", 3000);
return;
}
await router.push({ name: "帖子详情", params: { post_id: card.value.postId.toString() } });
}
function getActivityStatus(status: number): RenderStatus {
let idx = stats.findIndex((v) => v.stat === status);
if (idx === -1) idx = 0;
return stats[idx];
}
function getPostCover(item: TGApp.BBS.Post.FullData): string {
let cover;
if (item.cover) cover = item.cover.url;
else if (item.post.cover) cover = item.post.cover;
else if (item.post.images.length > 0) cover = item.post.images[0];
if (cover === undefined) return "";
if (cover.endsWith(".gif")) return cover;
return `${cover}?x-oss-process=image/resize,m_fill,w_690,h_320,limit_0/format,png`;
}
function getCommonCard(item: TGApp.BBS.Post.FullData): RenderCard {
let forumData: RenderForum | null = null;
let statData: RenderData | null = null;
if (item.forum !== null) {
forumData = { name: item.forum.name, icon: item.forum.icon, id: item.forum.id };
}
if (item.stat !== null) {
statData = {
mark: item.stat.bookmark_num,
forward: item.stat.forward_num,
like: item.stat.like_num,
reply: item.stat.reply_num,
view: item.stat.view_num,
};
}
const metaData: RenderMeta = {
create_time: timestampToDate(Number(item.post.created_at) * 1000),
update_time:
item.post.updated_at === 0 ? undefined : timestampToDate(Number(item.post.updated_at) * 1000),
};
return {
title: item.post.subject,
cover: getPostCover(item),
postId: Number(item.post.post_id),
subtitle: item.post.post_id,
user: item.user,
forum: forumData,
data: statData,
meta: metaData,
topics: item.topics,
reasons: item.recommend_reason?.tags ?? [],
};
}
function getPostCard(item: TGApp.BBS.Post.FullData): RenderCard {
const commonCard = getCommonCard(item);
if (
item.news_meta !== undefined &&
item.news_meta !== null &&
item.news_meta.start_at_sec !== "0"
) {
const startTime = new Date(Number(item.news_meta.start_at_sec) * 1000).toLocaleDateString();
const endTime = new Date(Number(item.news_meta.end_at_sec) * 1000).toLocaleDateString();
const statusInfo = getActivityStatus(item.news_meta.activity_status);
commonCard.subtitle = `${startTime} - ${endTime}`;
commonCard.status = statusInfo;
}
console.log(commonCard);
return commonCard;
}
async function shareCard(): Promise<void> {
if (props.selectMode) return;
if (!card.value) return;
const shareDom = document.querySelector<HTMLDivElement>(`#post-card-${card.value.postId}`);
if (shareDom === null) {
showSnackbar.error("分享内容不存在", 3000);
return;
}
const fileName = `PostCard_${card.value.postId}`;
await generateShareImg(fileName, shareDom, 2.5);
}
async function toTopic(topic: TGApp.BBS.Post.Topic): Promise<void> {
if (props.selectMode) return;
const gid = props.modelValue.post.game_id;
await emit("active_deep_link", `router?path=/posts/topic/${gid}/${topic.id}`);
}
async function toForum(forum: RenderForum): Promise<void> {
if (props.selectMode) return;
const gid = props.modelValue.post.game_id;
await emit("active_deep_link", `router?path=/posts/forum/${gid}/${forum.id}`);
}
function onUserClick(): void {
if (props.selectMode) return;
if (!card.value || card.value.user === null) return;
emits("onUserClick", card.value.user, props.modelValue.post.game_id);
}
</script>
<style lang="scss" scoped>
@use "@styles/github.styles.scss" as github-styles;
.tpc-card {
@include github-styles.github-card;
position: relative;
display: flex;
overflow: hidden;
width: 100%;
height: 100%;
flex-direction: column;
align-items: center;
justify-content: space-between;
border-radius: 4px;
row-gap: 8px;
&.select-mode {
cursor: pointer;
}
}
.dark .tpc-card {
@include github-styles.github-card("dark");
}
.tpc-top {
display: flex;
width: 100%;
flex-direction: column;
align-items: center;
justify-content: center;
row-gap: 4px;
}
.tpc-cover {
position: relative;
display: flex;
overflow: hidden;
width: 100%;
align-items: center;
justify-content: center;
aspect-ratio: 69 / 32;
background: var(--common-shadow-2);
cursor: pointer;
img {
width: 100%;
height: 100%;
object-fit: cover;
object-position: center;
transition: all 0.3s linear;
&:hover {
transform: scale(1.2);
}
}
}
.tpc-mid {
position: relative;
width: fit-content;
max-width: 100%;
box-sizing: border-box;
padding: 0 10px;
margin-right: auto;
}
.tpc-bottom {
position: relative;
display: flex;
width: 100%;
flex-direction: column;
padding: 4px 8px;
}
.tpc-title {
position: relative;
width: fit-content;
max-width: 100%;
padding: 4px 8px;
margin-right: auto;
cursor: pointer;
font-family: var(--font-title);
font-size: 18px;
}
.tpc-tags {
display: flex;
width: fit-content;
max-width: 100%;
flex-wrap: wrap;
align-items: flex-start;
justify-content: flex-start;
font-size: 12px;
gap: 4px;
}
.tpc-reason {
@include github-styles.github-tag-dark-gen(#d19a66);
display: flex;
box-sizing: border-box;
align-items: center;
justify-content: center;
padding: 0 6px;
border-radius: 12px;
gap: 2px;
user-select: none;
}
.tpc-forum {
position: absolute;
top: 0;
right: 0;
display: flex;
align-items: center;
justify-content: flex-start;
padding: 4px;
-webkit-backdrop-filter: blur(10px);
backdrop-filter: blur(10px);
border-bottom-left-radius: 4px;
box-shadow: 0 0 10px var(--tgc-dark-1);
color: var(--tgc-white-1);
cursor: pointer;
text-shadow: 0 0 4px var(--tgc-dark-1);
img {
width: 20px;
height: 20px;
margin-right: 4px;
}
}
.tpc-select {
position: absolute;
top: 0;
left: 0;
display: flex;
width: 30px;
height: 30px;
align-items: center;
justify-content: center;
-webkit-backdrop-filter: blur(20px);
backdrop-filter: blur(20px);
background: var(--box-bg-2);
border-bottom-right-radius: 4px;
box-shadow: 0 0 10px var(--tgc-dark-1);
color: var(--box-text-5);
}
.tpc-data {
display: flex;
width: fit-content;
max-width: 100%;
height: 20px;
align-items: center;
justify-content: center;
margin-left: auto;
column-gap: 8px;
}
.tpc-info-item {
display: flex;
align-items: center;
justify-content: flex-start;
color: var(--box-text-7);
font-size: 12px;
gap: 2px;
opacity: 0.6;
white-space: nowrap;
}
.tpc-act {
position: absolute;
bottom: 0;
left: 0;
display: flex;
width: 100%;
align-items: center;
justify-content: space-between;
-webkit-backdrop-filter: blur(20px);
backdrop-filter: blur(20px);
background: #00000080;
font-size: 12px;
}
.tpc-image-cnt {
position: absolute;
right: 0;
bottom: 0;
display: flex;
align-items: center;
justify-content: center;
padding-right: 4px;
padding-left: 8px;
background: var(--tgc-od-blue);
border-top-left-radius: 12px;
box-shadow: -1px -1px 6px var(--tgc-dark-1);
color: var(--tgc-white-1);
column-gap: 2px;
font-size: 12px;
line-height: 18px;
}
.tpc-status {
position: relative;
display: flex;
align-items: center;
justify-content: flex-start;
padding: 4px 30px 4px 4px;
background-color: v-bind(cardBg); /* stylelint-disable-line value-keyword-case */
clip-path: polygon(0 0, calc(100% - 15px) 0, 100% 50%, calc(100% - 15px) 100%, 0 100%);
color: var(--tgc-white-1);
&::after {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: #ffffff66;
clip-path: polygon(
calc(100% - 25px) 0,
100% 0,
100% 100%,
calc(100% - 25px) 100%,
calc(100% - 10px) 50%
);
content: "";
}
}
.tpc-time {
display: flex;
align-items: center;
justify-content: flex-start;
margin: 4px;
color: var(--tgc-white-1);
gap: 4px;
opacity: 0.8;
}
.tpc-info-id {
position: absolute;
top: 0;
left: 0;
display: flex;
align-items: center;
justify-content: center;
padding: 0 4px;
-webkit-backdrop-filter: blur(10px);
backdrop-filter: blur(10px);
background: var(--tgc-od-orange);
border-bottom-right-radius: 4px;
box-shadow: 1px 1px 6px var(--tgc-dark-1);
color: var(--tgc-white-1);
font-size: 12px;
text-shadow: 0 0 4px var(--tgc-dark-1);
}
</style>