| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515 |
- <template>
- <div class="qc-task-container">
- <el-card shadow="never">
- <template #header>
- <div class="flex justify-between items-center">
- <span class="text-lg font-bold">文件质控任务</span>
- </div>
- </template>
- <!-- 搜索栏 -->
- <el-form :model="queryParams" :inline="true" class="search-form">
- <el-form-item label="任务名称">
- <el-input v-model="queryParams.taskName" placeholder="请输入任务名称" clearable style="width: 200px"
- @keyup.enter="handleQuery" />
- </el-form-item>
- <el-form-item label="项目名称">
- <el-input v-model="queryParams.projectName" placeholder="请输入项目名称" clearable style="width: 200px"
- @keyup.enter="handleQuery" />
- </el-form-item>
- <el-form-item label="文档名称">
- <el-input v-model="queryParams.documentName" placeholder="请输入文档名称" clearable style="width: 200px"
- @keyup.enter="handleQuery" />
- </el-form-item>
- <el-form-item label="状态">
- <el-select v-model="queryParams.status" placeholder="请选择状态" clearable style="width: 120px">
- <el-option label="待审核" :value="0" />
- <el-option label="审核通过" :value="1" />
- <el-option label="审核拒绝" :value="2" />
- </el-select>
- </el-form-item>
- <el-form-item>
- <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
- <el-button icon="Refresh" @click="resetQuery">重置</el-button>
- </el-form-item>
- </el-form>
- <!-- 任务列表 -->
- <el-table v-loading="loading" :data="taskList" border style="margin-top: 10px">
- <el-table-column prop="id" label="序号" width="80" align="center" />
- <el-table-column prop="name" label="任务名称" min-width="150" show-overflow-tooltip>
- <template #default="scope">
- {{ formatDocumentName(scope.row.name) }}
- </template>
- </el-table-column>
- <el-table-column prop="projectName" label="项目名称" min-width="150" show-overflow-tooltip />
- <el-table-column prop="initiator" label="发起人" width="120" align="center" />
- <el-table-column prop="executor" label="执行人" width="120" align="center" />
- <el-table-column prop="status" label="状态" width="120" align="center">
- <template #default="scope">
- <el-tag v-if="scope.row.status === 0 || scope.row.status === 3" type="info">待审核</el-tag>
- <el-tag v-else-if="scope.row.status === 1" type="success">审核通过</el-tag>
- <el-tag v-else-if="scope.row.status === 2" type="danger">审核拒绝</el-tag>
- <el-tag v-else type="warning">未知</el-tag>
- </template>
- </el-table-column>
- <el-table-column prop="note" label="备注" min-width="150" show-overflow-tooltip />
- <el-table-column prop="executeTime" label="执行时间" width="180" align="center">
- <template #default="scope">
- <span v-if="scope.row.executeTime">{{ scope.row.executeTime }}</span>
- <span v-else>-</span>
- </template>
- </el-table-column>
- <el-table-column prop="createTime" label="创建时间" width="180" align="center">
- <template #default="scope">
- <span v-if="scope.row.createTime">{{ scope.row.createTime }}</span>
- <span v-else>-</span>
- </template>
- </el-table-column>
- <!-- 操作列 -->
- <el-table-column label="操作" width="280" align="center" fixed="right" class-name="small-padding fixed-width">
- <template #default="scope">
- <el-button
- v-if="scope.row.status === 2"
- v-hasPermi="['taskCenter:qc:submit']"
- type="primary"
- icon="Upload"
- style="padding: 0 5px; font-size: 10px; height: 24px"
- @click="handleSubmit(scope.row)"
- >
- 递交
- </el-button>
- <el-button
- v-if="scope.row.status === 0 || scope.row.status === 3"
- v-hasPermi="['taskCenter:qc:audit']"
- type="success"
- icon="Check"
- style="padding: 0 5px; font-size: 10px; height: 24px"
- @click="handleAudit(scope.row)"
- >
- 审核
- </el-button>
- <el-button
- v-hasPermi="['taskCenter:qc:logAudit']"
- type="warning"
- icon="Document"
- style="padding: 0 5px; font-size: 10px; height: 24px"
- @click="handleViewLog(scope.row)"
- >
- 查看审核记录
- </el-button>
- </template>
- </el-table-column>
- </el-table>
- <!-- 分页 -->
- <pagination v-show="total > 0" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize"
- :total="total" @pagination="getTaskList" />
- </el-card>
- <!-- 递交文档对话框 -->
- <el-dialog v-model="submitDialog.visible" :title="submitDialog.title" width="500px" append-to-body>
- <el-alert title="仅支持上传 PDF 格式文件" type="info" :closable="false" style="margin-bottom: 20px" />
- <el-form ref="submitFormRef" :model="submitForm" :rules="submitRules" label-width="100px">
- <el-form-item label="文件" prop="document">
- <fileUpload v-model="submitForm.document" :limit="1" :file-type="['pdf']" :is-show-tip="false" />
- </el-form-item>
- </el-form>
- <template #footer>
- <div class="dialog-footer">
- <el-button @click="cancelSubmit">取消</el-button>
- <el-button type="primary" :loading="submitButtonLoading" @click="submitSubmitForm">确认</el-button>
- </div>
- </template>
- </el-dialog>
- <!-- 审核弹窗 -->
- <el-dialog v-model="auditDialog.visible" title="审核" width="500px" append-to-body>
- <el-form ref="auditFormRef" :model="auditForm" :rules="auditRules" label-width="100px">
- <el-form-item label="结果" prop="result">
- <el-radio-group v-model="auditForm.result">
- <el-radio :label="1">通过</el-radio>
- <el-radio :label="2">驳回</el-radio>
- </el-radio-group>
- </el-form-item>
- <template v-if="auditForm.result === 2">
- <el-form-item label="问题类型" prop="rejectionType">
- <el-select v-model="auditForm.rejectionType" placeholder="请选择问题类型" style="width: 100%">
- <el-option
- v-for="dict in qc_question_type"
- :key="dict.value"
- :label="parseI18nName(dict.label)"
- :value="dict.value"
- />
- </el-select>
- </el-form-item>
- <el-form-item label="意见" prop="opinion">
- <el-input
- v-model="auditForm.opinion"
- type="textarea"
- :rows="3"
- placeholder="请输入意见"
- />
- </el-form-item>
- <el-form-item label="指定处理人" prop="designatedDealer">
- <el-select
- v-model="auditForm.designatedDealer"
- filterable
- remote
- reserve-keyword
- placeholder="请输入成员昵称搜索"
- :remote-method="searchDealers"
- :loading="dealerSearchLoading"
- style="width: 100%"
- >
- <el-option
- v-for="dealer in dealerOptions"
- :key="dealer.id"
- :label="`${dealer.name} / ${dealer.dept} --- ${dealer.phoneNumber}`"
- :value="dealer.id"
- />
- </el-select>
- </el-form-item>
- <el-form-item label="截止日期" prop="deadline">
- <el-date-picker
- v-model="auditForm.deadline"
- type="date"
- value-format="YYYY-MM-DD"
- placeholder="请选择截止日期"
- style="width: 100%"
- />
- </el-form-item>
- </template>
- </el-form>
- <template #footer>
- <div class="dialog-footer">
- <el-button @click="cancelAudit">取消</el-button>
- <el-button type="primary" :loading="auditLoading" @click="submitAudit">确认</el-button>
- </div>
- </template>
- </el-dialog>
- <!-- 审核记录弹窗 -->
- <QcLogDialog v-model="logDialogVisible" :detail-id="currentLogDetailId" />
- </div>
- </template>
- <script setup lang="ts">
- import { ref, reactive, onMounted, nextTick, computed, toRefs, getCurrentInstance } from 'vue';
- import { ElMessage } from 'element-plus';
- import type { FormInstance } from 'element-plus';
- import { listQcTasks, submitQcTask, auditQcTask } from '@/api/home/taskCenter/qc';
- import { QcTaskVO, QcTaskQuery, QcTaskSubmitForm, QcTaskAuditForm } from '@/api/home/taskCenter/qc/types';
- import { queryMemberNotInCenter } from '@/api/project/management';
- import { MemberNotInCenterVO, MemberNotInCenterQuery } from '@/api/project/management/types';
- import { parseI18nName } from '@/utils/i18n';
- import { formatDocumentName } from '@/utils/ruoyi';
- import fileUpload from '@/components/FileUpload/index.vue';
- import QcLogDialog from '@/components/QcLogDialog/index.vue';
- const { proxy } = getCurrentInstance() as any;
- const { qc_question_type } = toRefs<any>(proxy?.useDict('qc_question_type'));
- const loading = ref(false);
- const submitButtonLoading = ref(false);
- const auditLoading = ref(false);
- // 当前任务
- const currentTask = ref<QcTaskVO | null>(null);
- // 查询参数
- const queryParams = reactive<QcTaskQuery>({
- taskName: '',
- projectName: '',
- documentName: '',
- status: undefined,
- pageNum: 1,
- pageSize: 10
- });
- // 任务列表数据
- const taskList = ref<QcTaskVO[]>([]);
- const total = ref(0);
- // 递交对话框
- const submitDialog = reactive({
- visible: false,
- title: ''
- });
- // 递交表单ref
- const submitFormRef = ref<FormInstance>();
- // 递交表单数据
- const submitForm = ref({
- document: ''
- });
- // 递交表单验证规则
- const submitRules = {
- document: [{ required: true, message: '请上传文件', trigger: 'change' }]
- };
- // 审核相关
- const auditFormRef = ref<FormInstance>();
- const auditDialog = reactive({
- visible: false
- });
- // 审核表单初始数据
- const initAuditForm: QcTaskAuditForm = {
- taskId: 0,
- detailId: 0,
- result: 1,
- rejectionType: undefined,
- opinion: undefined,
- designatedDealer: undefined,
- deadline: undefined
- };
- const auditForm = ref<QcTaskAuditForm>({ ...initAuditForm });
- // 审核表单验证规则
- const auditRules = computed(() => ({
- result: [{ required: true, message: '请选择审核结果', trigger: 'change' }],
- rejectionType: auditForm.value.result === 2 ? [{ required: true, message: '请选择问题类型', trigger: 'change' }] : [],
- opinion: auditForm.value.result === 2 ? [{ required: true, message: '请输入意见', trigger: 'blur' }] : [],
- designatedDealer: auditForm.value.result === 2 ? [{ required: true, message: '请选择指定处理人', trigger: 'change' }] : [],
- deadline: auditForm.value.result === 2 ? [{ required: true, message: '请选择截止日期', trigger: 'change' }] : []
- }));
- // 处理人搜索相关
- const dealerSearchLoading = ref(false);
- const dealerOptions = ref<MemberNotInCenterVO[]>([]);
- let dealerSearchTimer: NodeJS.Timeout | null = null;
- // 审核记录相关
- const logDialogVisible = ref(false);
- const currentLogDetailId = ref<number | undefined>(undefined);
- /**
- * 获取任务列表
- */
- const getTaskList = async () => {
- try {
- loading.value = true;
- const res = await listQcTasks(queryParams);
- if (res.code === 200) {
- taskList.value = res.rows || [];
- total.value = res.total || 0;
- } else {
- ElMessage.error(res.msg || '获取任务列表失败');
- }
- } catch (error) {
- console.error('获取任务列表失败:', error);
- ElMessage.error('获取任务列表失败');
- } finally {
- loading.value = false;
- }
- };
- /**
- * 处理搜索
- */
- const handleQuery = () => {
- queryParams.pageNum = 1;
- getTaskList();
- };
- /**
- * 重置搜索
- */
- const resetQuery = () => {
- queryParams.taskName = '';
- queryParams.projectName = '';
- queryParams.documentName = '';
- queryParams.status = undefined;
- handleQuery();
- };
- /**
- * 处理递交
- */
- const handleSubmit = (row: QcTaskVO) => {
- currentTask.value = row;
- submitForm.value = {
- document: ''
- };
- submitDialog.visible = true;
- submitDialog.title = '递交文档';
- nextTick(() => {
- submitFormRef.value?.clearValidate();
- });
- };
- /**
- * 取消递交
- */
- const cancelSubmit = () => {
- submitDialog.visible = false;
- submitForm.value = {
- document: ''
- };
- currentTask.value = null;
- };
- /**
- * 提交递交表单
- */
- const submitSubmitForm = () => {
- submitFormRef.value?.validate(async (valid: boolean) => {
- if (valid && currentTask.value) {
- submitButtonLoading.value = true;
- try {
- const submitData: QcTaskSubmitForm = {
- detailId: currentTask.value.id,
- document: Number(submitForm.value.document)
- };
- await submitQcTask(submitData);
- ElMessage.success('递交成功');
- submitDialog.visible = false;
- await getTaskList();
- } catch (error) {
- console.error('递交失败:', error);
- ElMessage.error('递交失败');
- } finally {
- submitButtonLoading.value = false;
- }
- }
- });
- };
- /** 审核任务 */
- const handleAudit = (row: QcTaskVO) => {
- currentTask.value = row;
- auditForm.value = {
- ...initAuditForm,
- taskId: row.taskId || 0,
- detailId: row.id
- };
- dealerOptions.value = [];
- auditDialog.visible = true;
- };
- /** 搜索处理人 */
- const searchDealers = async (query: string) => {
- if (!query || query.trim() === '') {
- dealerOptions.value = [];
- return;
- }
- if (dealerSearchTimer) {
- clearTimeout(dealerSearchTimer);
- }
- dealerSearchTimer = setTimeout(async () => {
- dealerSearchLoading.value = true;
- try {
- const queryParams: MemberNotInCenterQuery = {
- pageNum: 1,
- pageSize: 10,
- projectId: currentTask.value?.projectId || 0,
- folderId: 0,
- name: query
- };
- const res = await queryMemberNotInCenter(queryParams);
- dealerOptions.value = res.rows || [];
- } catch (error) {
- console.error('搜索处理人失败:', error);
- ElMessage.error('搜索处理人失败');
- } finally {
- dealerSearchLoading.value = false;
- }
- }, 300);
- };
- /** 取消审核 */
- const cancelAudit = () => {
- auditDialog.visible = false;
- auditFormRef.value?.resetFields();
- currentTask.value = null;
- };
- /** 提交审核 */
- const submitAudit = () => {
- auditFormRef.value?.validate(async (valid: boolean) => {
- if (valid) {
- auditLoading.value = true;
- try {
- const submitData: QcTaskAuditForm = {
- taskId: auditForm.value.taskId,
- detailId: auditForm.value.detailId,
- result: auditForm.value.result
- };
- // 驳回时添加额外字段
- if (auditForm.value.result === 2) {
- submitData.rejectionType = auditForm.value.rejectionType;
- submitData.opinion = auditForm.value.opinion;
- submitData.designatedDealer = auditForm.value.designatedDealer;
- submitData.deadline = auditForm.value.deadline;
- }
- await auditQcTask(submitData);
- ElMessage.success('审核成功');
- auditDialog.visible = false;
- await getTaskList();
- } catch (error) {
- console.error('审核失败:', error);
- } finally {
- auditLoading.value = false;
- }
- }
- });
- };
- /** 查看审核记录 */
- const handleViewLog = (row: QcTaskVO) => {
- currentLogDetailId.value = row.id;
- logDialogVisible.value = true;
- };
- onMounted(() => {
- getTaskList();
- });
- </script>
- <style scoped lang="scss">
- .qc-task-container {
- padding: 20px;
- .flex {
- display: flex;
- }
- .justify-between {
- justify-content: space-between;
- }
- .items-center {
- align-items: center;
- }
- .text-lg {
- font-size: 1.125rem;
- }
- .font-bold {
- font-weight: 700;
- }
- .search-form {
- margin-bottom: 15px;
- }
- :deep(.el-table) {
- .cell {
- white-space: nowrap;
- }
- }
- }
- </style>
|