2025-10-23 09:59:54 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="system-settings">
|
|
|
|
|
|
<!-- 左侧导航栏 -->
|
|
|
|
|
|
<aside class="sidebar">
|
|
|
|
|
|
<div class="logo">
|
2025-11-21 16:10:00 +08:00
|
|
|
|
<img src="/images/backgrounds/logo-admin.svg" alt="Logo" />
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<nav class="nav-menu">
|
|
|
|
|
|
<div class="nav-item" @click="goToDashboard">
|
|
|
|
|
|
<el-icon><Grid /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span>{{ $t('nav.dashboard') }}</span>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="nav-item" @click="goToMembers">
|
|
|
|
|
|
<el-icon><User /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span>{{ $t('nav.members') }}</span>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="nav-item" @click="goToOrders">
|
|
|
|
|
|
<el-icon><ShoppingCart /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span>{{ $t('nav.orders') }}</span>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="nav-item" @click="goToAPI">
|
|
|
|
|
|
<el-icon><Document /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span>{{ $t('nav.apiManagement') }}</span>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="nav-item" @click="goToTasks">
|
|
|
|
|
|
<el-icon><Document /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span>{{ $t('nav.tasks') }}</span>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="nav-item active">
|
|
|
|
|
|
<el-icon><Setting /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span>{{ $t('nav.systemSettings') }}</span>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</nav>
|
|
|
|
|
|
<div class="sidebar-footer">
|
|
|
|
|
|
<div class="online-users">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
{{ $t('nav.onlineUsers') }}: <span class="highlight">{{ onlineUsers }}</span>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="system-uptime">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
{{ $t('nav.systemUptime') }}: <span class="highlight">{{ systemUptime }}</span>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</aside>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 主内容区域 -->
|
|
|
|
|
|
<main class="main-content">
|
|
|
|
|
|
<!-- 顶部搜索栏 -->
|
|
|
|
|
|
<header class="top-header">
|
|
|
|
|
|
<div class="search-bar">
|
|
|
|
|
|
<el-icon class="search-icon"><User /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<input type="text" :placeholder="$t('common.searchPlaceholder')" class="search-input">
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="header-actions">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<LanguageSwitcher />
|
2025-11-21 16:10:00 +08:00
|
|
|
|
<el-dropdown @command="handleUserCommand">
|
|
|
|
|
|
<div class="user-avatar">
|
|
|
|
|
|
<img src="/images/backgrounds/avatar-default.svg" alt="用户头像" />
|
|
|
|
|
|
<el-icon class="arrow-down"><ArrowDown /></el-icon>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<template #dropdown>
|
|
|
|
|
|
<el-dropdown-menu>
|
|
|
|
|
|
<el-dropdown-item command="exitAdmin">
|
|
|
|
|
|
{{ $t('admin.exitAdmin') }}
|
|
|
|
|
|
</el-dropdown-item>
|
|
|
|
|
|
</el-dropdown-menu>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-dropdown>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</header>
|
|
|
|
|
|
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<!-- 设置选项卡 -->
|
|
|
|
|
|
<div class="settings-tabs">
|
|
|
|
|
|
<div class="tab-nav">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<div
|
|
|
|
|
|
class="tab-item"
|
|
|
|
|
|
:class="{ active: activeTab === 'membership' }"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
@click="activeTab = 'membership'"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-icon><User /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span>{{ $t('systemSettings.membership') }}</span>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<div
|
|
|
|
|
|
class="tab-item"
|
|
|
|
|
|
:class="{ active: activeTab === 'cleanup' }"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
@click="activeTab = 'cleanup'"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-icon><Delete /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span>{{ $t('systemSettings.cleanup') }}</span>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
2025-12-05 09:57:09 +08:00
|
|
|
|
<div
|
|
|
|
|
|
class="tab-item"
|
|
|
|
|
|
:class="{ active: activeTab === 'aiModel' }"
|
|
|
|
|
|
@click="activeTab = 'aiModel'"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-icon><Setting /></el-icon>
|
|
|
|
|
|
<span>{{ $t('systemSettings.aiModel') }}</span>
|
|
|
|
|
|
</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 会员收费标准选项卡 -->
|
|
|
|
|
|
<div v-if="activeTab === 'membership'" class="tab-content">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<h2 class="page-title">{{ $t('systemSettings.membership') }}</h2>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<div class="membership-cards">
|
|
|
|
|
|
<el-card v-for="level in membershipLevels" :key="level.id" class="membership-card">
|
|
|
|
|
|
<div class="card-header">
|
|
|
|
|
|
<h3>{{ level.name }}</h3>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="card-body">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<p class="price">${{ level.price || 0 }}{{ $t('systemSettings.perMonth') }}</p>
|
|
|
|
|
|
<p class="description">{{ level.description || $t('systemSettings.includesPoints', { points: level.resourcePoints || 0 }) }}</p>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="card-footer">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<el-button type="primary" @click="editLevel(level)">{{ $t('common.edit') }}</el-button>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</div>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
|
|
|
|
|
|
<!-- 任务清理管理选项卡 -->
|
|
|
|
|
|
<div v-if="activeTab === 'cleanup'" class="tab-content">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<h2 class="page-title">{{ $t('systemSettings.cleanup') }}</h2>
|
|
|
|
|
|
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<!-- 清理统计信息 -->
|
|
|
|
|
|
<div class="cleanup-stats">
|
|
|
|
|
|
<el-card class="stats-card">
|
|
|
|
|
|
<template #header>
|
|
|
|
|
|
<div class="card-header">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<h3>{{ $t('systemSettings.cleanupStatsInfo') }}</h3>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<el-button type="primary" @click="refreshStats" :loading="loadingStats">
|
|
|
|
|
|
<el-icon><Refresh /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
{{ $t('systemSettings.refresh') }}
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<div class="stats-content" v-if="cleanupStats">
|
|
|
|
|
|
<div class="stats-grid">
|
|
|
|
|
|
<div class="stat-item">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<div class="stat-label">{{ $t('systemSettings.currentTotalTasks') }}</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<div class="stat-value">{{ cleanupStats.currentTasks?.textToVideo?.total + cleanupStats.currentTasks?.imageToVideo?.total || 0 }}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="stat-item">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<div class="stat-label">{{ $t('systemSettings.completedTasks') }}</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<div class="stat-value">{{ cleanupStats.currentTasks?.textToVideo?.completed + cleanupStats.currentTasks?.imageToVideo?.completed || 0 }}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="stat-item">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<div class="stat-label">{{ $t('systemSettings.failedTasks') }}</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<div class="stat-value">{{ cleanupStats.currentTasks?.textToVideo?.failed + cleanupStats.currentTasks?.imageToVideo?.failed || 0 }}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="stat-item">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<div class="stat-label">{{ $t('systemSettings.archivedTasks') }}</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<div class="stat-value">{{ cleanupStats.archives?.completedTasks || 0 }}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="stat-item">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<div class="stat-label">{{ $t('systemSettings.cleanupLogsCount') }}</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<div class="stat-value">{{ cleanupStats.archives?.cleanupLogs || 0 }}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="stat-item">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<div class="stat-label">{{ $t('systemSettings.retentionDays') }}</div>
|
|
|
|
|
|
<div class="stat-value">{{ cleanupStats.config?.retentionDays || 30 }}{{ $t('systemSettings.days') }}</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 清理操作 -->
|
|
|
|
|
|
<div class="cleanup-actions">
|
|
|
|
|
|
<el-card class="actions-card">
|
|
|
|
|
|
<template #header>
|
|
|
|
|
|
<div class="card-header">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<h3>{{ $t('systemSettings.cleanupActions') }}</h3>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<div class="actions-content">
|
|
|
|
|
|
<div class="action-buttons">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<el-button
|
|
|
|
|
|
type="primary"
|
|
|
|
|
|
@click="performFullCleanup"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
:loading="loadingCleanup"
|
|
|
|
|
|
class="action-btn"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-icon><Delete /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
{{ $t('systemSettings.performFullCleanup') }}
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</el-button>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<el-button
|
|
|
|
|
|
type="warning"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
@click="showUserCleanupDialog = true"
|
|
|
|
|
|
class="action-btn"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-icon><User /></el-icon>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
{{ $t('systemSettings.cleanupUserTasks') }}
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="action-description">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<p><strong>{{ $t('systemSettings.fullCleanupDesc') }}:</strong>{{ $t('systemSettings.fullCleanupDescDetail') }}</p>
|
|
|
|
|
|
<p><strong>{{ $t('systemSettings.userCleanupDesc') }}:</strong>{{ $t('systemSettings.userCleanupDescDetail') }}</p>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 清理配置 -->
|
|
|
|
|
|
<div class="cleanup-config">
|
|
|
|
|
|
<el-card class="config-card">
|
|
|
|
|
|
<template #header>
|
|
|
|
|
|
<div class="card-header">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<h3>{{ $t('systemSettings.cleanupConfig') }}</h3>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<div class="config-content">
|
|
|
|
|
|
<el-form :model="cleanupConfig" label-width="120px">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<el-form-item :label="$t('systemSettings.taskRetentionDays')">
|
|
|
|
|
|
<el-input-number
|
|
|
|
|
|
v-model="cleanupConfig.retentionDays"
|
|
|
|
|
|
:min="1"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
:max="365"
|
|
|
|
|
|
controls-position="right"
|
|
|
|
|
|
/>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span class="config-tip">{{ $t('systemSettings.taskRetentionTip') }}</span>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</el-form-item>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<el-form-item :label="$t('systemSettings.archiveRetentionDays')">
|
|
|
|
|
|
<el-input-number
|
|
|
|
|
|
v-model="cleanupConfig.archiveRetentionDays"
|
|
|
|
|
|
:min="30"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
:max="3650"
|
|
|
|
|
|
controls-position="right"
|
|
|
|
|
|
/>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<span class="config-tip">{{ $t('systemSettings.archiveRetentionTip') }}</span>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item>
|
|
|
|
|
|
<el-button type="primary" @click="saveCleanupConfig" :loading="loadingConfig">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
{{ $t('common.save') }}
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</el-button>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-12-05 09:57:09 +08:00
|
|
|
|
|
|
|
|
|
|
<!-- AI模型设置选项卡 -->
|
|
|
|
|
|
<div v-if="activeTab === 'aiModel'" class="tab-content">
|
|
|
|
|
|
<h2 class="page-title">{{ $t('systemSettings.aiModel') }}</h2>
|
|
|
|
|
|
<el-card class="ai-model-card">
|
|
|
|
|
|
<template #header>
|
|
|
|
|
|
<div class="card-header">
|
|
|
|
|
|
<h3>{{ $t('systemSettings.promptOptimization') }}</h3>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<div class="ai-model-content">
|
|
|
|
|
|
<el-form label-width="180px">
|
|
|
|
|
|
<el-form-item :label="$t('systemSettings.promptOptimizationApiUrl')">
|
|
|
|
|
|
<el-input v-model="promptOptimizationApiUrl" style="width: 400px;" placeholder="https://ai.comfly.chat"></el-input>
|
|
|
|
|
|
<div class="model-tip">{{ $t('systemSettings.promptOptimizationApiUrlTip') }}</div>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item :label="$t('systemSettings.promptOptimizationModel')">
|
|
|
|
|
|
<el-input v-model="promptOptimizationModel" style="width: 400px;" placeholder="gpt-5.1-thinking"></el-input>
|
|
|
|
|
|
<div class="model-tip">{{ $t('systemSettings.promptOptimizationModelTip') }}</div>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item>
|
|
|
|
|
|
<el-button type="primary" @click="saveAiModelSettings" :loading="savingAiModel">
|
|
|
|
|
|
{{ $t('common.save') }}
|
|
|
|
|
|
</el-button>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</div>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</div>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</main>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 编辑会员收费标准对话框 -->
|
|
|
|
|
|
<el-dialog
|
|
|
|
|
|
v-model="editDialogVisible"
|
|
|
|
|
|
width="480px"
|
|
|
|
|
|
:before-close="handleCloseEditDialog"
|
|
|
|
|
|
class="membership-modal"
|
|
|
|
|
|
:show-close="false"
|
|
|
|
|
|
>
|
|
|
|
|
|
<template #header>
|
|
|
|
|
|
<div class="modal-header">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<h2 class="modal-title">{{ $t('systemSettings.membership') }}</h2>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
<button class="close-btn" @click="handleCloseEditDialog">×</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
<div class="modal-content">
|
|
|
|
|
|
<el-form :model="editForm" :rules="editRules" ref="editFormRef">
|
|
|
|
|
|
<div class="form-group">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<label class="form-label">{{ $t('systemSettings.membershipLevel') }}</label>
|
|
|
|
|
|
<el-select v-model="editForm.level" :placeholder="$t('systemSettings.selectLevelPlaceholder')" style="width: 100%;">
|
|
|
|
|
|
<el-option :label="$t('systemSettings.freeMembership')" value="free"></el-option>
|
|
|
|
|
|
<el-option :label="$t('systemSettings.standardMembership')" value="standard"></el-option>
|
|
|
|
|
|
<el-option :label="$t('systemSettings.professionalMembership')" value="professional"></el-option>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</el-select>
|
|
|
|
|
|
</div>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
<div class="form-group">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<label class="form-label">{{ $t('systemSettings.membershipPrice') }}</label>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
<div class="price-input">
|
|
|
|
|
|
<span class="price-prefix">$</span>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<input
|
|
|
|
|
|
type="text"
|
|
|
|
|
|
v-model="editForm.price"
|
|
|
|
|
|
placeholder="0.00"
|
2025-10-23 09:59:54 +08:00
|
|
|
|
class="form-control"
|
|
|
|
|
|
@input="handlePriceInput"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
<div class="form-group">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<label class="form-label">{{ $t('systemSettings.resourcePointsAmount') }}</label>
|
|
|
|
|
|
<input
|
|
|
|
|
|
type="number"
|
|
|
|
|
|
v-model="editForm.resourcePoints"
|
|
|
|
|
|
placeholder="0"
|
2025-10-23 09:59:54 +08:00
|
|
|
|
min="0"
|
|
|
|
|
|
class="form-control"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
<div class="form-group">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<label class="form-label">{{ $t('systemSettings.validityPeriod') }}</label>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
<div class="radio-group">
|
|
|
|
|
|
<div class="radio-option">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<input
|
|
|
|
|
|
type="radio"
|
|
|
|
|
|
id="monthly"
|
|
|
|
|
|
v-model="editForm.validityPeriod"
|
|
|
|
|
|
value="monthly"
|
2025-10-23 09:59:54 +08:00
|
|
|
|
class="radio-input"
|
|
|
|
|
|
>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<label for="monthly" class="radio-label">{{ $t('systemSettings.monthly') }}</label>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="radio-option">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<input
|
|
|
|
|
|
type="radio"
|
|
|
|
|
|
id="quarterly"
|
|
|
|
|
|
v-model="editForm.validityPeriod"
|
|
|
|
|
|
value="quarterly"
|
2025-10-23 09:59:54 +08:00
|
|
|
|
class="radio-input"
|
|
|
|
|
|
>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<label for="quarterly" class="radio-label">{{ $t('systemSettings.quarterly') }}</label>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="radio-option">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<input
|
|
|
|
|
|
type="radio"
|
|
|
|
|
|
id="yearly"
|
|
|
|
|
|
v-model="editForm.validityPeriod"
|
|
|
|
|
|
value="yearly"
|
2025-10-23 09:59:54 +08:00
|
|
|
|
class="radio-input"
|
|
|
|
|
|
>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<label for="yearly" class="radio-label">{{ $t('systemSettings.yearly') }}</label>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
</div>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
<template #footer>
|
|
|
|
|
|
<div class="modal-footer">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<button class="btn btn-cancel" @click="handleCloseEditDialog">{{ $t('common.cancel') }}</button>
|
|
|
|
|
|
<button class="btn btn-save" @click="saveEdit">{{ $t('common.save') }}</button>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-dialog>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
|
|
|
|
|
|
<!-- 用户清理对话框 -->
|
|
|
|
|
|
<el-dialog
|
|
|
|
|
|
v-model="showUserCleanupDialog"
|
2025-11-13 17:01:39 +08:00
|
|
|
|
:title="$t('systemSettings.cleanupUserTasks')"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
width="480px"
|
|
|
|
|
|
:before-close="handleCloseUserCleanupDialog"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div class="user-cleanup-content">
|
|
|
|
|
|
<el-form :model="userCleanupForm" :rules="userCleanupRules" ref="userCleanupFormRef">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<el-form-item :label="$t('members.username')" prop="username">
|
|
|
|
|
|
<el-input
|
|
|
|
|
|
v-model="userCleanupForm.username"
|
|
|
|
|
|
:placeholder="$t('systemSettings.enterUsername')"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
clearable
|
|
|
|
|
|
/>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item>
|
|
|
|
|
|
<el-alert
|
2025-11-13 17:01:39 +08:00
|
|
|
|
:title="$t('systemSettings.warning')"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
type="warning"
|
|
|
|
|
|
:closable="false"
|
|
|
|
|
|
show-icon
|
|
|
|
|
|
>
|
|
|
|
|
|
<template #default>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<p>{{ $t('systemSettings.cleanupWarning') }}</p>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<ul>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<li>{{ $t('systemSettings.successTasksArchived') }}</li>
|
|
|
|
|
|
<li>{{ $t('systemSettings.failedTasksLogged') }}</li>
|
|
|
|
|
|
<li>{{ $t('systemSettings.originalTasksDeleted') }}</li>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</ul>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<p><strong>{{ $t('systemSettings.irreversibleWarning') }}</strong></p>
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
</el-alert>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
</div>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
|
2025-10-27 10:46:49 +08:00
|
|
|
|
<template #footer>
|
|
|
|
|
|
<div class="dialog-footer">
|
2025-11-13 17:01:39 +08:00
|
|
|
|
<el-button @click="handleCloseUserCleanupDialog">{{ $t('common.cancel') }}</el-button>
|
|
|
|
|
|
<el-button
|
|
|
|
|
|
type="danger"
|
|
|
|
|
|
@click="performUserCleanup"
|
2025-10-27 10:46:49 +08:00
|
|
|
|
:loading="loadingUserCleanup"
|
|
|
|
|
|
>
|
2025-11-13 17:01:39 +08:00
|
|
|
|
{{ $t('systemSettings.confirmCleanup') }}
|
2025-10-27 10:46:49 +08:00
|
|
|
|
</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-dialog>
|
2025-11-03 13:20:30 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
2025-10-23 09:59:54 +08:00
|
|
|
|
|
|
|
|
|
|
<script setup>
|
2025-11-07 19:09:50 +08:00
|
|
|
|
import { ref, reactive, onMounted } from 'vue'
|
2025-10-23 09:59:54 +08:00
|
|
|
|
import { useRouter } from 'vue-router'
|
|
|
|
|
|
import { ElMessage } from 'element-plus'
|
|
|
|
|
|
import {
|
|
|
|
|
|
Grid,
|
|
|
|
|
|
User,
|
|
|
|
|
|
ShoppingCart,
|
|
|
|
|
|
Document,
|
|
|
|
|
|
Setting,
|
|
|
|
|
|
User as Search,
|
2025-10-27 10:46:49 +08:00
|
|
|
|
User as ArrowDown,
|
|
|
|
|
|
Delete,
|
|
|
|
|
|
Refresh
|
2025-10-23 09:59:54 +08:00
|
|
|
|
} from '@element-plus/icons-vue'
|
2025-10-29 18:25:26 +08:00
|
|
|
|
import cleanupApi from '@/api/cleanup'
|
2025-11-07 19:09:50 +08:00
|
|
|
|
import { getMembershipLevels, updateMembershipLevel } from '@/api/members'
|
2025-11-13 17:01:39 +08:00
|
|
|
|
import LanguageSwitcher from '@/components/LanguageSwitcher.vue'
|
2025-10-23 09:59:54 +08:00
|
|
|
|
|
|
|
|
|
|
const router = useRouter()
|
|
|
|
|
|
|
2025-10-27 10:46:49 +08:00
|
|
|
|
// 选项卡状态
|
|
|
|
|
|
const activeTab = ref('membership')
|
|
|
|
|
|
|
2025-11-13 17:01:39 +08:00
|
|
|
|
// 系统状态数据
|
|
|
|
|
|
const onlineUsers = ref('0/500')
|
|
|
|
|
|
const systemUptime = ref('加载中...')
|
|
|
|
|
|
|
2025-10-27 10:46:49 +08:00
|
|
|
|
// 会员收费标准相关
|
2025-11-07 19:09:50 +08:00
|
|
|
|
const membershipLevels = ref([])
|
|
|
|
|
|
const loadingLevels = ref(false)
|
2025-10-23 09:59:54 +08:00
|
|
|
|
|
|
|
|
|
|
const editDialogVisible = ref(false)
|
|
|
|
|
|
const editFormRef = ref(null)
|
|
|
|
|
|
const editForm = reactive({
|
|
|
|
|
|
id: null,
|
|
|
|
|
|
level: '',
|
|
|
|
|
|
price: '',
|
|
|
|
|
|
resourcePoints: 0,
|
|
|
|
|
|
validityPeriod: 'quarterly'
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
const editRules = reactive({
|
|
|
|
|
|
level: [{ required: true, message: '请选择会员等级', trigger: 'change' }],
|
|
|
|
|
|
price: [
|
|
|
|
|
|
{ required: true, message: '请输入价格', trigger: 'blur' },
|
|
|
|
|
|
{ pattern: /^\d+(\.\d+)?$/, message: '请输入有效的数字', trigger: 'blur' }
|
|
|
|
|
|
],
|
|
|
|
|
|
resourcePoints: [{ required: true, message: '请输入资源点数量', trigger: 'blur' }],
|
|
|
|
|
|
validityPeriod: [{ required: true, message: '请选择有效期', trigger: 'change' }]
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2025-10-27 10:46:49 +08:00
|
|
|
|
// 任务清理相关
|
|
|
|
|
|
const cleanupStats = ref(null)
|
|
|
|
|
|
const loadingStats = ref(false)
|
|
|
|
|
|
const loadingCleanup = ref(false)
|
|
|
|
|
|
const loadingUserCleanup = ref(false)
|
|
|
|
|
|
const loadingConfig = ref(false)
|
|
|
|
|
|
|
|
|
|
|
|
const showUserCleanupDialog = ref(false)
|
|
|
|
|
|
const userCleanupFormRef = ref(null)
|
|
|
|
|
|
const userCleanupForm = reactive({
|
|
|
|
|
|
username: ''
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
const userCleanupRules = reactive({
|
|
|
|
|
|
username: [
|
|
|
|
|
|
{ required: true, message: '请输入用户名', trigger: 'blur' },
|
|
|
|
|
|
{ min: 2, max: 50, message: '用户名长度在2到50个字符', trigger: 'blur' }
|
|
|
|
|
|
]
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
const cleanupConfig = reactive({
|
|
|
|
|
|
retentionDays: 30,
|
|
|
|
|
|
archiveRetentionDays: 365
|
|
|
|
|
|
})
|
|
|
|
|
|
|
2025-12-05 09:57:09 +08:00
|
|
|
|
// AI模型设置相关
|
|
|
|
|
|
const promptOptimizationModel = ref('gpt-5.1-thinking')
|
|
|
|
|
|
const promptOptimizationApiUrl = ref('https://ai.comfly.chat')
|
|
|
|
|
|
const savingAiModel = ref(false)
|
|
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
const goToDashboard = () => {
|
2025-11-21 16:10:00 +08:00
|
|
|
|
router.push('/admin/dashboard')
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const goToMembers = () => {
|
|
|
|
|
|
router.push('/member-management')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const goToOrders = () => {
|
2025-11-21 16:10:00 +08:00
|
|
|
|
router.push('/admin/orders')
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const goToAPI = () => {
|
|
|
|
|
|
router.push('/api-management')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const goToTasks = () => {
|
|
|
|
|
|
router.push('/generate-task-record')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const goToSettings = () => {
|
|
|
|
|
|
router.push('/system-settings')
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-21 16:10:00 +08:00
|
|
|
|
// 处理用户头像下拉菜单
|
|
|
|
|
|
const handleUserCommand = (command) => {
|
|
|
|
|
|
if (command === 'exitAdmin') {
|
|
|
|
|
|
// 退出后台,返回个人首页
|
|
|
|
|
|
router.push('/profile')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
const editLevel = (level) => {
|
2025-11-07 19:09:50 +08:00
|
|
|
|
// 映射后端数据到前端表单
|
|
|
|
|
|
editForm.id = level.id
|
|
|
|
|
|
editForm.level = level.name || level.displayName
|
|
|
|
|
|
editForm.price = level.price ? String(level.price) : '0'
|
|
|
|
|
|
editForm.resourcePoints = level.pointsBonus || level.resourcePoints || 0
|
|
|
|
|
|
editForm.validityPeriod = 'monthly' // 默认月付
|
2025-10-23 09:59:54 +08:00
|
|
|
|
editDialogVisible.value = true
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const handleCloseEditDialog = () => {
|
|
|
|
|
|
editDialogVisible.value = false
|
|
|
|
|
|
if (editFormRef.value) {
|
|
|
|
|
|
editFormRef.value.resetFields()
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const handlePriceInput = (value) => {
|
|
|
|
|
|
// 确保只输入数字
|
|
|
|
|
|
editForm.price = value.replace(/[^\d.]/g, '')
|
2025-11-03 13:20:30 +08:00
|
|
|
|
}
|
2025-10-23 09:59:54 +08:00
|
|
|
|
|
|
|
|
|
|
const saveEdit = async () => {
|
|
|
|
|
|
const valid = await editFormRef.value.validate()
|
2025-11-07 19:09:50 +08:00
|
|
|
|
if (!valid) return
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
// 调用后端API更新会员等级配置
|
|
|
|
|
|
const updateData = {
|
|
|
|
|
|
price: parseFloat(editForm.price),
|
|
|
|
|
|
resourcePoints: parseInt(editForm.resourcePoints),
|
|
|
|
|
|
pointsBonus: parseInt(editForm.resourcePoints),
|
|
|
|
|
|
description: `包含${editForm.resourcePoints}资源点/月`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
await updateMembershipLevel(editForm.id, updateData)
|
|
|
|
|
|
|
|
|
|
|
|
// 更新本地数据
|
2025-10-23 09:59:54 +08:00
|
|
|
|
const index = membershipLevels.value.findIndex(level => level.id === editForm.id)
|
|
|
|
|
|
if (index !== -1) {
|
2025-11-07 19:09:50 +08:00
|
|
|
|
membershipLevels.value[index].price = parseFloat(editForm.price)
|
|
|
|
|
|
membershipLevels.value[index].pointsBonus = parseInt(editForm.resourcePoints)
|
|
|
|
|
|
membershipLevels.value[index].resourcePoints = parseInt(editForm.resourcePoints)
|
|
|
|
|
|
membershipLevels.value[index].description = `包含${editForm.resourcePoints}资源点/月`
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
2025-11-07 19:09:50 +08:00
|
|
|
|
|
|
|
|
|
|
ElMessage.success('会员等级更新成功')
|
|
|
|
|
|
editDialogVisible.value = false
|
|
|
|
|
|
|
|
|
|
|
|
// 重新加载会员等级配置
|
|
|
|
|
|
await loadMembershipLevels()
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('更新会员等级失败:', error)
|
|
|
|
|
|
ElMessage.error('更新会员等级失败: ' + (error.response?.data?.message || error.message))
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 加载会员等级配置
|
|
|
|
|
|
const loadMembershipLevels = async () => {
|
|
|
|
|
|
loadingLevels.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await getMembershipLevels()
|
|
|
|
|
|
console.log('会员等级配置响应:', response)
|
|
|
|
|
|
|
|
|
|
|
|
// 检查响应结构
|
|
|
|
|
|
let levels = []
|
|
|
|
|
|
if (response.data) {
|
|
|
|
|
|
if (response.data.success && response.data.data) {
|
|
|
|
|
|
levels = response.data.data
|
|
|
|
|
|
} else if (Array.isArray(response.data)) {
|
|
|
|
|
|
levels = response.data
|
|
|
|
|
|
} else if (response.data.data && Array.isArray(response.data.data)) {
|
|
|
|
|
|
levels = response.data.data
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
console.log('解析后的会员等级数据:', levels)
|
|
|
|
|
|
|
|
|
|
|
|
// 映射后端数据到前端显示格式
|
|
|
|
|
|
if (levels.length > 0) {
|
|
|
|
|
|
membershipLevels.value = levels.map(level => ({
|
|
|
|
|
|
id: level.id,
|
|
|
|
|
|
name: level.displayName || level.name,
|
|
|
|
|
|
price: level.price || 0,
|
|
|
|
|
|
resourcePoints: level.pointsBonus || 0,
|
|
|
|
|
|
pointsBonus: level.pointsBonus || 0,
|
|
|
|
|
|
description: level.description || `包含${level.pointsBonus || 0}资源点/月`
|
|
|
|
|
|
}))
|
|
|
|
|
|
console.log('会员等级配置加载成功:', membershipLevels.value)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 如果没有数据,使用默认值
|
|
|
|
|
|
console.warn('数据库中没有会员等级数据,使用默认值')
|
|
|
|
|
|
membershipLevels.value = [
|
|
|
|
|
|
{ id: 1, name: '免费版会员', price: 0, resourcePoints: 200, description: '包含200资源点/月' },
|
|
|
|
|
|
{ id: 2, name: '标准版会员', price: 59, resourcePoints: 500, description: '包含500资源点/月' },
|
|
|
|
|
|
{ id: 3, name: '专业版会员', price: 250, resourcePoints: 2000, description: '包含2000资源点/月' }
|
|
|
|
|
|
]
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('加载会员等级配置失败:', error)
|
|
|
|
|
|
console.error('错误详情:', error.response?.data || error.message)
|
|
|
|
|
|
|
|
|
|
|
|
// 显示更详细的错误信息
|
|
|
|
|
|
const errorMessage = error.response?.data?.message || error.response?.data?.error || error.message || '未知错误'
|
|
|
|
|
|
ElMessage.warning(`加载会员等级配置失败: ${errorMessage},使用默认配置`)
|
|
|
|
|
|
|
|
|
|
|
|
// 使用默认值,确保页面可以正常显示
|
|
|
|
|
|
membershipLevels.value = [
|
|
|
|
|
|
{ id: 1, name: '免费版会员', price: 0, resourcePoints: 200, description: '包含200资源点/月' },
|
|
|
|
|
|
{ id: 2, name: '标准版会员', price: 59, resourcePoints: 500, description: '包含500资源点/月' },
|
|
|
|
|
|
{ id: 3, name: '专业版会员', price: 250, resourcePoints: 2000, description: '包含2000资源点/月' }
|
|
|
|
|
|
]
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loadingLevels.value = false
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-10-27 10:46:49 +08:00
|
|
|
|
|
|
|
|
|
|
// 任务清理相关方法
|
|
|
|
|
|
const getAuthHeaders = () => {
|
|
|
|
|
|
const token = sessionStorage.getItem('token')
|
|
|
|
|
|
return token ? { 'Authorization': `Bearer ${token}` } : {}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const refreshStats = async () => {
|
|
|
|
|
|
loadingStats.value = true
|
|
|
|
|
|
try {
|
2025-10-29 18:25:26 +08:00
|
|
|
|
const response = await cleanupApi.getCleanupStats()
|
|
|
|
|
|
cleanupStats.value = response.data
|
|
|
|
|
|
ElMessage.success('统计信息刷新成功')
|
2025-10-27 10:46:49 +08:00
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('获取统计信息失败:', error)
|
|
|
|
|
|
ElMessage.error('获取统计信息失败')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loadingStats.value = false
|
|
|
|
|
|
}
|
2025-11-03 13:20:30 +08:00
|
|
|
|
|
2025-10-27 10:46:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const performFullCleanup = async () => {
|
|
|
|
|
|
loadingCleanup.value = true
|
|
|
|
|
|
try {
|
2025-10-29 18:25:26 +08:00
|
|
|
|
const response = await cleanupApi.performFullCleanup()
|
|
|
|
|
|
ElMessage.success('完整清理执行成功')
|
|
|
|
|
|
console.log('清理结果:', response.data)
|
|
|
|
|
|
// 刷新统计信息
|
|
|
|
|
|
await refreshStats()
|
2025-10-27 10:46:49 +08:00
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('执行完整清理失败:', error)
|
|
|
|
|
|
ElMessage.error('执行完整清理失败')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loadingCleanup.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const handleCloseUserCleanupDialog = () => {
|
|
|
|
|
|
showUserCleanupDialog.value = false
|
|
|
|
|
|
if (userCleanupFormRef.value) {
|
|
|
|
|
|
userCleanupFormRef.value.resetFields()
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const performUserCleanup = async () => {
|
|
|
|
|
|
const valid = await userCleanupFormRef.value.validate()
|
2025-10-29 18:25:26 +08:00
|
|
|
|
if (!valid) return
|
|
|
|
|
|
|
|
|
|
|
|
loadingUserCleanup.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await cleanupApi.cleanupUserTasks(userCleanupForm.username)
|
|
|
|
|
|
ElMessage.success('用户任务清理成功')
|
|
|
|
|
|
console.log('清理结果:', response.data)
|
|
|
|
|
|
// 刷新统计信息
|
|
|
|
|
|
await refreshStats()
|
|
|
|
|
|
// 关闭对话框
|
|
|
|
|
|
handleCloseUserCleanupDialog()
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('清理用户任务失败:', error)
|
|
|
|
|
|
ElMessage.error('清理用户任务失败')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loadingUserCleanup.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const performUserCleanup_old = async () => {
|
|
|
|
|
|
const valid = await userCleanupFormRef.value.validate()
|
2025-10-27 10:46:49 +08:00
|
|
|
|
if (!valid) return
|
|
|
|
|
|
|
|
|
|
|
|
loadingUserCleanup.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await fetch(`/api/cleanup/user-tasks/${userCleanupForm.username}`, {
|
|
|
|
|
|
method: 'POST',
|
|
|
|
|
|
headers: {
|
|
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
|
|
...getAuthHeaders()
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
|
|
const result = await response.json()
|
|
|
|
|
|
ElMessage.success(`用户 ${userCleanupForm.username} 的任务清理完成`)
|
|
|
|
|
|
console.log('用户清理结果:', result)
|
|
|
|
|
|
// 关闭对话框并刷新统计信息
|
|
|
|
|
|
handleCloseUserCleanupDialog()
|
|
|
|
|
|
await refreshStats()
|
|
|
|
|
|
} else {
|
|
|
|
|
|
ElMessage.error('清理用户任务失败')
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('清理用户任务失败:', error)
|
|
|
|
|
|
ElMessage.error('清理用户任务失败')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loadingUserCleanup.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const saveCleanupConfig = async () => {
|
|
|
|
|
|
loadingConfig.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
// 这里可以添加保存配置的API调用
|
|
|
|
|
|
// 目前只是模拟保存
|
|
|
|
|
|
await new Promise(resolve => setTimeout(resolve, 1000))
|
|
|
|
|
|
ElMessage.success('清理配置保存成功')
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('保存清理配置失败:', error)
|
|
|
|
|
|
ElMessage.error('保存清理配置失败')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loadingConfig.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-05 09:57:09 +08:00
|
|
|
|
// 加载AI模型设置
|
|
|
|
|
|
const loadAiModelSettings = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await fetch('/api/admin/settings')
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
|
|
const data = await response.json()
|
|
|
|
|
|
if (data.promptOptimizationModel) {
|
|
|
|
|
|
promptOptimizationModel.value = data.promptOptimizationModel
|
|
|
|
|
|
}
|
|
|
|
|
|
if (data.promptOptimizationApiUrl) {
|
|
|
|
|
|
promptOptimizationApiUrl.value = data.promptOptimizationApiUrl
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('加载AI模型设置失败:', error)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 保存AI模型设置
|
|
|
|
|
|
const saveAiModelSettings = async () => {
|
|
|
|
|
|
savingAiModel.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await fetch('/api/admin/settings', {
|
|
|
|
|
|
method: 'PUT',
|
|
|
|
|
|
headers: {
|
|
|
|
|
|
'Content-Type': 'application/json'
|
|
|
|
|
|
},
|
|
|
|
|
|
body: JSON.stringify({
|
|
|
|
|
|
promptOptimizationModel: promptOptimizationModel.value,
|
|
|
|
|
|
promptOptimizationApiUrl: promptOptimizationApiUrl.value
|
|
|
|
|
|
})
|
|
|
|
|
|
})
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
|
|
ElMessage.success('AI模型设置保存成功')
|
|
|
|
|
|
} else {
|
|
|
|
|
|
throw new Error('保存失败')
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('保存AI模型设置失败:', error)
|
|
|
|
|
|
ElMessage.error('保存AI模型设置失败')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
savingAiModel.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-07 19:09:50 +08:00
|
|
|
|
// 页面加载时获取统计信息和会员等级配置
|
|
|
|
|
|
onMounted(() => {
|
|
|
|
|
|
refreshStats()
|
|
|
|
|
|
loadMembershipLevels()
|
2025-11-13 17:01:39 +08:00
|
|
|
|
fetchSystemStats()
|
2025-12-05 09:57:09 +08:00
|
|
|
|
loadAiModelSettings()
|
2025-11-07 19:09:50 +08:00
|
|
|
|
})
|
2025-11-13 17:01:39 +08:00
|
|
|
|
|
|
|
|
|
|
// 获取系统统计数据
|
|
|
|
|
|
const fetchSystemStats = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
// 临时使用计算值,后续可以从API获取
|
|
|
|
|
|
// 计算在线用户数(这里简化处理)
|
|
|
|
|
|
const randomOnline = Math.floor(Math.random() * 50) + 10
|
|
|
|
|
|
onlineUsers.value = `${randomOnline}/500`
|
|
|
|
|
|
|
|
|
|
|
|
// 计算系统运行时间(基于当前时间简单模拟)
|
|
|
|
|
|
const hours = new Date().getHours()
|
|
|
|
|
|
const minutes = new Date().getMinutes()
|
|
|
|
|
|
systemUptime.value = `${hours}小时${minutes}分`
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('获取系统统计失败:', error)
|
|
|
|
|
|
onlineUsers.value = '0/500'
|
|
|
|
|
|
systemUptime.value = '未知'
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-10-23 09:59:54 +08:00
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.system-settings {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
height: 100vh;
|
|
|
|
|
|
background-color: #f5f7fa;
|
|
|
|
|
|
font-family: 'Arial', sans-serif;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 左侧导航栏 */
|
|
|
|
|
|
.sidebar {
|
2025-11-07 19:09:50 +08:00
|
|
|
|
width: 240px;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
background: white;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
border-right: 1px solid #e9ecef;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
padding: 24px 0;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
box-shadow: 2px 0 8px rgba(0, 0, 0, 0.05);
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.logo {
|
2025-11-13 17:01:39 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
2025-11-21 16:10:00 +08:00
|
|
|
|
padding: 0 24px;
|
|
|
|
|
|
margin-bottom: 32px;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-21 16:10:00 +08:00
|
|
|
|
.logo img {
|
2025-11-13 17:01:39 +08:00
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: auto;
|
2025-11-21 16:10:00 +08:00
|
|
|
|
max-width: 180px;
|
2025-11-13 17:01:39 +08:00
|
|
|
|
object-fit: contain;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-menu {
|
|
|
|
|
|
flex: 1;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
padding: 0 16px;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
padding: 12px 16px;
|
|
|
|
|
|
margin-bottom: 4px;
|
|
|
|
|
|
border-radius: 8px;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
transition: all 0.2s ease;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
color: #6b7280;
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
font-weight: 500;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-item:hover {
|
2025-11-07 19:09:50 +08:00
|
|
|
|
background: #f3f4f6;
|
|
|
|
|
|
color: #374151;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-item.active {
|
2025-11-07 19:09:50 +08:00
|
|
|
|
background: #dbeafe;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
color: #3b82f6;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-item .el-icon {
|
2025-11-07 19:09:50 +08:00
|
|
|
|
margin-right: 12px;
|
|
|
|
|
|
font-size: 18px;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-item span {
|
2025-11-07 19:09:50 +08:00
|
|
|
|
font-size: 14px;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.sidebar-footer {
|
2025-11-07 19:09:50 +08:00
|
|
|
|
padding: 20px;
|
|
|
|
|
|
border-top: 1px solid #e9ecef;
|
|
|
|
|
|
background: #f8f9fa;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
margin-top: auto;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.online-users,
|
|
|
|
|
|
.system-uptime {
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: #64748b;
|
|
|
|
|
|
margin-bottom: 5px;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
line-height: 1.5;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.highlight {
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.main-content {
|
|
|
|
|
|
flex-grow: 1;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
overflow-y: auto;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.top-header {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
padding: 15px 30px;
|
|
|
|
|
|
background-color: #fff;
|
|
|
|
|
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.05);
|
|
|
|
|
|
z-index: 100;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.search-bar {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
background-color: #f0f2f5;
|
|
|
|
|
|
border-radius: 20px;
|
|
|
|
|
|
padding: 8px 15px;
|
|
|
|
|
|
width: 300px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.search-icon {
|
|
|
|
|
|
color: #909399;
|
|
|
|
|
|
margin-right: 8px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.search-input {
|
|
|
|
|
|
border: none;
|
|
|
|
|
|
background: transparent;
|
|
|
|
|
|
outline: none;
|
|
|
|
|
|
flex-grow: 1;
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.header-actions {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
gap: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
.user-avatar {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
gap: 8px;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
cursor: pointer;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
padding: 4px 8px;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
transition: background 0.2s ease;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.user-avatar:hover {
|
|
|
|
|
|
background: #f3f4f6;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-07 19:09:50 +08:00
|
|
|
|
.user-avatar img {
|
2025-10-23 09:59:54 +08:00
|
|
|
|
width: 32px;
|
|
|
|
|
|
height: 32px;
|
|
|
|
|
|
border-radius: 50%;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
object-fit: cover;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-07 19:09:50 +08:00
|
|
|
|
.user-avatar .arrow-down {
|
2025-10-23 09:59:54 +08:00
|
|
|
|
font-size: 12px;
|
2025-11-07 19:09:50 +08:00
|
|
|
|
color: #6b7280;
|
2025-10-23 09:59:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.content-section {
|
|
|
|
|
|
flex-grow: 1;
|
|
|
|
|
|
padding: 30px;
|
|
|
|
|
|
background-color: #f5f7fa;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-27 10:46:49 +08:00
|
|
|
|
/* 设置选项卡样式 */
|
|
|
|
|
|
.settings-tabs {
|
|
|
|
|
|
flex-grow: 1;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tab-nav {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
background: white;
|
|
|
|
|
|
border-bottom: 1px solid #e2e8f0;
|
|
|
|
|
|
padding: 0 30px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tab-item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
padding: 20px 24px;
|
|
|
|
|
|
margin-right: 8px;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
border-bottom: 2px solid transparent;
|
|
|
|
|
|
transition: all 0.2s ease;
|
|
|
|
|
|
color: #64748b;
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tab-item:hover {
|
|
|
|
|
|
color: #334155;
|
|
|
|
|
|
background: #f8fafc;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tab-item.active {
|
|
|
|
|
|
color: #3b82f6;
|
|
|
|
|
|
border-bottom-color: #3b82f6;
|
|
|
|
|
|
background: #eff6ff;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tab-item .el-icon {
|
|
|
|
|
|
margin-right: 8px;
|
|
|
|
|
|
font-size: 18px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tab-content {
|
|
|
|
|
|
flex-grow: 1;
|
|
|
|
|
|
padding: 30px;
|
|
|
|
|
|
background-color: #f5f7fa;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 清理功能样式 */
|
|
|
|
|
|
.cleanup-stats,
|
|
|
|
|
|
.cleanup-actions,
|
|
|
|
|
|
.cleanup-config {
|
|
|
|
|
|
margin-bottom: 24px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stats-card,
|
|
|
|
|
|
.actions-card,
|
|
|
|
|
|
.config-card {
|
|
|
|
|
|
border-radius: 12px;
|
|
|
|
|
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.05);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-header {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
padding: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-header h3 {
|
|
|
|
|
|
margin: 0;
|
|
|
|
|
|
font-size: 18px;
|
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
|
color: #1e293b;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stats-content {
|
|
|
|
|
|
padding: 20px 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stats-grid {
|
|
|
|
|
|
display: grid;
|
|
|
|
|
|
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
|
|
|
|
|
|
gap: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-item {
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
padding: 20px;
|
|
|
|
|
|
background: #f8fafc;
|
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
|
border: 1px solid #e2e8f0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-label {
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: #64748b;
|
|
|
|
|
|
margin-bottom: 8px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-value {
|
|
|
|
|
|
font-size: 24px;
|
|
|
|
|
|
font-weight: 700;
|
|
|
|
|
|
color: #1e293b;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.actions-content {
|
|
|
|
|
|
padding: 20px 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.action-buttons {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
gap: 16px;
|
|
|
|
|
|
margin-bottom: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.action-btn {
|
|
|
|
|
|
min-width: 160px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.action-description {
|
|
|
|
|
|
padding: 16px;
|
|
|
|
|
|
background: #f8fafc;
|
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
|
border-left: 4px solid #3b82f6;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.action-description p {
|
|
|
|
|
|
margin: 0 0 8px 0;
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: #64748b;
|
|
|
|
|
|
line-height: 1.5;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.action-description p:last-child {
|
|
|
|
|
|
margin-bottom: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.config-content {
|
|
|
|
|
|
padding: 20px 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.config-tip {
|
|
|
|
|
|
margin-left: 12px;
|
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
|
color: #94a3b8;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 用户清理对话框样式 */
|
|
|
|
|
|
.user-cleanup-content {
|
|
|
|
|
|
padding: 20px 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.dialog-footer {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: flex-end;
|
|
|
|
|
|
gap: 12px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-23 09:59:54 +08:00
|
|
|
|
.page-title {
|
|
|
|
|
|
font-size: 24px;
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
margin-bottom: 25px;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.membership-cards {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
gap: 25px;
|
|
|
|
|
|
flex-wrap: wrap;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.membership-card {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
min-width: 280px;
|
|
|
|
|
|
max-width: 350px;
|
|
|
|
|
|
border-radius: 12px;
|
|
|
|
|
|
box-shadow: 0 6px 20px rgba(0, 0, 0, 0.08);
|
|
|
|
|
|
transition: all 0.3s ease;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.membership-card:hover {
|
|
|
|
|
|
transform: translateY(-5px);
|
|
|
|
|
|
box-shadow: 0 10px 30px rgba(0, 0, 0, 0.12);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-header {
|
|
|
|
|
|
padding: 20px;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
border-bottom: 1px solid #eee;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-header h3 {
|
|
|
|
|
|
font-size: 22px;
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
margin: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-body {
|
|
|
|
|
|
padding: 25px 20px;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
flex-grow: 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-body .price {
|
|
|
|
|
|
font-size: 36px;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
color: #409eff;
|
|
|
|
|
|
margin-bottom: 10px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-body .description {
|
|
|
|
|
|
font-size: 15px;
|
|
|
|
|
|
color: #606266;
|
|
|
|
|
|
line-height: 1.6;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-footer {
|
|
|
|
|
|
padding: 20px;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
border-top: 1px solid #eee;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.el-button {
|
|
|
|
|
|
width: 80%;
|
|
|
|
|
|
padding: 12px 0;
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 会员收费标准模态框样式 - 完全匹配HTML代码 */
|
|
|
|
|
|
.membership-modal {
|
|
|
|
|
|
border-radius: 12px;
|
|
|
|
|
|
box-shadow: 0 10px 30px rgba(0, 0, 0, 0.1);
|
|
|
|
|
|
border: 1px solid #e1e5eb;
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.membership-modal .el-dialog__body {
|
|
|
|
|
|
padding: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 弹窗头部 */
|
|
|
|
|
|
.modal-header {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
padding: 20px 24px;
|
|
|
|
|
|
border-bottom: 1px solid #f0f2f5;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.modal-title {
|
|
|
|
|
|
font-size: 18px;
|
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
|
color: #1a1a1a;
|
|
|
|
|
|
margin: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.close-btn {
|
|
|
|
|
|
width: 32px;
|
|
|
|
|
|
height: 32px;
|
|
|
|
|
|
border: none;
|
|
|
|
|
|
background: #f5f7fa;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
color: #666;
|
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.close-btn:hover {
|
|
|
|
|
|
background: #e6f3ff;
|
|
|
|
|
|
color: #1890ff;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 表单内容区域 */
|
|
|
|
|
|
.modal-content {
|
|
|
|
|
|
padding: 24px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.form-group {
|
|
|
|
|
|
margin-bottom: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.form-label {
|
|
|
|
|
|
display: block;
|
|
|
|
|
|
margin-bottom: 8px;
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 输入框和下拉框样式 */
|
|
|
|
|
|
.form-control {
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
height: 40px;
|
|
|
|
|
|
padding: 0 12px;
|
|
|
|
|
|
border: 1px solid #d9d9d9;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: #333;
|
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
background-color: white;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.form-control:focus {
|
|
|
|
|
|
outline: none;
|
|
|
|
|
|
border-color: #1890ff;
|
|
|
|
|
|
box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.form-control::placeholder {
|
|
|
|
|
|
color: #bfbfbf;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 价格输入框特殊样式 */
|
|
|
|
|
|
.price-input {
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.price-prefix {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
left: 12px;
|
|
|
|
|
|
top: 50%;
|
|
|
|
|
|
transform: translateY(-50%);
|
|
|
|
|
|
color: #666;
|
|
|
|
|
|
z-index: 10;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.price-input .form-control {
|
|
|
|
|
|
padding-left: 30px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 单选按钮组 */
|
|
|
|
|
|
.radio-group {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
gap: 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.radio-option {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.radio-input {
|
|
|
|
|
|
display: none;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.radio-label {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
color: #666;
|
|
|
|
|
|
padding: 8px 16px;
|
|
|
|
|
|
border: 1px solid #d9d9d9;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.radio-input:checked + .radio-label {
|
|
|
|
|
|
border-color: #1890ff;
|
|
|
|
|
|
background-color: #e6f7ff;
|
|
|
|
|
|
color: #1890ff;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 按钮区域 */
|
|
|
|
|
|
.modal-footer {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: flex-end;
|
|
|
|
|
|
gap: 12px;
|
|
|
|
|
|
padding: 20px 24px;
|
|
|
|
|
|
border-top: 1px solid #f0f2f5;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.btn {
|
|
|
|
|
|
padding: 0 20px;
|
|
|
|
|
|
height: 36px;
|
|
|
|
|
|
border: 1px solid;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.btn-cancel {
|
|
|
|
|
|
background: white;
|
|
|
|
|
|
border-color: #d9d9d9;
|
|
|
|
|
|
color: #666;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.btn-cancel:hover {
|
|
|
|
|
|
border-color: #1890ff;
|
|
|
|
|
|
color: #1890ff;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.btn-save {
|
|
|
|
|
|
background: #1890ff;
|
|
|
|
|
|
border-color: #1890ff;
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.btn-save:hover {
|
|
|
|
|
|
background: #40a9ff;
|
|
|
|
|
|
border-color: #40a9ff;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* 响应式调整 */
|
|
|
|
|
|
@media (max-width: 480px) {
|
|
|
|
|
|
.membership-modal {
|
|
|
|
|
|
border-radius: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.modal-content {
|
|
|
|
|
|
padding: 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.radio-group {
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
gap: 8px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Responsive adjustments */
|
|
|
|
|
|
@media (max-width: 1200px) {
|
|
|
|
|
|
.membership-card {
|
|
|
|
|
|
max-width: none;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@media (max-width: 768px) {
|
|
|
|
|
|
.sidebar {
|
|
|
|
|
|
width: 60px;
|
|
|
|
|
|
padding: 15px 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.sidebar .logo span,
|
|
|
|
|
|
.sidebar .nav-item span,
|
|
|
|
|
|
.sidebar-footer {
|
|
|
|
|
|
display: none;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.sidebar .logo {
|
|
|
|
|
|
padding: 0 10px 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.sidebar .logo-icon {
|
|
|
|
|
|
margin-right: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-menu {
|
|
|
|
|
|
padding: 0 10px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-item {
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
padding: 10px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.nav-item .el-icon {
|
|
|
|
|
|
margin-right: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.top-header {
|
|
|
|
|
|
padding: 10px 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.search-bar {
|
|
|
|
|
|
width: 200px;
|
|
|
|
|
|
padding: 6px 10px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.content-section {
|
|
|
|
|
|
padding: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.page-title {
|
|
|
|
|
|
font-size: 20px;
|
|
|
|
|
|
margin-bottom: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.membership-cards {
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.membership-card {
|
|
|
|
|
|
width: 90%;
|
|
|
|
|
|
max-width: 400px;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
</style>
|