fix: 增加邀请链接逻辑

This commit is contained in:
Gsh
2025-11-01 18:39:02 +08:00
parent 5019a36138
commit 114b41144e
3 changed files with 219 additions and 17 deletions

View File

@@ -10,9 +10,15 @@
*/
import type { CardFlipRecord, CardFlipStatusOutput, FlipCardOutput } from '@/api/cardFlip/types';
import { ElMessage, ElMessageBox } from 'element-plus';
import { onMounted, ref } from 'vue';
import { onMounted, ref, watch } from 'vue';
import { flipCard, generateMyInviteCode, getWeeklyTaskStatus, useInviteCode } from '@/api/cardFlip';
// ============ Props ============
/** 接收外部传入的邀请码(通过 URL 参数传递) */
const props = defineProps<{
externalInviteCode?: string;
}>();
// ============ 状态管理 ============
/** 任务数据:包含翻牌记录、次数统计、邀请码等信息 */
const taskData = ref<CardFlipStatusOutput | null>(null);
@@ -83,6 +89,29 @@ onMounted(async () => {
await playShuffleAnimation();
});
// ============ 监听外部邀请码 ============
/**
* 监听外部传入的邀请码
* 如果有邀请码,自动打开对话框并预填
*/
watch(
() => props.externalInviteCode,
(newCode) => {
if (newCode && newCode.trim()) {
// 延迟500ms确保页面已经渲染完成
setTimeout(() => {
inputInviteCode.value = newCode.trim().toUpperCase();
inviteCodeDialog.value = true;
ElMessage.info({
message: '🎁 检测到邀请码,已为您自动填充,请点击确认使用',
duration: 3000,
});
}, 500);
}
},
{ immediate: true }
);
// ============ 新增:初始洗牌动画 ============
/**
* 初始洗牌动画
@@ -738,6 +767,52 @@ function copyInviteCode() {
});
}
/**
* 生成分享链接
* 生成包含邀请码的完整 URL
*/
function generateShareLink(): string {
const inviteCode = taskData.value?.myInviteCode;
if (!inviteCode) return '';
const baseUrl = window.location.origin;
const path = window.location.pathname;
return `${baseUrl}${path}?inviteCode=${inviteCode}`;
}
/**
* 生成分享文案
* 类似小红书风格的分享内容
*/
function generateShareContent(): string {
const shareLink = generateShareLink();
return `🎁 邀请你来翻牌抽奖赢取Token大奖
使用我的邀请码可解锁3次额外翻牌机会必定中奖最大奖励翻倍💰
👉 点击链接立即参与:
${shareLink}
🍀 快来试试手气吧!`;
}
/**
* 复制分享链接和文案
*/
function copyShareContent() {
if (!taskData.value?.myInviteCode) {
ElMessage.warning('暂无邀请码');
return;
}
const shareContent = generateShareContent();
navigator.clipboard.writeText(shareContent).then(() => {
ElMessage.success('🎉 分享内容已复制到剪贴板!快去分享给好友吧~');
}).catch(() => {
ElMessage.error('复制失败,请手动复制');
});
}
// ============ 工具函数 ============
/**
* 格式化 Token 显示(单位:万)
@@ -1063,16 +1138,31 @@ function getCardClass(record: CardFlipRecord): string[] {
<div class="code-box">
<span class="code-text">{{ taskData.myInviteCode }}</span>
</div>
<el-button
type="primary"
icon="CopyDocument"
@click="copyInviteCode"
>
复制邀请码
</el-button>
<div class="button-group">
<el-button
type="primary"
icon="CopyDocument"
@click="copyInviteCode"
>
复制邀请码
</el-button>
<el-button
type="success"
icon="Share"
@click="copyShareContent"
>
复制分享链接
</el-button>
</div>
<p class="invite-tip">
💌 分享给好友好友使用后可解锁最后3次翻牌机会
</p>
<div class="share-preview">
<div class="share-preview-title">📱 分享预览</div>
<div class="share-preview-content">
{{ generateShareContent() }}
</div>
</div>
</div>
<!-- 生成邀请码 -->
@@ -1999,16 +2089,68 @@ function getCardClass(record: CardFlipRecord): string[] {
}
}
.el-button {
width: 100%;
.button-group {
display: flex;
gap: 8px;
margin-bottom: 12px;
.el-button {
flex: 1;
}
}
.invite-tip {
font-size: 13px;
color: #909399;
line-height: 1.5;
margin: 0;
margin: 0 0 16px 0;
}
.share-preview {
background: #f5f7fa;
border: 1px solid #e4e7ed;
border-radius: 8px;
padding: 12px;
text-align: left;
.share-preview-title {
font-size: 14px;
font-weight: bold;
color: #303133;
margin-bottom: 8px;
text-align: center;
}
.share-preview-content {
font-size: 13px;
color: #606266;
line-height: 1.6;
white-space: pre-line;
background: #fff;
padding: 10px;
border-radius: 4px;
max-height: 200px;
overflow-y: auto;
/* 自定义滚动条 */
&::-webkit-scrollbar {
width: 4px;
}
&::-webkit-scrollbar-track {
background: rgba(0, 0, 0, 0.05);
border-radius: 2px;
}
&::-webkit-scrollbar-thumb {
background: rgba(0, 0, 0, 0.2);
border-radius: 2px;
&:hover {
background: rgba(0, 0, 0, 0.3);
}
}
}
}
}

View File

