Files
nexus-terminal/packages/frontend/src/components/ProxyList.vue
T
2025-05-06 20:36:57 +08:00

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>