新增项目基础结构,包括Vue3、Pinia、Element Plus等核心依赖 添加路由配置和用户认证状态管理 实现销售数据看板、客户画像、团队管理等核心功能模块 集成图表库和API请求工具,完成基础样式配置
124 lines
6.3 KiB
Vue
124 lines
6.3 KiB
Vue
<template>
|
|
<div class="dashboard-card table-section">
|
|
<div class="card-header">
|
|
<h3>详细数据表格</h3>
|
|
</div>
|
|
<div class="data-table-container">
|
|
<!-- 筛选器 -->
|
|
<div class="table-filters">
|
|
<div class="filter-group"><label>中心:</label><select v-model="filters.department"><option value="">全部中心</option><option>销售一部</option><option>销售二部</option><option>销售三部</option></select></div>
|
|
<div class="filter-group"><label>高级经理:</label><select v-model="filters.position"><option value="">全部高级经理</option><option>经理1</option><option>经理2</option><option>经理3</option></select></div>
|
|
<div class="filter-group"><label>经理:</label><select v-model="filters.timeRange"><option value="today">经理1</option><option value="week">经理2</option><option value="month">经理3</option></select></div>
|
|
<!-- <div class="filter-group"><label>成交状态:</label><select v-model="filters.dealStatus"><option value="">全部</option><option>已成交</option><option>跟进中</option><option>已失效</option></select></div> -->
|
|
</div>
|
|
|
|
<!-- 数据表格 -->
|
|
<div class="data-table">
|
|
<table>
|
|
<thead>
|
|
<tr>
|
|
<th>人员</th>
|
|
<th @click="sortBy('dealRate')" class="sortable">成交率 <span class="sort-icon" :class="{ active: sortField === 'dealRate' }">{{ sortOrder === 'desc' ? '↓' : '↑' }}</span></th>
|
|
<th @click="sortBy('callDuration')" class="sortable">通话时长 <span class="sort-icon" :class="{ active: sortField === 'callDuration' }">{{ sortOrder === 'desc' ? '↓' : '↑' }}</span></th>
|
|
<th>通话次数</th>
|
|
<th>成交金额</th>
|
|
<th>部门</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr v-for="person in filteredTableData" :key="person.id" @click="$emit('update:selectedPerson', person)" :class="{ selected: selectedPerson && selectedPerson.id === person.id }">
|
|
<td>
|
|
<div class="person-info">
|
|
<div class="person-avatar">{{ person.name.charAt(0) }}</div>
|
|
<div>
|
|
<div class="person-name">{{ person.name }}</div>
|
|
<div class="person-position">{{ person.position }}</div>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<td>
|
|
<div class="deal-rate">
|
|
<span class="rate-value" :class="getRateClass(person.dealRate)">{{ person.dealRate }}%</span>
|
|
<div class="rate-bar"><div class="rate-fill" :style="{ width: person.dealRate + '%', backgroundColor: getRateColor(person.dealRate) }"></div></div>
|
|
</div>
|
|
</td>
|
|
<td>{{ formatDuration(person.callDuration) }}</td>
|
|
<td>{{ person.callCount }}次</td>
|
|
<td>¥{{ person.dealAmount.toLocaleString() }}</td>
|
|
<td>{{ person.department }}</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { ref, computed } from 'vue';
|
|
|
|
const props = defineProps({
|
|
tableData: { type: Array, required: true },
|
|
selectedPerson: { type: Object, default: null }
|
|
});
|
|
defineEmits(['update:selectedPerson']);
|
|
|
|
const filters = ref({ department: '', position: '', timeRange: 'month', dealStatus: '' });
|
|
const sortField = ref('dealRate');
|
|
const sortOrder = ref('desc');
|
|
|
|
const filteredTableData = computed(() => {
|
|
let filtered = props.tableData;
|
|
if (filters.value.department) filtered = filtered.filter(item => item.department === filters.value.department);
|
|
if (filters.value.position) filtered = filtered.filter(item => item.position === filters.value.position);
|
|
return filtered.sort((a, b) => {
|
|
const aValue = a[sortField.value], bValue = b[sortField.value];
|
|
return sortOrder.value === 'desc' ? bValue - aValue : aValue - bValue;
|
|
});
|
|
});
|
|
|
|
const sortBy = (field) => {
|
|
if (sortField.value === field) sortOrder.value = sortOrder.value === 'desc' ? 'asc' : 'desc';
|
|
else { sortField.value = field; sortOrder.value = 'desc'; }
|
|
};
|
|
const getRateClass = (rate) => {
|
|
if (rate >= 80) return 'high'; if (rate >= 60) return 'medium'; return 'low';
|
|
};
|
|
const getRateColor = (rate) => {
|
|
if (rate >= 80) return '#4CAF50'; if (rate >= 60) return '#FF9800'; return '#f44336';
|
|
};
|
|
const formatDuration = (minutes) => {
|
|
const h = Math.floor(minutes / 60), m = minutes % 60;
|
|
return h > 0 ? `${h}h${m}m` : `${m}m`;
|
|
};
|
|
</script>
|
|
|
|
<style scoped>
|
|
.table-section { height: 600px; }
|
|
.data-table-container { height: calc(100% - 60px); overflow-y: auto; padding: 24px; }
|
|
.table-filters { display: grid; grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); gap: 16px; margin-bottom: 24px; padding: 16px; background: #f8fafc; border-radius: 8px; }
|
|
.filter-group { display: flex; flex-direction: column; gap: 4px; }
|
|
.filter-group label { font-size: 12px; font-weight: 600; color: #4a5568; }
|
|
.filter-group select { padding: 8px 12px; border: 1px solid #e2e8f0; border-radius: 6px; }
|
|
.data-table { overflow-x: auto; border-radius: 8px; border: 1px solid #e2e8f0; }
|
|
table { width: 100%; border-collapse: collapse; }
|
|
th { background: #f7fafc; padding: 12px 16px; text-align: left; font-weight: 600; color: #4a5568; border-bottom: 1px solid #e2e8f0; font-size: 12px; }
|
|
th.sortable { cursor: pointer; }
|
|
.sort-icon { margin-left: 4px; opacity: 0.5; }
|
|
.sort-icon.active { opacity: 1; color: #4299e1; }
|
|
td { padding: 16px; border-bottom: 1px solid #f1f5f9; vertical-align: middle; }
|
|
tr { cursor: pointer; transition: background-color 0.2s ease; }
|
|
tr:hover { background-color: #f8fafc; }
|
|
tr.selected { background-color: #e0f2fe; border-left: 4px solid #0ea5e9; }
|
|
.person-info { display: flex; align-items: center; gap: 12px; }
|
|
.person-avatar { width: 40px; height: 40px; border-radius: 50%; background: #4299e1; color: white; display: flex; align-items: center; justify-content: center; font-weight: 600; }
|
|
.person-name { font-weight: 600; }
|
|
.person-position { font-size: 12px; color: #718096; }
|
|
.deal-rate { min-width: 80px; }
|
|
.rate-value { font-weight: 600; }
|
|
.rate-value.high { color: #4CAF50; }
|
|
.rate-value.medium { color: #FF9800; }
|
|
.rate-value.low { color: #f44336; }
|
|
.rate-bar { height: 4px; background: #edf2f7; border-radius: 2px; overflow: hidden; }
|
|
.rate-fill { height: 100%; }
|
|
</style> |