240 lines
8.0 KiB
Vue
240 lines
8.0 KiB
Vue
<template>
|
|
<div class="booking-list">
|
|
<el-card>
|
|
<template #header>
|
|
<div class="card-header">
|
|
<span class="card-title">预约管理</span>
|
|
</div>
|
|
</template>
|
|
|
|
<el-form :inline="true" :model="queryForm" class="search-form">
|
|
<el-form-item label="预约编号">
|
|
<el-input v-model="queryForm.bookingNumber" placeholder="请输入预约编号" clearable />
|
|
</el-form-item>
|
|
<el-form-item label="联系人">
|
|
<el-input v-model="queryForm.contactName" placeholder="请输入联系人" clearable />
|
|
</el-form-item>
|
|
<el-form-item label="状态">
|
|
<el-select v-model="queryForm.status" placeholder="请选择" clearable>
|
|
<el-option label="待确认" value="pending" />
|
|
<el-option label="已确认" value="confirmed" />
|
|
<el-option label="进行中" value="in_progress" />
|
|
<el-option label="已完成" value="completed" />
|
|
<el-option label="已取消" value="cancelled" />
|
|
</el-select>
|
|
</el-form-item>
|
|
<el-form-item>
|
|
<el-button type="primary" :icon="Search" @click="handleSearch">搜索</el-button>
|
|
<el-button :icon="RefreshLeft" @click="handleReset">重置</el-button>
|
|
</el-form-item>
|
|
</el-form>
|
|
|
|
<el-table :data="tableData" v-loading="loading" stripe>
|
|
<el-table-column prop="bookingNumber" label="预约编号" width="160" />
|
|
<el-table-column prop="contactName" label="联系人" width="100" />
|
|
<el-table-column prop="contactPhone" label="联系电话" width="130" />
|
|
<el-table-column prop="service.name" label="服务类型" min-width="120" />
|
|
<el-table-column prop="address" label="地址" min-width="150" show-overflow-tooltip />
|
|
<el-table-column prop="appointmentTime" label="预约时间" width="170">
|
|
<template #default="{ row }">
|
|
{{ formatDate(row.appointmentTime) }}
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column prop="status" label="状态" width="100">
|
|
<template #default="{ row }">
|
|
<el-tag :type="getStatusType(row.status)">{{ getStatusText(row.status) }}</el-tag>
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column label="操作" width="220" fixed="right">
|
|
<template #default="{ row }">
|
|
<el-button type="primary" link :icon="View" @click="handleView(row)">查看</el-button>
|
|
<el-dropdown @command="(cmd) => handleStatusCommand(cmd, row)">
|
|
<el-button type="primary" link>
|
|
更新状态 <el-icon class="el-icon--right"><arrow-down /></el-icon>
|
|
</el-button>
|
|
<template #dropdown>
|
|
<el-dropdown-menu>
|
|
<el-dropdown-item command="confirmed">确认</el-dropdown-item>
|
|
<el-dropdown-item command="in_progress">进行中</el-dropdown-item>
|
|
<el-dropdown-item command="completed">完成</el-dropdown-item>
|
|
<el-dropdown-item command="cancelled">取消</el-dropdown-item>
|
|
</el-dropdown-menu>
|
|
</template>
|
|
</el-dropdown>
|
|
</template>
|
|
</el-table-column>
|
|
</el-table>
|
|
|
|
<el-pagination
|
|
v-model:current-page="queryForm.page"
|
|
v-model:page-size="queryForm.limit"
|
|
:page-sizes="[10, 20, 50, 100]"
|
|
:total="total"
|
|
layout="total, sizes, prev, pager, next, jumper"
|
|
@size-change="loadData"
|
|
@current-change="loadData"
|
|
style="margin-top: 20px; justify-content: center"
|
|
/>
|
|
</el-card>
|
|
|
|
<el-dialog v-model="detailVisible" title="预约详情" width="700px">
|
|
<el-descriptions :column="2" border v-if="currentRow">
|
|
<el-descriptions-item label="预约编号">{{ currentRow.bookingNumber }}</el-descriptions-item>
|
|
<el-descriptions-item label="状态">
|
|
<el-tag :type="getStatusType(currentRow.status)">{{ getStatusText(currentRow.status) }}</el-tag>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="联系人">{{ currentRow.contactName }}</el-descriptions-item>
|
|
<el-descriptions-item label="联系电话">{{ currentRow.contactPhone }}</el-descriptions-item>
|
|
<el-descriptions-item label="服务类型" :span="2">{{ currentRow.service?.name }}</el-descriptions-item>
|
|
<el-descriptions-item label="预约时间" :span="2">
|
|
{{ formatDate(currentRow.appointmentTime) }}
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="服务地址" :span="2">{{ currentRow.address }}</el-descriptions-item>
|
|
<el-descriptions-item label="问题描述" :span="2">
|
|
{{ currentRow.description || '无' }}
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="图片" :span="2" v-if="currentRow.images && currentRow.images.length">
|
|
<div style="display: flex; gap: 10px; flex-wrap: wrap">
|
|
<el-image
|
|
v-for="(img, index) in currentRow.images"
|
|
:key="index"
|
|
:src="img"
|
|
fit="cover"
|
|
style="width: 100px; height: 100px; border-radius: 4px"
|
|
:preview-src-list="currentRow.images"
|
|
:initial-index="index"
|
|
/>
|
|
</div>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="创建时间" :span="2">
|
|
{{ formatDate(currentRow.createdAt) }}
|
|
</el-descriptions-item>
|
|
</el-descriptions>
|
|
<template #footer>
|
|
<el-button @click="detailVisible = false">关闭</el-button>
|
|
</template>
|
|
</el-dialog>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, reactive, onMounted } from 'vue'
|
|
import { ElMessage } from 'element-plus'
|
|
import { Search, RefreshLeft, View, ArrowDown } from '@element-plus/icons-vue'
|
|
import { getBookingList, updateBooking } from '@/api/booking'
|
|
import dayjs from 'dayjs'
|
|
|
|
const loading = ref(false)
|
|
const tableData = ref<any[]>([])
|
|
const total = ref(0)
|
|
|
|
const queryForm = reactive({
|
|
bookingNumber: '',
|
|
contactName: '',
|
|
status: '',
|
|
page: 1,
|
|
limit: 50 // 增加默认显示数量
|
|
})
|
|
|
|
const detailVisible = ref(false)
|
|
const currentRow = ref<any>(null)
|
|
|
|
const loadData = async () => {
|
|
loading.value = true
|
|
try {
|
|
const params = {
|
|
...queryForm,
|
|
bookingNumber: queryForm.bookingNumber || undefined,
|
|
contactName: queryForm.contactName || undefined,
|
|
status: queryForm.status || undefined
|
|
}
|
|
const res = await getBookingList(params)
|
|
tableData.value = res.data.list || []
|
|
total.value = res.data.total || 0
|
|
} catch (error) {
|
|
console.error('加载数据失败:', error)
|
|
} finally {
|
|
loading.value = false
|
|
}
|
|
}
|
|
|
|
const handleSearch = () => {
|
|
queryForm.page = 1
|
|
loadData()
|
|
}
|
|
|
|
const handleReset = () => {
|
|
queryForm.bookingNumber = ''
|
|
queryForm.contactName = ''
|
|
queryForm.status = ''
|
|
queryForm.page = 1
|
|
loadData()
|
|
}
|
|
|
|
const handleView = (row: any) => {
|
|
currentRow.value = row
|
|
detailVisible.value = true
|
|
}
|
|
|
|
const handleStatusCommand = async (command: string, row: any) => {
|
|
try {
|
|
await updateBooking(row.id, { status: command })
|
|
ElMessage.success('状态更新成功')
|
|
loadData()
|
|
} catch (error) {
|
|
console.error('更新失败:', error)
|
|
}
|
|
}
|
|
|
|
const formatDate = (date: string) => {
|
|
return dayjs(date).format('YYYY-MM-DD HH:mm:ss')
|
|
}
|
|
|
|
const getStatusType = (status: string) => {
|
|
const map: Record<string, any> = {
|
|
pending: 'warning',
|
|
confirmed: 'success',
|
|
in_progress: 'primary',
|
|
completed: 'info',
|
|
cancelled: 'danger'
|
|
}
|
|
return map[status] || 'info'
|
|
}
|
|
|
|
const getStatusText = (status: string) => {
|
|
const map: Record<string, string> = {
|
|
pending: '待确认',
|
|
confirmed: '已确认',
|
|
in_progress: '进行中',
|
|
completed: '已完成',
|
|
cancelled: '已取消'
|
|
}
|
|
return map[status] || status
|
|
}
|
|
|
|
onMounted(() => {
|
|
loadData()
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.booking-list {
|
|
padding: 20px;
|
|
}
|
|
|
|
.card-header {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
}
|
|
|
|
.card-title {
|
|
font-size: 18px;
|
|
font-weight: 600;
|
|
}
|
|
|
|
.search-form {
|
|
margin-bottom: 20px;
|
|
}
|
|
</style>
|