修复下一页图标不显示问题,增加缓存前端配置页面
This commit is contained in:
@@ -0,0 +1,633 @@
|
||||
<template>
|
||||
<div class="settings-widget">
|
||||
<!-- 设置按钮 -->
|
||||
<button @click="toggleSettings" class="settings-toggle" :class="{ active: isOpen }" title="设置">
|
||||
<svg viewBox="0 0 24 24" fill="currentColor" class="settings-icon">
|
||||
<path
|
||||
d="M19.14,12.94c0.04-0.3,0.06-0.61,0.06-0.94c0-0.32-0.02-0.64-0.07-0.94l2.03-1.58c0.18-0.14,0.23-0.41,0.12-0.61 l-1.92-3.32c-0.12-0.22-0.37-0.29-0.59-0.22l-2.39,0.96c-0.5-0.38-1.03-0.7-1.62-0.94L14.4,2.81c-0.04-0.24-0.24-0.41-0.48-0.41 h-3.84c-0.24,0-0.43,0.17-0.47,0.41L9.25,5.35C8.66,5.59,8.12,5.92,7.63,6.29L5.24,5.33c-0.22-0.08-0.47,0-0.59,0.22L2.74,8.87 C2.62,9.08,2.66,9.34,2.86,9.48l2.03,1.58C4.84,11.36,4.8,11.69,4.8,12s0.02,0.64,0.07,0.94l-2.03,1.58 c-0.18,0.14-0.23,0.41-0.12,0.61l1.92,3.32c0.12,0.22,0.37,0.29,0.59,0.22l2.39-0.96c0.5,0.38,1.03,0.7,1.62,0.94l0.36,2.54 c0.05,0.24,0.24,0.41,0.48,0.41h3.84c0.24,0,0.44-0.17,0.47-0.41l0.36-2.54c0.59-0.24,1.13-0.56,1.62-0.94l2.39,0.96 c0.22,0.08,0.47,0,0.59-0.22l1.92-3.32c0.12-0.22,0.07-0.47-0.12-0.61L19.14,12.94z M12,15.6c-1.98,0-3.6-1.62-3.6-3.6 s1.62-3.6,3.6-3.6s3.6,1.62,3.6,3.6S13.98,15.6,12,15.6z" />
|
||||
</svg>
|
||||
</button>
|
||||
|
||||
<!-- 设置面板 -->
|
||||
<div v-if="isOpen" class="settings-panel">
|
||||
<div class="settings-header">
|
||||
<h3>设置</h3>
|
||||
<button @click="toggleSettings" class="close-btn" title="关闭">
|
||||
<svg viewBox="0 0 24 24" fill="currentColor" class="close-icon">
|
||||
<path
|
||||
d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z" />
|
||||
</svg>
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div class="settings-content">
|
||||
<!-- 缓存设置 -->
|
||||
<div class="settings-section">
|
||||
<h4>缓存设置</h4>
|
||||
|
||||
<div v-if="loading" class="loading">
|
||||
<LoadingSpinner text="加载中..." />
|
||||
</div>
|
||||
|
||||
<div v-else-if="error" class="error">
|
||||
<ErrorMessage :error="error" @dismiss="clearError" />
|
||||
</div>
|
||||
|
||||
<!-- 成功提示 -->
|
||||
<div v-if="successMessage" class="success-message">
|
||||
<div class="success-content">
|
||||
<svg viewBox="0 0 24 24" fill="currentColor" class="success-icon">
|
||||
<path d="M9 16.17L4.83 12l-1.42 1.41L9 19 21 7l-1.41-1.41z" />
|
||||
</svg>
|
||||
<span>{{ successMessage }}</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div v-else class="cache-settings">
|
||||
<!-- 缓存统计 -->
|
||||
<div class="cache-stats">
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">缓存文件数:</span>
|
||||
<span class="stat-value">{{ stats?.fileCount || 0 }}</span>
|
||||
</div>
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">缓存大小:</span>
|
||||
<span class="stat-value">{{ formatFileSize(stats?.totalSize || 0) }}</span>
|
||||
</div>
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">使用率:</span>
|
||||
<span class="stat-value">{{ Math.round((stats?.usagePercentage || 0) * 100) }}%</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 缓存配置 -->
|
||||
<div class="config-form">
|
||||
<div class="form-group">
|
||||
<label>
|
||||
<input type="checkbox" v-model="config.enabled" />
|
||||
启用缓存
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label>最大缓存大小 (MB)</label>
|
||||
<input type="number" v-model.number="maxSizeMB" min="10" max="10000" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label>缓存过期时间 (小时)</label>
|
||||
<input type="number" v-model.number="maxAgeHours" min="1" max="720" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label>清理间隔 (小时)</label>
|
||||
<input type="number" v-model.number="cleanupIntervalHours" min="1" max="24" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label>
|
||||
<input type="checkbox" v-model="config.proxy.enabled" />
|
||||
启用代理
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label>代理超时 (秒)</label>
|
||||
<input type="number" v-model.number="timeoutSeconds" min="5" max="300" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label>重试次数</label>
|
||||
<input type="number" v-model.number="config.proxy.retryCount" min="0" max="10" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 缓存操作 -->
|
||||
<div class="cache-actions">
|
||||
<button @click="saveConfig" class="btn btn-primary" :disabled="saving">
|
||||
{{ saving ? '保存中...' : '保存配置' }}
|
||||
</button>
|
||||
<button @click="clearExpiredCache" class="btn btn-secondary" :disabled="clearing">
|
||||
{{ clearing ? '清理中...' : '清理过期缓存' }}
|
||||
</button>
|
||||
<button @click="clearAllCache" class="btn btn-danger" :disabled="clearing">
|
||||
{{ clearing ? '清理中...' : '清理所有缓存' }}
|
||||
</button>
|
||||
<button @click="resetConfig" class="btn btn-warning" :disabled="resetting">
|
||||
{{ resetting ? '重置中...' : '重置配置' }}
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import cacheService, { type CacheConfig, type CacheStats } from '@/services/cache';
|
||||
import LoadingSpinner from './LoadingSpinner.vue';
|
||||
import ErrorMessage from './ErrorMessage.vue';
|
||||
|
||||
// 状态
|
||||
const isOpen = ref(false);
|
||||
const loading = ref(false);
|
||||
const error = ref<string | null>(null);
|
||||
const clearing = ref(false);
|
||||
const resetting = ref(false);
|
||||
const saving = ref(false);
|
||||
const successMessage = ref<string | null>(null);
|
||||
|
||||
// 数据
|
||||
const stats = ref<CacheStats | null>(null);
|
||||
const config = ref<CacheConfig>({
|
||||
maxAge: 24 * 60 * 60 * 1000,
|
||||
maxSize: 100 * 1024 * 1024,
|
||||
cleanupInterval: 60 * 60 * 1000,
|
||||
enabled: true,
|
||||
proxy: {
|
||||
enabled: true,
|
||||
timeout: 30000,
|
||||
retryCount: 3,
|
||||
retryDelay: 1000,
|
||||
},
|
||||
allowedExtensions: ['.jpg', '.jpeg', '.png', '.gif', '.webp', '.bmp'],
|
||||
lastUpdated: new Date().toISOString()
|
||||
});
|
||||
|
||||
// 计算属性 - 转换为用户友好的单位
|
||||
const maxSizeMB = computed({
|
||||
get: () => Math.round(config.value.maxSize / (1024 * 1024)),
|
||||
set: (value) => {
|
||||
config.value.maxSize = value * 1024 * 1024;
|
||||
}
|
||||
});
|
||||
|
||||
const maxAgeHours = computed({
|
||||
get: () => Math.round(config.value.maxAge / (1000 * 60 * 60)),
|
||||
set: (value) => {
|
||||
config.value.maxAge = value * 1000 * 60 * 60;
|
||||
}
|
||||
});
|
||||
|
||||
const cleanupIntervalHours = computed({
|
||||
get: () => Math.round(config.value.cleanupInterval / (1000 * 60 * 60)),
|
||||
set: (value) => {
|
||||
config.value.cleanupInterval = value * 1000 * 60 * 60;
|
||||
}
|
||||
});
|
||||
|
||||
const timeoutSeconds = computed({
|
||||
get: () => Math.round(config.value.proxy.timeout / 1000),
|
||||
set: (value) => {
|
||||
config.value.proxy.timeout = value * 1000;
|
||||
}
|
||||
});
|
||||
|
||||
// 方法
|
||||
const toggleSettings = () => {
|
||||
isOpen.value = !isOpen.value;
|
||||
if (isOpen.value) {
|
||||
loadData();
|
||||
}
|
||||
};
|
||||
|
||||
const loadData = async () => {
|
||||
try {
|
||||
loading.value = true;
|
||||
error.value = null;
|
||||
|
||||
// 并行加载配置和统计信息
|
||||
const [configResponse, statsResponse] = await Promise.all([
|
||||
cacheService.getCacheConfig(),
|
||||
cacheService.getCacheStats()
|
||||
]);
|
||||
|
||||
if (configResponse.success && configResponse.data) {
|
||||
config.value = configResponse.data;
|
||||
}
|
||||
|
||||
if (statsResponse.success && statsResponse.data) {
|
||||
stats.value = statsResponse.data;
|
||||
}
|
||||
|
||||
// 标记初始加载完成
|
||||
isInitialLoad.value = false;
|
||||
} catch (err) {
|
||||
error.value = err instanceof Error ? err.message : '加载设置失败';
|
||||
console.error('加载设置失败:', err);
|
||||
} finally {
|
||||
loading.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const saveConfig = async () => {
|
||||
try {
|
||||
saving.value = true;
|
||||
const response = await cacheService.updateCacheConfig(config.value);
|
||||
if (response.success && response.data) {
|
||||
config.value = response.data;
|
||||
showSuccess('配置已保存');
|
||||
}
|
||||
} catch (err) {
|
||||
error.value = err instanceof Error ? err.message : '保存配置失败';
|
||||
console.error('保存配置失败:', err);
|
||||
} finally {
|
||||
saving.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const clearExpiredCache = async () => {
|
||||
try {
|
||||
clearing.value = true;
|
||||
const response = await cacheService.clearExpiredCache();
|
||||
if (response.success) {
|
||||
// 重新加载统计信息
|
||||
await loadData();
|
||||
}
|
||||
} catch (err) {
|
||||
error.value = err instanceof Error ? err.message : '清理过期缓存失败';
|
||||
console.error('清理过期缓存失败:', err);
|
||||
} finally {
|
||||
clearing.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const clearAllCache = async () => {
|
||||
if (!confirm('确定要清理所有缓存吗?这将删除所有缓存的图片文件。')) {
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
clearing.value = true;
|
||||
const response = await cacheService.clearAllCache();
|
||||
if (response.success) {
|
||||
// 重新加载统计信息
|
||||
await loadData();
|
||||
}
|
||||
} catch (err) {
|
||||
error.value = err instanceof Error ? err.message : '清理所有缓存失败';
|
||||
console.error('清理所有缓存失败:', err);
|
||||
} finally {
|
||||
clearing.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const resetConfig = async () => {
|
||||
if (!confirm('确定要重置缓存配置为默认值吗?')) {
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
resetting.value = true;
|
||||
const response = await cacheService.resetCacheConfig();
|
||||
if (response.success && response.data) {
|
||||
config.value = response.data;
|
||||
}
|
||||
} catch (err) {
|
||||
error.value = err instanceof Error ? err.message : '重置配置失败';
|
||||
console.error('重置配置失败:', err);
|
||||
} finally {
|
||||
resetting.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const clearError = () => {
|
||||
error.value = null;
|
||||
};
|
||||
|
||||
const showSuccess = (message: string) => {
|
||||
successMessage.value = message;
|
||||
setTimeout(() => {
|
||||
successMessage.value = null;
|
||||
}, 3000);
|
||||
};
|
||||
|
||||
const formatFileSize = (bytes: number): string => {
|
||||
if (bytes === 0) return '0 B';
|
||||
|
||||
const k = 1024;
|
||||
const sizes = ['B', 'KB', 'MB', 'GB'];
|
||||
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
||||
|
||||
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
|
||||
};
|
||||
|
||||
// 是否正在加载初始数据
|
||||
const isInitialLoad = ref(true);
|
||||
|
||||
onMounted(() => {
|
||||
// 初始加载数据
|
||||
loadData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.settings-widget {
|
||||
position: fixed;
|
||||
bottom: 2rem;
|
||||
right: 2rem;
|
||||
z-index: 1000;
|
||||
}
|
||||
|
||||
.settings-toggle {
|
||||
width: 3.5rem;
|
||||
height: 3.5rem;
|
||||
border-radius: 50%;
|
||||
background: #3b82f6;
|
||||
color: white;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
box-shadow: 0 4px 12px rgba(59, 130, 246, 0.3);
|
||||
transition: all 0.3s ease;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.settings-toggle:hover {
|
||||
background: #2563eb;
|
||||
transform: translateY(-2px);
|
||||
box-shadow: 0 6px 16px rgba(59, 130, 246, 0.4);
|
||||
}
|
||||
|
||||
.settings-toggle.active {
|
||||
background: #1d4ed8;
|
||||
}
|
||||
|
||||
.settings-icon {
|
||||
width: 1.5rem;
|
||||
height: 1.5rem;
|
||||
}
|
||||
|
||||
.settings-panel {
|
||||
position: absolute;
|
||||
bottom: 4rem;
|
||||
right: 0;
|
||||
width: 400px;
|
||||
max-height: 600px;
|
||||
background: white;
|
||||
border-radius: 1rem;
|
||||
box-shadow: 0 10px 25px rgba(0, 0, 0, 0.15);
|
||||
border: 1px solid #e5e7eb;
|
||||
overflow: hidden;
|
||||
animation: slideUp 0.3s ease-out;
|
||||
}
|
||||
|
||||
@keyframes slideUp {
|
||||
from {
|
||||
opacity: 0;
|
||||
transform: translateY(20px);
|
||||
}
|
||||
|
||||
to {
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
}
|
||||
}
|
||||
|
||||
.settings-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding: 1.5rem;
|
||||
border-bottom: 1px solid #e5e7eb;
|
||||
background: #f9fafb;
|
||||
}
|
||||
|
||||
.settings-header h3 {
|
||||
margin: 0;
|
||||
font-size: 1.25rem;
|
||||
font-weight: 600;
|
||||
color: #1f2937;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
padding: 0.5rem;
|
||||
border-radius: 0.375rem;
|
||||
color: #6b7280;
|
||||
transition: all 0.2s;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #e5e7eb;
|
||||
color: #374151;
|
||||
}
|
||||
|
||||
.close-icon {
|
||||
width: 1.25rem;
|
||||
height: 1.25rem;
|
||||
}
|
||||
|
||||
.settings-content {
|
||||
padding: 1.5rem;
|
||||
max-height: 500px;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.settings-section {
|
||||
margin-bottom: 2rem;
|
||||
}
|
||||
|
||||
.settings-section h4 {
|
||||
margin: 0 0 1rem 0;
|
||||
font-size: 1.125rem;
|
||||
font-weight: 600;
|
||||
color: #1f2937;
|
||||
}
|
||||
|
||||
.loading {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
padding: 2rem;
|
||||
}
|
||||
|
||||
.error {
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
.success-message {
|
||||
background: #10b981;
|
||||
color: white;
|
||||
padding: 0.75rem 1rem;
|
||||
border-radius: 0.5rem;
|
||||
margin-bottom: 1rem;
|
||||
animation: slideIn 0.3s ease-out;
|
||||
}
|
||||
|
||||
.success-content {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 0.5rem;
|
||||
}
|
||||
|
||||
.success-icon {
|
||||
width: 1.25rem;
|
||||
height: 1.25rem;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
@keyframes slideIn {
|
||||
from {
|
||||
opacity: 0;
|
||||
transform: translateY(-10px);
|
||||
}
|
||||
|
||||
to {
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
}
|
||||
}
|
||||
|
||||
.cache-stats {
|
||||
background: #f9fafb;
|
||||
border-radius: 0.5rem;
|
||||
padding: 1rem;
|
||||
margin-bottom: 1.5rem;
|
||||
border: 1px solid #e5e7eb;
|
||||
}
|
||||
|
||||
.stat-item {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 0.5rem;
|
||||
}
|
||||
|
||||
.stat-item:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.stat-label {
|
||||
color: #6b7280;
|
||||
font-size: 0.875rem;
|
||||
}
|
||||
|
||||
.stat-value {
|
||||
font-weight: 600;
|
||||
color: #1f2937;
|
||||
}
|
||||
|
||||
.config-form {
|
||||
margin-bottom: 1.5rem;
|
||||
}
|
||||
|
||||
.form-group {
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
display: block;
|
||||
margin-bottom: 0.5rem;
|
||||
font-weight: 500;
|
||||
color: #374151;
|
||||
font-size: 0.875rem;
|
||||
}
|
||||
|
||||
.form-group input[type="number"],
|
||||
.form-group input[type="text"] {
|
||||
width: 100%;
|
||||
padding: 0.5rem 0.75rem;
|
||||
border: 1px solid #d1d5db;
|
||||
border-radius: 0.375rem;
|
||||
font-size: 0.875rem;
|
||||
transition: border-color 0.2s;
|
||||
}
|
||||
|
||||
.form-group input[type="number"]:focus,
|
||||
.form-group input[type="text"]:focus {
|
||||
outline: none;
|
||||
border-color: #3b82f6;
|
||||
box-shadow: 0 0 0 3px rgba(59, 130, 246, 0.1);
|
||||
}
|
||||
|
||||
.form-group input[type="checkbox"] {
|
||||
margin-right: 0.5rem;
|
||||
}
|
||||
|
||||
.cache-actions {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 0.75rem;
|
||||
}
|
||||
|
||||
.btn {
|
||||
padding: 0.75rem 1rem;
|
||||
border-radius: 0.5rem;
|
||||
font-weight: 600;
|
||||
text-decoration: none;
|
||||
transition: all 0.2s;
|
||||
border: none;
|
||||
cursor: pointer;
|
||||
font-size: 0.875rem;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.btn:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: #3b82f6;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-primary:hover:not(:disabled) {
|
||||
background: #2563eb;
|
||||
}
|
||||
|
||||
.btn-secondary {
|
||||
background: #f3f4f6;
|
||||
color: #374151;
|
||||
border: 1px solid #d1d5db;
|
||||
}
|
||||
|
||||
.btn-secondary:hover:not(:disabled) {
|
||||
background: #e5e7eb;
|
||||
}
|
||||
|
||||
.btn-danger {
|
||||
background: #ef4444;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-danger:hover:not(:disabled) {
|
||||
background: #dc2626;
|
||||
}
|
||||
|
||||
.btn-warning {
|
||||
background: #f59e0b;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-warning:hover:not(:disabled) {
|
||||
background: #d97706;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.settings-widget {
|
||||
bottom: 1rem;
|
||||
right: 1rem;
|
||||
}
|
||||
|
||||
.settings-panel {
|
||||
width: calc(100vw - 2rem);
|
||||
max-width: 400px;
|
||||
bottom: 3.5rem;
|
||||
}
|
||||
|
||||
.settings-content {
|
||||
padding: 1rem;
|
||||
}
|
||||
|
||||
.cache-actions {
|
||||
flex-direction: column;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user