228 lines
8.0 KiB
Vue
228 lines
8.0 KiB
Vue
<template>
|
||
<section class="mx-auto max-w-md px-4 pb-28 pt-6 space-y-6">
|
||
<!-- Header -->
|
||
<header class="flex items-center gap-3">
|
||
<div class="w-12 h-12 rounded-2xl bg-gradient-to-br from-mosquito-primary to-mosquito-primary-light flex items-center justify-center text-white shadow-lg">
|
||
<Icons name="share" class="w-6 h-6" />
|
||
</div>
|
||
<div>
|
||
<h1 class="mos-title text-2xl font-bold">分享推广</h1>
|
||
<p class="text-sm text-mosquito-muted">生成专属链接,邀请好友参与</p>
|
||
</div>
|
||
</header>
|
||
|
||
<!-- Auth Warning -->
|
||
<div v-if="!hasAuth" class="mos-card border-2 border-dashed border-mosquito-warning/30 bg-mosquito-warning/5 p-5">
|
||
<div class="flex items-start gap-3">
|
||
<div class="flex h-10 w-10 shrink-0 items-center justify-center rounded-xl bg-mosquito-warning/20 text-mosquito-warning">
|
||
<Icons name="zap" class="w-5 h-5" />
|
||
</div>
|
||
<div>
|
||
<div class="font-semibold text-mosquito-ink">请先配置鉴权信息</div>
|
||
<div class="text-sm text-mosquito-muted mt-1">需要 API Key 与用户令牌才可生成链接与海报</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Share Link Card -->
|
||
<div class="mos-card-gradient p-6 space-y-4">
|
||
<div class="flex items-center gap-2 text-white/90">
|
||
<Icons name="rocket" class="w-4 h-4" />
|
||
<span class="text-xs font-bold uppercase tracking-wider">默认模板</span>
|
||
<span class="text-xs opacity-75">· {{ activityLabel }}</span>
|
||
</div>
|
||
|
||
<div class="flex flex-wrap items-center gap-3">
|
||
<template v-if="hasAuth">
|
||
<MosquitoShareButton :activity-id="activityId" :user-id="userId" @copied="handleCopied" @error="handleCopyError" />
|
||
<button class="mos-btn mos-btn-accent !py-2 !px-4" @click="handleCopyLink">
|
||
<Icons name="copy" class="w-4 h-4" />
|
||
{{ copyButtonText }}
|
||
</button>
|
||
</template>
|
||
<template v-else>
|
||
<button class="mos-btn mos-btn-accent !py-2 !px-4 opacity-50 cursor-not-allowed" disabled>
|
||
<Icons name="copy" class="w-4 h-4" />
|
||
复制链接
|
||
</button>
|
||
</template>
|
||
</div>
|
||
|
||
<p class="text-xs text-white/70 flex items-center gap-1">
|
||
<Icons name="check-circle" class="w-3 h-3" />
|
||
分享按钮会自动复制链接,方便一键转发
|
||
</p>
|
||
</div>
|
||
|
||
<!-- Poster Card -->
|
||
<div class="mos-card p-5">
|
||
<div class="flex items-center justify-between mb-4">
|
||
<div class="flex items-center gap-2">
|
||
<div class="w-8 h-8 rounded-lg bg-gradient-to-br from-mosquito-accent to-mosquito-accent-light flex items-center justify-center">
|
||
<Icons name="gift" class="w-4 h-4 text-mosquito-ink" />
|
||
</div>
|
||
<h2 class="mos-title text-base font-bold">分享海报</h2>
|
||
</div>
|
||
<span class="mos-pill mos-pill-secondary">点击可重试</span>
|
||
</div>
|
||
|
||
<div class="flex justify-center bg-mosquito-bg rounded-2xl p-4">
|
||
<MosquitoPosterCard
|
||
v-if="hasAuth"
|
||
:activity-id="activityId"
|
||
:user-id="userId"
|
||
template="default"
|
||
width="280px"
|
||
height="380px"
|
||
/>
|
||
<div v-else class="flex h-[380px] w-[280px] items-center justify-center rounded-xl border border-dashed border-mosquito-muted/40 text-sm text-mosquito-muted">
|
||
配置鉴权后可预览海报
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Guide Card -->
|
||
<div class="mos-card space-y-4 p-5">
|
||
<div class="flex items-center gap-2">
|
||
<div class="w-8 h-8 rounded-lg bg-mosquito-secondary/10 flex items-center justify-center text-mosquito-secondary-dark">
|
||
<Icons name="target" class="w-4 h-4" />
|
||
</div>
|
||
<h3 class="mos-title text-base font-bold">分享指引</h3>
|
||
</div>
|
||
|
||
<ul class="space-y-3">
|
||
<li v-for="(step, index) in guideSteps" :key="index" class="flex gap-3 items-start">
|
||
<span class="flex h-6 w-6 shrink-0 items-center justify-center rounded-full bg-mosquito-primary text-white text-xs font-bold">
|
||
{{ index + 1 }}
|
||
</span>
|
||
<span class="text-sm text-mosquito-ink-light">{{ step }}</span>
|
||
</li>
|
||
</ul>
|
||
</div>
|
||
|
||
<!-- Error Message -->
|
||
<div v-if="loadError" class="mos-card border-2 border-mosquito-error/30 bg-mosquito-error/5 p-4">
|
||
<div class="flex items-center gap-2 text-mosquito-error">
|
||
<ZapIcon class="w-4 h-4" />
|
||
<span class="text-sm font-medium">{{ loadError }}</span>
|
||
</div>
|
||
</div>
|
||
</section>
|
||
</template>
|
||
|
||
<script setup lang="ts">
|
||
import { computed, onMounted, ref } from 'vue'
|
||
import { useRoute } from 'vue-router'
|
||
import MosquitoShareButton from '../../../components/MosquitoShareButton.vue'
|
||
import MosquitoPosterCard from '../../../components/MosquitoPosterCard.vue'
|
||
import { MosquitoError, useMosquito } from '../../../index'
|
||
import { getUserIdFromToken, parseUserId } from '../../../shared/auth'
|
||
import Icons from '../components/Icons.vue'
|
||
|
||
type ActivitySummary = {
|
||
id: number
|
||
name: string
|
||
}
|
||
|
||
const route = useRoute()
|
||
const { getActivities, getShareUrl } = useMosquito()
|
||
const apiKey = import.meta.env.VITE_MOSQUITO_API_KEY
|
||
const userToken = import.meta.env.VITE_MOSQUITO_USER_TOKEN
|
||
const routeUserId = computed(() => parseUserId(route.query.userId ?? route.params.userId))
|
||
const userId = computed(() => getUserIdFromToken(userToken) ?? routeUserId.value ?? 0)
|
||
const activityId = ref(1)
|
||
const activityLabel = computed(() => `活动 #${activityId.value}`)
|
||
const loadError = ref('')
|
||
const hasAuth = computed(() => Boolean(apiKey && userToken && userId.value))
|
||
const copyButtonText = ref('复制链接')
|
||
const copyFeedback = ref<'success' | 'error' | null>(null)
|
||
|
||
const guideSteps = [
|
||
'点击"分享给好友"生成专属链接',
|
||
'发送给好友,完成注册后即可计入转化',
|
||
'回到首页查看最新排行和奖励进度'
|
||
]
|
||
|
||
// 复制链接处理
|
||
const handleCopyLink = async () => {
|
||
try {
|
||
const shareResponse = await getShareUrl(activityId.value, userId.value, 'default')
|
||
let urlToCopy: string
|
||
|
||
if (shareResponse && typeof shareResponse === 'object') {
|
||
if (shareResponse.originalUrl) {
|
||
urlToCopy = shareResponse.originalUrl
|
||
} else if (shareResponse.path) {
|
||
urlToCopy = shareResponse.path.startsWith('http')
|
||
? shareResponse.path
|
||
: `${window.location.origin}${shareResponse.path}`
|
||
} else {
|
||
throw new Error('分享链接响应格式异常')
|
||
}
|
||
} else {
|
||
throw new Error('分享链接响应格式异常')
|
||
}
|
||
|
||
// 复制到剪贴板
|
||
try {
|
||
await navigator.clipboard.writeText(urlToCopy)
|
||
showCopyFeedback('success')
|
||
} catch {
|
||
// 回退到传统方法
|
||
const textArea = document.createElement('textarea')
|
||
textArea.value = urlToCopy
|
||
document.body.appendChild(textArea)
|
||
textArea.select()
|
||
document.execCommand('copy')
|
||
document.body.removeChild(textArea)
|
||
showCopyFeedback('success')
|
||
}
|
||
} catch (error) {
|
||
console.error('复制链接失败:', error)
|
||
showCopyFeedback('error')
|
||
}
|
||
}
|
||
|
||
// 显示复制反馈
|
||
const showCopyFeedback = (type: 'success' | 'error') => {
|
||
copyFeedback.value = type
|
||
copyButtonText.value = type === 'success' ? '已复制!' : '复制失败'
|
||
setTimeout(() => {
|
||
copyButtonText.value = '复制链接'
|
||
copyFeedback.value = null
|
||
}, 2000)
|
||
}
|
||
|
||
// MosquitoShareButton 回调处理
|
||
const handleCopied = () => {
|
||
showCopyFeedback('success')
|
||
}
|
||
|
||
const handleCopyError = () => {
|
||
showCopyFeedback('error')
|
||
}
|
||
|
||
const loadActivity = async () => {
|
||
if (!hasAuth.value) {
|
||
return
|
||
}
|
||
loadError.value = ''
|
||
try {
|
||
const list: ActivitySummary[] = await getActivities()
|
||
if (list.length) {
|
||
activityId.value = list[0].id
|
||
}
|
||
} catch (error) {
|
||
if (error instanceof MosquitoError && error.statusCode === 401) {
|
||
loadError.value = '鉴权失败:无法加载活动信息'
|
||
return
|
||
}
|
||
loadError.value = '活动信息加载失败'
|
||
}
|
||
}
|
||
|
||
onMounted(() => {
|
||
loadActivity()
|
||
})
|
||
</script>
|