80 lines
3.9 KiB
Vue
80 lines
3.9 KiB
Vue
<script setup lang="ts">
|
|
import { storeToRefs } from 'pinia';
|
|
import { useI18n } from 'vue-i18n';
|
|
import { useProxiesStore, ProxyInfo } from '../stores/proxies.store';
|
|
|
|
const { t } = useI18n();
|
|
const proxiesStore = useProxiesStore();
|
|
const { proxies, isLoading, error } = storeToRefs(proxiesStore);
|
|
|
|
// 定义组件发出的事件
|
|
const emit = defineEmits(['edit-proxy']);
|
|
|
|
// 处理删除代理的方法
|
|
const handleDelete = async (proxy: ProxyInfo) => {
|
|
const confirmMessage = t('proxies.prompts.confirmDelete', { name: proxy.name });
|
|
if (window.confirm(confirmMessage)) {
|
|
const success = await proxiesStore.deleteProxy(proxy.id);
|
|
if (!success) {
|
|
alert(t('proxies.errors.deleteFailed', { error: proxiesStore.error || '未知错误' }));
|
|
}
|
|
}
|
|
};
|
|
|
|
// 辅助函数:格式化时间戳 (可以考虑提取到公共工具函数)
|
|
const formatTimestamp = (timestamp: number | null): string => {
|
|
if (!timestamp) return '-';
|
|
return new Date(timestamp * 1000).toLocaleString();
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div class="mt-4"> <!-- Container with top margin -->
|
|
<!-- Loading state (Only show if loading AND no proxies are displayed yet) -->
|
|
<div v-if="isLoading && proxies.length === 0" class="p-4 border border-border rounded-md mb-4 text-text-secondary bg-header/50 text-center italic">
|
|
{{ t('proxies.loading') }}
|
|
</div>
|
|
<div v-else-if="error" class="p-4 mb-4 border-l-4 border-error bg-error/10 text-error rounded"> <!-- Error state consistent with Notifications -->
|
|
{{ t('proxies.error', { error: error }) }}
|
|
</div>
|
|
<div v-else-if="proxies.length === 0" class="p-4 mb-4 border-l-4 border-blue-400 bg-blue-100 text-blue-700 rounded"> <!-- No proxies state consistent with Notifications (using blue for now) -->
|
|
{{ t('proxies.noProxies') }}
|
|
</div>
|
|
|
|
<!-- Proxy List using Card Layout -->
|
|
<div v-else class="grid gap-4 mt-4">
|
|
<div v-for="proxy in proxies" :key="proxy.id" class="bg-background border border-border rounded-lg p-4 flex justify-between items-start gap-4 shadow-sm hover:shadow-md transition-shadow duration-200"> <!-- Proxy item card -->
|
|
<div class="flex-grow space-y-1"> <!-- Details section -->
|
|
<strong class="block font-semibold text-base text-foreground">{{ proxy.name }}</strong>
|
|
<div class="flex items-center space-x-2"> <!-- Type Badge -->
|
|
<span class="px-2 py-0.5 rounded-full text-xs font-semibold bg-header border border-border text-text-secondary uppercase tracking-wider">
|
|
{{ proxy.type }}
|
|
</span>
|
|
</div>
|
|
<div class="text-sm text-text-secondary"> <!-- Host & Port -->
|
|
<i class="fas fa-server mr-1 text-xs opacity-70"></i> {{ proxy.host }}:{{ proxy.port }}
|
|
</div>
|
|
<div v-if="proxy.username" class="text-sm text-text-secondary"> <!-- Username (optional) -->
|
|
<i class="fas fa-user mr-1 text-xs opacity-70"></i> {{ proxy.username }}
|
|
</div>
|
|
<div class="text-xs text-text-secondary pt-1"> <!-- Updated At -->
|
|
<i class="fas fa-clock mr-1 opacity-70"></i> {{ formatTimestamp(proxy.updated_at) }}
|
|
</div>
|
|
</div>
|
|
<div class="flex items-center flex-shrink-0 space-x-3 pt-1"> <!-- Actions section -->
|
|
<button @click="emit('edit-proxy', proxy)" class="text-link hover:text-link-hover text-sm font-medium hover:underline"> <!-- Edit button (link style) -->
|
|
<i class="fas fa-pencil-alt mr-1 text-xs"></i>{{ t('proxies.actions.edit') }}
|
|
</button>
|
|
<button @click="handleDelete(proxy)" class="text-error hover:text-error/80 text-sm font-medium hover:underline"> <!-- Delete button (error color) -->
|
|
<i class="fas fa-trash-alt mr-1 text-xs"></i>{{ t('proxies.actions.delete') }}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped>
|
|
/* Remove all scoped styles as they are now handled by Tailwind utility classes */
|
|
</style>
|