@@ -1,8 +1,9 @@
<!-- 头像 -->
<script setup lang="ts">
import { computed } from 'vue';
import { useRouter } from 'vue-router';
import { ChatLineRound } from '@element-plus/icons-vue';
import { ElMessage, ElMessageBox } from 'element-plus';
import { computed, nextTick, onMounted, ref, watch } from 'vue';
import { useRouter } from 'vue-router';
import Popover from '@/components/Popover/index.vue';
import SvgIcon from '@/components/SvgIcon/index.vue';
import { useUserStore } from '@/stores';
@@ -62,6 +63,10 @@ const dialogVisible = ref(false);
const rechargeLogRef = ref();
const activeNav = ref('user');
// ============ 邀请码分享功能 ============
/** 从 URL 获取的邀请码 */
const externalInviteCode = ref<string>('');
const navItems = [
{ name: 'user', label: '用户信息', icon: 'User' },
// { name: 'role', label: '角色管理', icon: 'Avatar' },
@@ -73,7 +78,7 @@ const navItems = [
{ name: 'usageStatistics', label: '用量统计', icon: 'Histogram' },
{ name: 'premiumService', label: '尊享服务', icon: 'ColdDrink' },
{ name: 'dailyTask', label: '每日任务(限时)', icon: 'Trophy' },
{ name: 'cardFlip', label: '每周邀请(限时)', icon: 'Present' }
{ name: 'cardFlip', label: '每周邀请(限时)', icon: 'Present' },
// { name: 'usageStatistics2', label: '用量统计2', icon: 'Histogram' },
];
function openDialog() {
@@ -183,7 +188,61 @@ function openVipGuide() {
function onProductPackage() {
showProductPackage();
}
// 直接调用
// ============ 监听对话框打开事件,切换到邀请码标签页 ============
watch(dialogVisible, (newVal) => {
if (newVal && externalInviteCode.value) {
// 对话框打开后,切换标签页(已通过 :default-active 绑定,会自动响应)
// console.log('[Avatar] watch: 对话框已打开,切换到 cardFlip 标签页');
nextTick(() => {
activeNav.value = 'cardFlip';
// console.log('[Avatar] watch: 已设置 activeNav 为', activeNav.value);
});
}
// 对话框关闭时,清除邀请码状态和 URL 参数
if (!newVal && externalInviteCode.value) {
// console.log('[Avatar] watch: 对话框关闭,清除邀请码状态');
externalInviteCode.value = '';
// 清除 URL 中的 inviteCode 参数
const url = new URL(window.location.href);
if (url.searchParams.has('inviteCode')) {
url.searchParams.delete('inviteCode');
window.history.replaceState({}, '', url.toString());
// console.log('[Avatar] watch: 已清除 URL 中的 inviteCode 参数');
}
}
});
// ============ 监听 URL 参数,实现邀请码快捷分享 ============
onMounted(() => {
// 获取 URL 查询参数
const urlParams = new URLSearchParams(window.location.search);
const inviteCode = urlParams.get('inviteCode');
if (inviteCode && inviteCode.trim()) {
// console.log('[Avatar] onMounted: 检测到邀请码', inviteCode);
// 保存邀请码
externalInviteCode.value = inviteCode.trim();
// 先设置标签页为 cardFlip
activeNav.value = 'cardFlip';
// console.log('[Avatar] onMounted: 设置 activeNav 为', activeNav.value);
// 延迟打开对话框,确保状态已更新
nextTick(() => {
setTimeout(() => {
// console.log('[Avatar] onMounted: 打开用户中心对话框');
dialogVisible.value = true;
}, 200);
});
// 注意:不立即清除 URL 参数,保留给登录后使用
// URL 参数会在对话框关闭时清除
}
});
</script>
<template>
@@ -323,6 +382,7 @@ function onProductPackage() {
v-model="dialogVisible"
title="用户中心"
:nav-items="navItems"
:default-active="activeNav"
@confirm="handleConfirm"
@nav-change="handleNavChange"
>
@@ -370,7 +430,7 @@ function onProductPackage() {
<daily-task />
</template>
<template #cardFlip>
<card-flip-activity />
<card-flip-activity :external-invite-code="externalInviteCode" />
</template>
<template #rechargeLog>
<recharge-log ref="rechargeLogRef" />

View File

@@ -11,6 +11,7 @@ declare module 'vue' {
AccountPassword: typeof import('./../src/components/LoginDialog/components/FormLogin/AccountPassword.vue')['default']
APIKeyManagement: typeof import('./../src/components/userPersonalCenter/components/APIKeyManagement.vue')['default']
CardFlipActivity: typeof import('./../src/components/userPersonalCenter/components/CardFlipActivity.vue')['default']
CardFlipActivity2: typeof import('./../src/components/userPersonalCenter/components/CardFlipActivity2.vue')['default']
DailyTask: typeof import('./../src/components/userPersonalCenter/components/DailyTask.vue')['default']
DeepThinking: typeof import('./../src/components/DeepThinking/index.vue')['default']
ElAlert: typeof import('element-plus/es')['ElAlert']
@@ -20,7 +21,6 @@ declare module 'vue' {
ElCard: typeof import('element-plus/es')['ElCard']
ElCollapse: typeof import('element-plus/es')['ElCollapse']
ElCollapseItem: typeof import('element-plus/es')['ElCollapseItem']
ElCollapseTransition: typeof import('element-plus/es')['ElCollapseTransition']
ElContainer: typeof import('element-plus/es')['ElContainer']
ElDialog: typeof import('element-plus/es')['ElDialog']
ElDivider: typeof import('element-plus/es')['ElDivider']