feat: 完成所有前端页面和路由配置
项目成员管理:
- 创建 projectMember.vue 页面(318行)
- 按项目查询成员列表
- 添加/编辑/移除成员
- 角色选择(项目经理/开发/测试/财务/普通成员)
- 工作量占比滑块
- 状态管理(在职/已离开)
应收款管理:
- 创建 receivable.vue 页面(475行)
- 多条件搜索(编号、项目、状态)
- 表格展示(应收款金额、已收款金额、未收款金额、逾期天数)
- 新增/编辑应收款
- 记录收款功能
- 逾期提醒(红色显示)
- 状态标签(待收款/部分收款/已收款/逾期)
路由配置:
- 添加项目成员管理路由(/project/member)
- 添加应收款管理路由(/finance/receivable)
所有核心模块前端页面全部完成!✅
This commit is contained in:
parent
0323717110
commit
abded8ec75
@ -92,6 +92,12 @@ const routes = [
|
|||||||
name: 'Requirement',
|
name: 'Requirement',
|
||||||
component: () => import('../views/project/requirement.vue'),
|
component: () => import('../views/project/requirement.vue'),
|
||||||
meta: { title: '需求工单' }
|
meta: { title: '需求工单' }
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: 'member',
|
||||||
|
name: 'ProjectMember',
|
||||||
|
component: () => import('../views/project/projectMember.vue'),
|
||||||
|
meta: { title: '项目成员' }
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@ -111,6 +117,12 @@ const routes = [
|
|||||||
name: 'Expense',
|
name: 'Expense',
|
||||||
component: () => import('../views/finance/expense.vue'),
|
component: () => import('../views/finance/expense.vue'),
|
||||||
meta: { title: '支出管理' }
|
meta: { title: '支出管理' }
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: 'receivable',
|
||||||
|
name: 'Receivable',
|
||||||
|
component: () => import('../views/finance/receivable.vue'),
|
||||||
|
meta: { title: '应收款管理' }
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
474
fund-admin/src/views/finance/receivable.vue
Normal file
474
fund-admin/src/views/finance/receivable.vue
Normal file
@ -0,0 +1,474 @@
|
|||||||
|
<template>
|
||||||
|
<div class="receivable-container">
|
||||||
|
<el-card>
|
||||||
|
<!-- 搜索栏 -->
|
||||||
|
<el-form :inline="true" :model="searchForm">
|
||||||
|
<el-form-item label="应收款编号">
|
||||||
|
<el-input v-model="searchForm.receivableCode" placeholder="请输入编号" clearable />
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item label="项目">
|
||||||
|
<el-select v-model="searchForm.projectId" placeholder="请选择项目" clearable>
|
||||||
|
<el-option
|
||||||
|
v-for="project in projectOptions"
|
||||||
|
:key="project.projectId"
|
||||||
|
:label="project.projectName"
|
||||||
|
:value="project.projectId"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item label="状态">
|
||||||
|
<el-select v-model="searchForm.status" placeholder="请选择状态" clearable>
|
||||||
|
<el-option label="待收款" value="pending" />
|
||||||
|
<el-option label="部分收款" value="partial" />
|
||||||
|
<el-option label="已收款" value="received" />
|
||||||
|
<el-option label="逾期" value="overdue" />
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-button type="primary" @click="handleSearch">搜索</el-button>
|
||||||
|
<el-button @click="handleReset">重置</el-button>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
|
||||||
|
<!-- 操作栏 -->
|
||||||
|
<el-row style="margin-bottom: 15px;">
|
||||||
|
<el-button type="primary" @click="handleAdd">新增应收款</el-button>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<!-- 表格 -->
|
||||||
|
<el-table :data="tableData" border v-loading="loading">
|
||||||
|
<el-table-column prop="receivableCode" label="应收款编号" width="150" />
|
||||||
|
<el-table-column prop="projectId" label="项目" width="150">
|
||||||
|
<template #default="{ row }">
|
||||||
|
{{ getProjectName(row.projectId) }}
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="receivableAmount" label="应收款金额" width="120">
|
||||||
|
<template #default="{ row }">
|
||||||
|
¥{{ row.receivableAmount }}
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="receivedAmount" label="已收款金额" width="120">
|
||||||
|
<template #default="{ row }">
|
||||||
|
¥{{ row.receivedAmount }}
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="unpaidAmount" label="未收款金额" width="120">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<span :style="{ color: row.unpaidAmount > 0 ? 'red' : 'green' }">
|
||||||
|
¥{{ row.unpaidAmount }}
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="receivableDate" label="应收款日期" width="120" />
|
||||||
|
<el-table-column prop="paymentDueDate" label="付款截止日期" width="130" />
|
||||||
|
<el-table-column prop="status" label="状态" width="100">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-tag v-if="row.status === 'pending'" type="warning">待收款</el-tag>
|
||||||
|
<el-tag v-else-if="row.status === 'partial'" type="primary">部分收款</el-tag>
|
||||||
|
<el-tag v-else-if="row.status === 'received'" type="success">已收款</el-tag>
|
||||||
|
<el-tag v-else type="danger">逾期</el-tag>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="overdueDays" label="逾期天数" width="100">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<span v-if="row.overdueDays > 0" style="color: red;">{{ row.overdueDays }}天</span>
|
||||||
|
<span v-else>-</span>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="createdTime" label="创建时间" width="180" />
|
||||||
|
<el-table-column label="操作" width="250" fixed="right">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-button type="primary" size="small" @click="handleEdit(row)">编辑</el-button>
|
||||||
|
<el-button type="success" size="small" @click="handleRecordPayment(row)">记录收款</el-button>
|
||||||
|
<el-button type="danger" size="small" @click="handleDelete(row)">删除</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
|
||||||
|
<!-- 分页 -->
|
||||||
|
<el-pagination
|
||||||
|
v-model:current-page="page.current"
|
||||||
|
v-model:page-size="page.size"
|
||||||
|
:total="page.total"
|
||||||
|
:page-sizes="[10, 20, 50, 100]"
|
||||||
|
layout="total, sizes, prev, pager, next, jumper"
|
||||||
|
@size-change="fetchData"
|
||||||
|
@current-change="fetchData"
|
||||||
|
style="margin-top: 20px; justify-content: flex-end;"
|
||||||
|
/>
|
||||||
|
</el-card>
|
||||||
|
|
||||||
|
<!-- 新增/编辑对话框 -->
|
||||||
|
<el-dialog
|
||||||
|
v-model="dialogVisible"
|
||||||
|
:title="form.receivableId ? '编辑应收款' : '新增应收款'"
|
||||||
|
width="700px"
|
||||||
|
>
|
||||||
|
<el-form ref="formRef" :model="form" :rules="rules" label-width="130px">
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="应收款编号" prop="receivableCode">
|
||||||
|
<el-input v-model="form.receivableCode" placeholder="请输入编号" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="需求ID" prop="requirementId">
|
||||||
|
<el-input-number v-model="form.requirementId" :min="1" style="width: 100%;" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="项目" prop="projectId">
|
||||||
|
<el-select v-model="form.projectId" placeholder="请选择项目" style="width: 100%;">
|
||||||
|
<el-option
|
||||||
|
v-for="project in projectOptions"
|
||||||
|
:key="project.projectId"
|
||||||
|
:label="project.projectName"
|
||||||
|
:value="project.projectId"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="客户ID" prop="customerId">
|
||||||
|
<el-input-number v-model="form.customerId" :min="1" style="width: 100%;" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="应收款金额" prop="receivableAmount">
|
||||||
|
<el-input-number v-model="form.receivableAmount" :min="0" :precision="2" style="width: 100%;" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="应收款日期" prop="receivableDate">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="form.receivableDate"
|
||||||
|
type="date"
|
||||||
|
placeholder="选择日期"
|
||||||
|
style="width: 100%;"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<el-form-item label="付款截止日期" prop="paymentDueDate">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="form.paymentDueDate"
|
||||||
|
type="date"
|
||||||
|
placeholder="选择日期"
|
||||||
|
style="width: 100%;"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="付款方式" prop="paymentMethod">
|
||||||
|
<el-select v-model="form.paymentMethod" placeholder="请选择付款方式" style="width: 100%;">
|
||||||
|
<el-option label="转账" value="transfer" />
|
||||||
|
<el-option label="现金" value="cash" />
|
||||||
|
<el-option label="支票" value="check" />
|
||||||
|
<el-option label="其他" value="other" />
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="收款账户" prop="bankAccount">
|
||||||
|
<el-input v-model="form.bankAccount" placeholder="请输入收款账户" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<el-form-item label="备注说明" prop="remark">
|
||||||
|
<el-input
|
||||||
|
v-model="form.remark"
|
||||||
|
type="textarea"
|
||||||
|
:rows="3"
|
||||||
|
placeholder="请输入备注说明"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
|
||||||
|
<template #footer>
|
||||||
|
<el-button @click="dialogVisible = false">取消</el-button>
|
||||||
|
<el-button type="primary" @click="handleSubmit">确定</el-button>
|
||||||
|
</template>
|
||||||
|
</el-dialog>
|
||||||
|
|
||||||
|
<!-- 记录收款对话框 -->
|
||||||
|
<el-dialog v-model="paymentDialogVisible" title="记录收款" width="400px">
|
||||||
|
<el-form>
|
||||||
|
<el-form-item label="收款金额">
|
||||||
|
<el-input-number v-model="paymentForm.amount" :min="0" :precision="2" style="width: 100%;" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
<template #footer>
|
||||||
|
<el-button @click="paymentDialogVisible = false">取消</el-button>
|
||||||
|
<el-button type="primary" @click="handleSubmitPayment">确定</el-button>
|
||||||
|
</template>
|
||||||
|
</el-dialog>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import { ref, reactive, onMounted } from 'vue'
|
||||||
|
import { ElMessage, ElMessageBox } from 'element-plus'
|
||||||
|
import {
|
||||||
|
getReceivableList,
|
||||||
|
createReceivable,
|
||||||
|
updateReceivable,
|
||||||
|
deleteReceivable,
|
||||||
|
recordPayment
|
||||||
|
} from '../../api/receivable'
|
||||||
|
import { getProjectList } from '../../api/project'
|
||||||
|
|
||||||
|
// 搜索表单
|
||||||
|
const searchForm = reactive({
|
||||||
|
receivableCode: '',
|
||||||
|
projectId: null,
|
||||||
|
status: ''
|
||||||
|
})
|
||||||
|
|
||||||
|
// 表格数据
|
||||||
|
const tableData = ref([])
|
||||||
|
const loading = ref(false)
|
||||||
|
|
||||||
|
// 分页
|
||||||
|
const page = reactive({
|
||||||
|
current: 1,
|
||||||
|
size: 10,
|
||||||
|
total: 0
|
||||||
|
})
|
||||||
|
|
||||||
|
// 项目选项
|
||||||
|
const projectOptions = ref([])
|
||||||
|
|
||||||
|
// 对话框
|
||||||
|
const dialogVisible = ref(false)
|
||||||
|
const formRef = ref(null)
|
||||||
|
|
||||||
|
// 表单数据
|
||||||
|
const form = reactive({
|
||||||
|
receivableId: null,
|
||||||
|
receivableCode: '',
|
||||||
|
requirementId: null,
|
||||||
|
projectId: null,
|
||||||
|
customerId: null,
|
||||||
|
receivableAmount: 0,
|
||||||
|
receivableDate: null,
|
||||||
|
paymentDueDate: null,
|
||||||
|
paymentMethod: 'transfer',
|
||||||
|
bankAccount: '',
|
||||||
|
remark: ''
|
||||||
|
})
|
||||||
|
|
||||||
|
// 表单验证规则
|
||||||
|
const rules = {
|
||||||
|
receivableCode: [
|
||||||
|
{ required: true, message: '请输入应收款编号', trigger: 'blur' }
|
||||||
|
],
|
||||||
|
requirementId: [
|
||||||
|
{ required: true, message: '请输入需求ID', trigger: 'blur' }
|
||||||
|
],
|
||||||
|
projectId: [
|
||||||
|
{ required: true, message: '请选择项目', trigger: 'change' }
|
||||||
|
],
|
||||||
|
customerId: [
|
||||||
|
{ required: true, message: '请输入客户ID', trigger: 'blur' }
|
||||||
|
],
|
||||||
|
receivableAmount: [
|
||||||
|
{ required: true, message: '请输入应收款金额', trigger: 'blur' }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
// 记录收款对话框
|
||||||
|
const paymentDialogVisible = ref(false)
|
||||||
|
const paymentForm = reactive({
|
||||||
|
receivableId: null,
|
||||||
|
amount: 0
|
||||||
|
})
|
||||||
|
|
||||||
|
// 加载项目列表
|
||||||
|
const loadProjects = async () => {
|
||||||
|
try {
|
||||||
|
const res = await getProjectList({ current: 1, size: 1000 })
|
||||||
|
projectOptions.value = res.records || []
|
||||||
|
} catch (error) {
|
||||||
|
console.error('加载项目列表失败:', error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取项目名称
|
||||||
|
const getProjectName = (projectId) => {
|
||||||
|
const project = projectOptions.value.find(p => p.projectId === projectId)
|
||||||
|
return project ? project.projectName : '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
// 加载数据
|
||||||
|
const fetchData = async () => {
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
const params = {
|
||||||
|
current: page.current,
|
||||||
|
size: page.size,
|
||||||
|
receivableCode: searchForm.receivableCode || undefined,
|
||||||
|
projectId: searchForm.projectId || undefined,
|
||||||
|
status: searchForm.status || undefined
|
||||||
|
}
|
||||||
|
const res = await getReceivableList(params)
|
||||||
|
tableData.value = res.records
|
||||||
|
page.total = res.total
|
||||||
|
} catch (error) {
|
||||||
|
console.error('加载数据失败:', error)
|
||||||
|
ElMessage.error(error.message || '加载数据失败')
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 搜索
|
||||||
|
const handleSearch = () => {
|
||||||
|
page.current = 1
|
||||||
|
fetchData()
|
||||||
|
}
|
||||||
|
|
||||||
|
// 重置
|
||||||
|
const handleReset = () => {
|
||||||
|
searchForm.receivableCode = ''
|
||||||
|
searchForm.projectId = null
|
||||||
|
searchForm.status = ''
|
||||||
|
page.current = 1
|
||||||
|
fetchData()
|
||||||
|
}
|
||||||
|
|
||||||
|
// 新增
|
||||||
|
const handleAdd = () => {
|
||||||
|
Object.assign(form, {
|
||||||
|
receivableId: null,
|
||||||
|
receivableCode: '',
|
||||||
|
requirementId: null,
|
||||||
|
projectId: null,
|
||||||
|
customerId: null,
|
||||||
|
receivableAmount: 0,
|
||||||
|
receivableDate: null,
|
||||||
|
paymentDueDate: null,
|
||||||
|
paymentMethod: 'transfer',
|
||||||
|
bankAccount: '',
|
||||||
|
remark: ''
|
||||||
|
})
|
||||||
|
dialogVisible.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编辑
|
||||||
|
const handleEdit = (row) => {
|
||||||
|
Object.assign(form, {
|
||||||
|
receivableId: row.receivableId,
|
||||||
|
receivableCode: row.receivableCode,
|
||||||
|
requirementId: row.requirementId,
|
||||||
|
projectId: row.projectId,
|
||||||
|
customerId: row.customerId,
|
||||||
|
receivableAmount: row.receivableAmount,
|
||||||
|
receivableDate: row.receivableDate,
|
||||||
|
paymentDueDate: row.paymentDueDate,
|
||||||
|
paymentMethod: row.paymentMethod,
|
||||||
|
bankAccount: row.bankAccount,
|
||||||
|
remark: row.remark
|
||||||
|
})
|
||||||
|
dialogVisible.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
// 提交表单
|
||||||
|
const handleSubmit = async () => {
|
||||||
|
if (!formRef.value) return
|
||||||
|
|
||||||
|
try {
|
||||||
|
await formRef.value.validate()
|
||||||
|
} catch (error) {
|
||||||
|
ElMessage.warning('请检查表单填写是否完整')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (form.receivableId) {
|
||||||
|
await updateReceivable(form.receivableId, form)
|
||||||
|
ElMessage.success('更新成功')
|
||||||
|
} else {
|
||||||
|
await createReceivable(form)
|
||||||
|
ElMessage.success('创建成功')
|
||||||
|
}
|
||||||
|
dialogVisible.value = false
|
||||||
|
await fetchData()
|
||||||
|
} catch (error) {
|
||||||
|
console.error('保存失败:', error)
|
||||||
|
ElMessage.error(error.message || '操作失败')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 记录收款
|
||||||
|
const handleRecordPayment = (row) => {
|
||||||
|
paymentForm.receivableId = row.receivableId
|
||||||
|
paymentForm.amount = 0
|
||||||
|
paymentDialogVisible.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
// 提交收款
|
||||||
|
const handleSubmitPayment = async () => {
|
||||||
|
if (paymentForm.amount <= 0) {
|
||||||
|
ElMessage.warning('请输入有效的收款金额')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await recordPayment(paymentForm.receivableId, paymentForm.amount)
|
||||||
|
ElMessage.success('收款记录成功')
|
||||||
|
paymentDialogVisible.value = false
|
||||||
|
await fetchData()
|
||||||
|
} catch (error) {
|
||||||
|
console.error('收款记录失败:', error)
|
||||||
|
ElMessage.error(error.message || '收款记录失败')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除
|
||||||
|
const handleDelete = async (row) => {
|
||||||
|
try {
|
||||||
|
await ElMessageBox.confirm('确定要删除该应收款吗?', '提示', {
|
||||||
|
confirmButtonText: '确定',
|
||||||
|
cancelButtonText: '取消',
|
||||||
|
type: 'warning'
|
||||||
|
})
|
||||||
|
|
||||||
|
await deleteReceivable(row.receivableId)
|
||||||
|
ElMessage.success('删除成功')
|
||||||
|
|
||||||
|
// 如果是最后一条且不是第一页,返回上一页
|
||||||
|
if (tableData.value.length === 1 && page.current > 1) {
|
||||||
|
page.current--
|
||||||
|
}
|
||||||
|
|
||||||
|
await fetchData()
|
||||||
|
} catch (error) {
|
||||||
|
if (error !== 'cancel') {
|
||||||
|
console.error('删除失败:', error)
|
||||||
|
ElMessage.error(error.message || '删除失败')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化
|
||||||
|
onMounted(() => {
|
||||||
|
loadProjects()
|
||||||
|
fetchData()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.receivable-container {
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
317
fund-admin/src/views/project/projectMember.vue
Normal file
317
fund-admin/src/views/project/projectMember.vue
Normal file
@ -0,0 +1,317 @@
|
|||||||
|
<template>
|
||||||
|
<div class="project-member-container">
|
||||||
|
<el-card>
|
||||||
|
<!-- 搜索栏 -->
|
||||||
|
<el-form :inline="true" :model="searchForm">
|
||||||
|
<el-form-item label="项目">
|
||||||
|
<el-select v-model="searchForm.projectId" placeholder="请选择项目" clearable @change="fetchData">
|
||||||
|
<el-option
|
||||||
|
v-for="project in projectOptions"
|
||||||
|
:key="project.projectId"
|
||||||
|
:label="project.projectName"
|
||||||
|
:value="project.projectId"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-button type="primary" @click="handleAdd">添加成员</el-button>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
|
||||||
|
<!-- 表格 -->
|
||||||
|
<el-table :data="tableData" border v-loading="loading">
|
||||||
|
<el-table-column prop="userId" label="用户ID" width="100" />
|
||||||
|
<el-table-column prop="role" label="项目角色" width="120">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-tag v-if="row.role === 'pm'" type="danger">项目经理</el-tag>
|
||||||
|
<el-tag v-else-if="row.role === 'dev'" type="primary">开发</el-tag>
|
||||||
|
<el-tag v-else-if="row.role === 'test'" type="warning">测试</el-tag>
|
||||||
|
<el-tag v-else-if="row.role === 'finance'" type="success">财务</el-tag>
|
||||||
|
<el-tag v-else type="info">普通成员</el-tag>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="joinDate" label="加入日期" width="120" />
|
||||||
|
<el-table-column prop="leaveDate" label="离开日期" width="120" />
|
||||||
|
<el-table-column prop="workload" label="工作量占比" width="120">
|
||||||
|
<template #default="{ row }">
|
||||||
|
{{ row.workload }}%
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="status" label="状态" width="100">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-tag v-if="row.status === 1" type="success">在职</el-tag>
|
||||||
|
<el-tag v-else type="info">已离开</el-tag>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="remark" label="备注" min-width="200" show-overflow-tooltip />
|
||||||
|
<el-table-column prop="createdTime" label="创建时间" width="180" />
|
||||||
|
<el-table-column label="操作" width="200" fixed="right">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-button type="primary" size="small" @click="handleEdit(row)">编辑</el-button>
|
||||||
|
<el-button type="danger" size="small" @click="handleRemove(row)">移除</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
</el-card>
|
||||||
|
|
||||||
|
<!-- 新增/编辑对话框 -->
|
||||||
|
<el-dialog
|
||||||
|
v-model="dialogVisible"
|
||||||
|
:title="form.memberId ? '编辑成员' : '添加成员'"
|
||||||
|
width="600px"
|
||||||
|
>
|
||||||
|
<el-form ref="formRef" :model="form" :rules="rules" label-width="120px">
|
||||||
|
<el-form-item label="项目" prop="projectId">
|
||||||
|
<el-select v-model="form.projectId" placeholder="请选择项目" style="width: 100%;">
|
||||||
|
<el-option
|
||||||
|
v-for="project in projectOptions"
|
||||||
|
:key="project.projectId"
|
||||||
|
:label="project.projectName"
|
||||||
|
:value="project.projectId"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
|
||||||
|
<el-form-item label="用户ID" prop="userId">
|
||||||
|
<el-input-number v-model="form.userId" :min="1" style="width: 100%;" />
|
||||||
|
</el-form-item>
|
||||||
|
|
||||||
|
<el-form-item label="项目角色" prop="role">
|
||||||
|
<el-select v-model="form.role" placeholder="请选择角色" style="width: 100%;">
|
||||||
|
<el-option label="项目经理" value="pm" />
|
||||||
|
<el-option label="开发" value="dev" />
|
||||||
|
<el-option label="测试" value="test" />
|
||||||
|
<el-option label="财务" value="finance" />
|
||||||
|
<el-option label="普通成员" value="member" />
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="加入日期" prop="joinDate">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="form.joinDate"
|
||||||
|
type="date"
|
||||||
|
placeholder="选择日期"
|
||||||
|
style="width: 100%;"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="离开日期">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="form.leaveDate"
|
||||||
|
type="date"
|
||||||
|
placeholder="选择日期"
|
||||||
|
style="width: 100%;"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<el-form-item label="工作量占比" prop="workload">
|
||||||
|
<el-slider v-model="form.workload" :min="0" :max="100" show-input />
|
||||||
|
</el-form-item>
|
||||||
|
|
||||||
|
<el-form-item label="状态" prop="status">
|
||||||
|
<el-radio-group v-model="form.status">
|
||||||
|
<el-radio :label="1">在职</el-radio>
|
||||||
|
<el-radio :label="0">已离开</el-radio>
|
||||||
|
</el-radio-group>
|
||||||
|
</el-form-item>
|
||||||
|
|
||||||
|
<el-form-item label="备注说明">
|
||||||
|
<el-input
|
||||||
|
v-model="form.remark"
|
||||||
|
type="textarea"
|
||||||
|
:rows="3"
|
||||||
|
placeholder="请输入备注说明"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
|
||||||
|
<template #footer>
|
||||||
|
<el-button @click="dialogVisible = false">取消</el-button>
|
||||||
|
<el-button type="primary" @click="handleSubmit">确定</el-button>
|
||||||
|
</template>
|
||||||
|
</el-dialog>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import { ref, reactive, onMounted } from 'vue'
|
||||||
|
import { ElMessage, ElMessageBox } from 'element-plus'
|
||||||
|
import {
|
||||||
|
getProjectMembers,
|
||||||
|
addProjectMember,
|
||||||
|
updateProjectMember,
|
||||||
|
removeProjectMember
|
||||||
|
} from '../../api/projectMember'
|
||||||
|
import { getProjectList } from '../../api/project'
|
||||||
|
|
||||||
|
// 搜索表单
|
||||||
|
const searchForm = reactive({
|
||||||
|
projectId: null
|
||||||
|
})
|
||||||
|
|
||||||
|
// 表格数据
|
||||||
|
const tableData = ref([])
|
||||||
|
const loading = ref(false)
|
||||||
|
|
||||||
|
// 项目选项
|
||||||
|
const projectOptions = ref([])
|
||||||
|
|
||||||
|
// 对话框
|
||||||
|
const dialogVisible = ref(false)
|
||||||
|
const formRef = ref(null)
|
||||||
|
|
||||||
|
// 表单数据
|
||||||
|
const form = reactive({
|
||||||
|
memberId: null,
|
||||||
|
projectId: null,
|
||||||
|
userId: null,
|
||||||
|
role: 'member',
|
||||||
|
joinDate: null,
|
||||||
|
leaveDate: null,
|
||||||
|
workload: 0,
|
||||||
|
status: 1,
|
||||||
|
remark: ''
|
||||||
|
})
|
||||||
|
|
||||||
|
// 表单验证规则
|
||||||
|
const rules = {
|
||||||
|
projectId: [
|
||||||
|
{ required: true, message: '请选择项目', trigger: 'change' }
|
||||||
|
],
|
||||||
|
userId: [
|
||||||
|
{ required: true, message: '请输入用户ID', trigger: 'blur' }
|
||||||
|
],
|
||||||
|
role: [
|
||||||
|
{ required: true, message: '请选择角色', trigger: 'change' }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
// 加载项目列表
|
||||||
|
const loadProjects = async () => {
|
||||||
|
try {
|
||||||
|
const res = await getProjectList({ current: 1, size: 1000 })
|
||||||
|
projectOptions.value = res.records || []
|
||||||
|
} catch (error) {
|
||||||
|
console.error('加载项目列表失败:', error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 加载数据
|
||||||
|
const fetchData = async () => {
|
||||||
|
if (!searchForm.projectId) {
|
||||||
|
ElMessage.warning('请选择项目')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
const res = await getProjectMembers(searchForm.projectId)
|
||||||
|
tableData.value = res
|
||||||
|
} catch (error) {
|
||||||
|
console.error('加载数据失败:', error)
|
||||||
|
ElMessage.error(error.message || '加载数据失败')
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 新增
|
||||||
|
const handleAdd = () => {
|
||||||
|
if (!searchForm.projectId) {
|
||||||
|
ElMessage.warning('请先选择项目')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
Object.assign(form, {
|
||||||
|
memberId: null,
|
||||||
|
projectId: searchForm.projectId,
|
||||||
|
userId: null,
|
||||||
|
role: 'member',
|
||||||
|
joinDate: null,
|
||||||
|
leaveDate: null,
|
||||||
|
workload: 0,
|
||||||
|
status: 1,
|
||||||
|
remark: ''
|
||||||
|
})
|
||||||
|
dialogVisible.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编辑
|
||||||
|
const handleEdit = (row) => {
|
||||||
|
Object.assign(form, {
|
||||||
|
memberId: row.memberId,
|
||||||
|
projectId: row.projectId,
|
||||||
|
userId: row.userId,
|
||||||
|
role: row.role,
|
||||||
|
joinDate: row.joinDate,
|
||||||
|
leaveDate: row.leaveDate,
|
||||||
|
workload: row.workload,
|
||||||
|
status: row.status,
|
||||||
|
remark: row.remark
|
||||||
|
})
|
||||||
|
dialogVisible.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
// 提交表单
|
||||||
|
const handleSubmit = async () => {
|
||||||
|
if (!formRef.value) return
|
||||||
|
|
||||||
|
try {
|
||||||
|
await formRef.value.validate()
|
||||||
|
} catch (error) {
|
||||||
|
ElMessage.warning('请检查表单填写是否完整')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (form.memberId) {
|
||||||
|
await updateProjectMember(form.memberId, form)
|
||||||
|
ElMessage.success('更新成功')
|
||||||
|
} else {
|
||||||
|
await addProjectMember(form)
|
||||||
|
ElMessage.success('添加成功')
|
||||||
|
}
|
||||||
|
dialogVisible.value = false
|
||||||
|
await fetchData()
|
||||||
|
} catch (error) {
|
||||||
|
console.error('保存失败:', error)
|
||||||
|
ElMessage.error(error.message || '操作失败')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 移除
|
||||||
|
const handleRemove = async (row) => {
|
||||||
|
try {
|
||||||
|
await ElMessageBox.confirm('确定要移除该成员吗?', '提示', {
|
||||||
|
confirmButtonText: '确定',
|
||||||
|
cancelButtonText: '取消',
|
||||||
|
type: 'warning'
|
||||||
|
})
|
||||||
|
|
||||||
|
await removeProjectMember(row.memberId)
|
||||||
|
ElMessage.success('移除成功')
|
||||||
|
await fetchData()
|
||||||
|
} catch (error) {
|
||||||
|
if (error !== 'cancel') {
|
||||||
|
console.error('移除失败:', error)
|
||||||
|
ElMessage.error(error.message || '移除失败')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化
|
||||||
|
onMounted(() => {
|
||||||
|
loadProjects()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.project-member-container {
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@ -1 +1 @@
|
|||||||
Subproject commit 91ba1879f285919f8c9d01f74e2a5c6a1842511e
|
Subproject commit 184919142741a540dcd8ed6e0862eba5153458d5
|
||||||
Loading…
x
Reference in New Issue
Block a user