|
|
|
|
<script setup lang="ts">
|
|
|
|
|
import { computed, ref } from 'vue'
|
|
|
|
|
import { ElProgress } from 'element-plus'
|
|
|
|
|
import { useAuthStore } from '@/stores/auth'
|
|
|
|
|
import { useCharacterStore } from '@/stores/character'
|
|
|
|
|
import { useRouter } from 'vue-router'
|
|
|
|
|
import Particles from '@/components/Particles/Particles.vue'
|
|
|
|
|
import GlareHover from '@/components/GlareHover/GlareHover.vue'
|
|
|
|
|
import ChatBox from '@/components/ChatBox.vue'
|
|
|
|
|
import ShinyText from '@/components/ShinyText/ShinyText.vue'
|
|
|
|
|
import StarBorder from '@/components/StarBorder/StarBorder.vue'
|
|
|
|
|
import trainingIcon from '@/assets/images/training.svg'
|
|
|
|
|
import missionIcon from '@/assets/images/mission.svg'
|
|
|
|
|
import scrapIcon from '@/assets/images/scrap.svg'
|
|
|
|
|
import characterIco from '@/assets/images/character.svg'
|
|
|
|
|
|
|
|
|
|
const authStore = useAuthStore()
|
|
|
|
|
const characterStore = useCharacterStore()
|
|
|
|
|
const router = useRouter()
|
|
|
|
|
|
|
|
|
|
const isTraining = computed(() => !!characterStore.currentCharacter?.trainingOn)
|
|
|
|
|
|
|
|
|
|
const canBreakthrough = computed(() => characterStore.currentCharacter?.canBreakthrough ?? false)
|
|
|
|
|
const currentLevelName = computed(() => characterStore.currentCharacter?.levelName ?? '')
|
|
|
|
|
const currentExp = computed(() => characterStore.currentCharacter?.currentExp ?? 0)
|
|
|
|
|
const nextLevelName = computed(() => characterStore.currentCharacter?.nextLevelName ?? '')
|
|
|
|
|
const nextLevelMinExp = computed(() => characterStore.currentCharacter?.nextLevelMinExp ?? 0)
|
|
|
|
|
const breakthroughRate = computed(() => characterStore.currentCharacter?.breakthroughRate ?? 0)
|
|
|
|
|
const requiredPillName = computed(() => characterStore.currentCharacter?.nextLevelRequiredPillName ?? '')
|
|
|
|
|
const requiredPillQuantity = computed(() => characterStore.currentCharacter?.nextLevelRequiredPillQuantity ?? 0)
|
|
|
|
|
const hasPillCost = computed(() => requiredPillQuantity.value > 0 && !!requiredPillName.value)
|
|
|
|
|
|
|
|
|
|
const expProgress = computed(() => {
|
|
|
|
|
if (!nextLevelMinExp.value) return 0
|
|
|
|
|
return Math.min(100, (currentExp.value / nextLevelMinExp.value) * 100)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const breakthroughMessage = ref('')
|
|
|
|
|
const showBreakthroughMessage = ref(false)
|
|
|
|
|
|
|
|
|
|
const menuItems = computed(() => [
|
|
|
|
|
{ label: '任务', icon: missionIcon, useImage: true },
|
|
|
|
|
{ label: '角色', icon: characterIco, useImage: true },
|
|
|
|
|
{ label: isTraining.value ? '打坐中' : '打坐', icon: trainingIcon, useImage: true, isTraining: isTraining.value },
|
|
|
|
|
{ label: '捡垃圾', icon: scrapIcon, useImage: true },
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
const formatNumber = (num: number) => {
|
|
|
|
|
return Math.floor(num).toLocaleString('zh-CN')
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const handleLogout = () => {
|
|
|
|
|
authStore.logout()
|
|
|
|
|
characterStore.clearCurrentCharacter()
|
|
|
|
|
window.location.href = '/login'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const handleSwitchCharacter = () => {
|
|
|
|
|
window.location.href = '/character'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const navigateTo = (item: { label: string }) => {
|
|
|
|
|
if (item.label === '打坐' || item.label === '打坐中') {
|
|
|
|
|
router.push('/training')
|
|
|
|
|
} else if (item.label === '任务') {
|
|
|
|
|
router.push('/daily-mission')
|
|
|
|
|
} else if (item.label === '捡垃圾') {
|
|
|
|
|
router.push('/scrap')
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const handleBreakthrough = async () => {
|
|
|
|
|
const result = await characterStore.breakthrough()
|
|
|
|
|
breakthroughMessage.value = result.message
|
|
|
|
|
showBreakthroughMessage.value = true
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
showBreakthroughMessage.value = false
|
|
|
|
|
}, 3000)
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<template>
|
|
|
|
|
<div class="game-page">
|
|
|
|
|
<Particles :particle-count="50" :particle-colors="['#ffffff', '#cccccc']" class="particles-bg" />
|
|
|
|
|
<div class="game-container">
|
|
|
|
|
<div style="text-align: center; margin-bottom: 20px;">
|
|
|
|
|
<StarBorder as="div" color="#22c55e" speed="5s" :thickness="2"
|
|
|
|
|
style="display: block; width: 100%; max-width: 480px;">
|
|
|
|
|
<div class="character-header" @click="handleSwitchCharacter">
|
|
|
|
|
<div class="character-info">
|
|
|
|
|
<span class="character-name">{{ characterStore.currentCharacter?.name || '未选择角色' }}</span>
|
|
|
|
|
<span class="character-level">{{ characterStore.currentCharacter?.levelName || '' }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
<span class="switch-btn">切换角色</span>
|
|
|
|
|
</div>
|
|
|
|
|
</StarBorder>
|
|
|
|
|
</div>
|
|
|
|
|
<div style="text-align: center;">
|
|
|
|
|
<ShinyText class="welcome-text" text="✨ 欢迎来到我的世界" :speed="2" :delay="0.5" :disabled="false" :color="'#b5b5b5'"
|
|
|
|
|
:shine-color="'#34fef1'" :spread="120" :direction="'left'" :yoyo="false" :pause-on-hover="false" />
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="menu-grid">
|
|
|
|
|
<GlareHover v-for="item in menuItems" :key="item.label" width="100%" height="100px"
|
|
|
|
|
:background="item.isTraining ? 'rgba(255,136,68,0.1)' : 'rgba(255,255,255,0.02)'" border-radius="16px"
|
|
|
|
|
:border-color="item.isTraining ? 'rgba(255,136,68,0.3)' : 'rgba(255,255,255,0.08)'"
|
|
|
|
|
:glare-color="item.isTraining ? '#ff8844' : '#ffffff'" :glare-opacity="0.1" class="menu-card"
|
|
|
|
|
:class="{ 'training-active': item.isTraining }" @click="navigateTo(item)">
|
|
|
|
|
<div class="menu-content">
|
|
|
|
|
<img v-if="item.useImage" :src="item.icon" class="menu-icon-img" />
|
|
|
|
|
<span v-else class="menu-icon">{{ item.icon }}</span>
|
|
|
|
|
<span class="menu-label">{{ item.label }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
</GlareHover>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div v-if="nextLevelName" class="breakthrough-section">
|
|
|
|
|
<div class="breakthrough-header">
|
|
|
|
|
<span class="breakthrough-title">境界突破</span>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="breakthrough-info">
|
|
|
|
|
<div class="level-info">
|
|
|
|
|
<span class="current-level">{{ currentLevelName }}</span>
|
|
|
|
|
<span class="arrow">→</span>
|
|
|
|
|
<span class="next-level">{{ nextLevelName }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="exp-info">
|
|
|
|
|
<span class="exp-text">{{ formatNumber(currentExp) }} / {{ formatNumber(nextLevelMinExp) }} 经验</span>
|
|
|
|
|
<el-progress :percentage="Math.floor(expProgress)" color="#ff8844" :show-text="false" :stroke-width="8" />
|
|
|
|
|
<span class="exp-percent">{{ Math.floor(expProgress) }}%</span>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="rate-info">
|
|
|
|
|
<span class="rate-label">突破成功率</span>
|
|
|
|
|
<el-progress :percentage="breakthroughRate" color="#22c55e" :show-text="false" :stroke-width="8" />
|
|
|
|
|
<span class="exp-percent">{{ breakthroughRate + `%` }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
<div v-if="hasPillCost" class="pill-cost-info">
|
|
|
|
|
<span class="pill-cost-label">突破消耗</span>
|
|
|
|
|
<span class="pill-cost-value">{{ requiredPillName }} ×{{ requiredPillQuantity }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
<button class="breakthrough-btn" :class="{ 'can-breakthrough': canBreakthrough }" :disabled="!canBreakthrough"
|
|
|
|
|
@click="handleBreakthrough">
|
|
|
|
|
{{ canBreakthrough ? '突破' : '未满足突破条件' }}
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div v-if="showBreakthroughMessage" class="toast-message">
|
|
|
|
|
{{ breakthroughMessage }}
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="page-footer">
|
|
|
|
|
<StarBorder as="div" color="#e63d3d" speed="3s" :thickness="3">
|
|
|
|
|
<div class="btn-out" @click="handleLogout">
|
|
|
|
|
退出登录
|
|
|
|
|
</div>
|
|
|
|
|
</StarBorder>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<ChatBox />
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
.game-page {
|
|
|
|
|
min-height: 100vh;
|
|
|
|
|
background: #000000;
|
|
|
|
|
padding: 20px;
|
|
|
|
|
padding-bottom: 60px;
|
|
|
|
|
position: relative;
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.particles-bg {
|
|
|
|
|
position: fixed !important;
|
|
|
|
|
top: 0;
|
|
|
|
|
left: 0;
|
|
|
|
|
width: 100% !important;
|
|
|
|
|
height: 100% !important;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.game-container {
|
|
|
|
|
max-width: 480px;
|
|
|
|
|
margin: 0 auto;
|
|
|
|
|
padding-top: 20px;
|
|
|
|
|
position: relative;
|
|
|
|
|
z-index: 10;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.character-header {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
align-items: center;
|
|
|
|
|
background: rgba(255, 255, 255, 0.03);
|
|
|
|
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
|
|
|
border-radius: 12px;
|
|
|
|
|
padding: 12px 16px;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
transition: all 0.2s ease;
|
|
|
|
|
height: 100%;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.character-header:hover {
|
|
|
|
|
background: rgba(255, 255, 255, 0.05);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.character-info {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
gap: 12px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.character-name {
|
|
|
|
|
color: #ffffff;
|
|
|
|
|
font-size: 1rem;
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.character-level {
|
|
|
|
|
color: #888888;
|
|
|
|
|
font-size: 0.8rem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.switch-btn {
|
|
|
|
|
color: #666666;
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.welcome-text {
|
|
|
|
|
font-size: 1.5rem;
|
|
|
|
|
font-weight: 300;
|
|
|
|
|
text-align: center;
|
|
|
|
|
color: #ffffff;
|
|
|
|
|
letter-spacing: 0.1em;
|
|
|
|
|
margin-bottom: 24px;
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.star-border {
|
|
|
|
|
background: transparent;
|
|
|
|
|
border: none;
|
|
|
|
|
padding: 12px 16px;
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
align-items: center;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-grid {
|
|
|
|
|
display: grid;
|
|
|
|
|
grid-template-columns: repeat(4, 1fr);
|
|
|
|
|
gap: 16px;
|
|
|
|
|
margin-bottom: 32px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-card {
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-card.training-active .menu-label {
|
|
|
|
|
color: #ff8844;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-content {
|
|
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
gap: 8px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-icon {
|
|
|
|
|
font-size: 1.75rem;
|
|
|
|
|
opacity: 0.8;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-icon-img {
|
|
|
|
|
width: 3rem;
|
|
|
|
|
height: 3rem;
|
|
|
|
|
object-fit: contain;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-label {
|
|
|
|
|
color: #cccccc;
|
|
|
|
|
font-size: 0.875rem;
|
|
|
|
|
font-weight: 400;
|
|
|
|
|
letter-spacing: 0.1em;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breakthrough-section {
|
|
|
|
|
background: rgba(255, 255, 255, 0.03);
|
|
|
|
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
|
|
|
border-radius: 16px;
|
|
|
|
|
padding: 20px;
|
|
|
|
|
margin-bottom: 24px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breakthrough-header {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
margin-bottom: 16px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breakthrough-title {
|
|
|
|
|
color: #ff8844;
|
|
|
|
|
font-size: 1rem;
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
letter-spacing: 0.1em;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breakthrough-info {
|
|
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
gap: 12px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.level-info {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
align-items: center;
|
|
|
|
|
gap: 12px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.current-level {
|
|
|
|
|
color: #888888;
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.arrow {
|
|
|
|
|
color: #555555;
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.next-level {
|
|
|
|
|
color: #ffffff;
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.exp-info {
|
|
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
gap: 6px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.exp-text {
|
|
|
|
|
color: #666666;
|
|
|
|
|
font-size: 0.8rem;
|
|
|
|
|
text-align: center;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.exp-percent {
|
|
|
|
|
color: #888888;
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
text-align: center;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.rate-info {
|
|
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
gap: 6px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.rate-label {
|
|
|
|
|
color: #666666;
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.pill-cost-info {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
align-items: center;
|
|
|
|
|
padding: 10px 14px;
|
|
|
|
|
background: rgba(255, 68, 68, 0.1);
|
|
|
|
|
border: 1px solid rgba(255, 68, 68, 0.2);
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.pill-cost-label {
|
|
|
|
|
color: #888888;
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.pill-cost-value {
|
|
|
|
|
color: #ff4444;
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breakthrough-btn {
|
|
|
|
|
width: 100%;
|
|
|
|
|
padding: 14px;
|
|
|
|
|
border-radius: 10px;
|
|
|
|
|
font-size: 0.95rem;
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
transition: all 0.3s ease;
|
|
|
|
|
border: none;
|
|
|
|
|
background: rgba(255, 255, 255, 0.05);
|
|
|
|
|
color: #555555;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breakthrough-btn:disabled {
|
|
|
|
|
cursor: not-allowed;
|
|
|
|
|
opacity: 0.5;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breakthrough-btn.can-breakthrough {
|
|
|
|
|
background: linear-gradient(135deg, #ff8844 0%, #ff6644 100%);
|
|
|
|
|
color: #ffffff;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breakthrough-btn.can-breakthrough:hover {
|
|
|
|
|
transform: scale(1.02);
|
|
|
|
|
box-shadow: 0 4px 20px rgba(255, 136, 68, 0.4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.toast-message {
|
|
|
|
|
position: fixed;
|
|
|
|
|
top: 80px;
|
|
|
|
|
left: 50%;
|
|
|
|
|
transform: translateX(-50%);
|
|
|
|
|
background: rgba(255, 255, 255, 0.9);
|
|
|
|
|
color: #000000;
|
|
|
|
|
padding: 12px 24px;
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
z-index: 1000;
|
|
|
|
|
animation: fadeIn 0.3s ease;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@keyframes fadeIn {
|
|
|
|
|
from {
|
|
|
|
|
opacity: 0;
|
|
|
|
|
transform: translateX(-50%) translateY(-10px);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
to {
|
|
|
|
|
opacity: 1;
|
|
|
|
|
transform: translateX(-50%) translateY(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logout-button {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
margin: 0 auto;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
width: 200px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.page-footer {
|
|
|
|
|
margin-top: 32px;
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.btn-out {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
align-items: center;
|
|
|
|
|
background: rgba(255, 255, 255, 0.03);
|
|
|
|
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
|
|
|
|
border-radius: 12px;
|
|
|
|
|
padding: 12px 16px;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
width: 200px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logout-text {
|
|
|
|
|
color: #888888;
|
|
|
|
|
font-size: 0.8rem;
|
|
|
|
|
font-weight: 400;
|
|
|
|
|
letter-spacing: 0.1em;
|
|
|
|
|
transition: color 0.3s ease;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logout-button:hover .logout-text {
|
|
|
|
|
color: #ffffff;
|
|
|
|
|
}
|
|
|
|
|
</style>
|