index.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515
  1. <template>
  2. <div class="qc-task-container">
  3. <el-card shadow="never">
  4. <template #header>
  5. <div class="flex justify-between items-center">
  6. <span class="text-lg font-bold">文件质控任务</span>
  7. </div>
  8. </template>
  9. <!-- 搜索栏 -->
  10. <el-form :model="queryParams" :inline="true" class="search-form">
  11. <el-form-item label="任务名称">
  12. <el-input v-model="queryParams.taskName" placeholder="请输入任务名称" clearable style="width: 200px"
  13. @keyup.enter="handleQuery" />
  14. </el-form-item>
  15. <el-form-item label="项目名称">
  16. <el-input v-model="queryParams.projectName" placeholder="请输入项目名称" clearable style="width: 200px"
  17. @keyup.enter="handleQuery" />
  18. </el-form-item>
  19. <el-form-item label="文档名称">
  20. <el-input v-model="queryParams.documentName" placeholder="请输入文档名称" clearable style="width: 200px"
  21. @keyup.enter="handleQuery" />
  22. </el-form-item>
  23. <el-form-item label="状态">
  24. <el-select v-model="queryParams.status" placeholder="请选择状态" clearable style="width: 120px">
  25. <el-option label="待审核" :value="0" />
  26. <el-option label="审核通过" :value="1" />
  27. <el-option label="审核拒绝" :value="2" />
  28. </el-select>
  29. </el-form-item>
  30. <el-form-item>
  31. <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
  32. <el-button icon="Refresh" @click="resetQuery">重置</el-button>
  33. </el-form-item>
  34. </el-form>
  35. <!-- 任务列表 -->
  36. <el-table v-loading="loading" :data="taskList" border style="margin-top: 10px">
  37. <el-table-column prop="id" label="序号" width="80" align="center" />
  38. <el-table-column prop="name" label="任务名称" min-width="150" show-overflow-tooltip>
  39. <template #default="scope">
  40. {{ formatDocumentName(scope.row.name) }}
  41. </template>
  42. </el-table-column>
  43. <el-table-column prop="projectName" label="项目名称" min-width="150" show-overflow-tooltip />
  44. <el-table-column prop="initiator" label="发起人" width="120" align="center" />
  45. <el-table-column prop="executor" label="执行人" width="120" align="center" />
  46. <el-table-column prop="status" label="状态" width="120" align="center">
  47. <template #default="scope">
  48. <el-tag v-if="scope.row.status === 0 || scope.row.status === 3" type="info">待审核</el-tag>
  49. <el-tag v-else-if="scope.row.status === 1" type="success">审核通过</el-tag>
  50. <el-tag v-else-if="scope.row.status === 2" type="danger">审核拒绝</el-tag>
  51. <el-tag v-else type="warning">未知</el-tag>
  52. </template>
  53. </el-table-column>
  54. <el-table-column prop="note" label="备注" min-width="150" show-overflow-tooltip />
  55. <el-table-column prop="executeTime" label="执行时间" width="180" align="center">
  56. <template #default="scope">
  57. <span v-if="scope.row.executeTime">{{ scope.row.executeTime }}</span>
  58. <span v-else>-</span>
  59. </template>
  60. </el-table-column>
  61. <el-table-column prop="createTime" label="创建时间" width="180" align="center">
  62. <template #default="scope">
  63. <span v-if="scope.row.createTime">{{ scope.row.createTime }}</span>
  64. <span v-else>-</span>
  65. </template>
  66. </el-table-column>
  67. <!-- 操作列 -->
  68. <el-table-column label="操作" width="280" align="center" fixed="right" class-name="small-padding fixed-width">
  69. <template #default="scope">
  70. <el-button
  71. v-if="scope.row.status === 2"
  72. v-hasPermi="['taskCenter:qc:submit']"
  73. type="primary"
  74. icon="Upload"
  75. style="padding: 0 5px; font-size: 10px; height: 24px"
  76. @click="handleSubmit(scope.row)"
  77. >
  78. 递交
  79. </el-button>
  80. <el-button
  81. v-if="scope.row.status === 0 || scope.row.status === 3"
  82. v-hasPermi="['taskCenter:qc:audit']"
  83. type="success"
  84. icon="Check"
  85. style="padding: 0 5px; font-size: 10px; height: 24px"
  86. @click="handleAudit(scope.row)"
  87. >
  88. 审核
  89. </el-button>
  90. <el-button
  91. v-hasPermi="['taskCenter:qc:logAudit']"
  92. type="warning"
  93. icon="Document"
  94. style="padding: 0 5px; font-size: 10px; height: 24px"
  95. @click="handleViewLog(scope.row)"
  96. >
  97. 查看审核记录
  98. </el-button>
  99. </template>
  100. </el-table-column>
  101. </el-table>
  102. <!-- 分页 -->
  103. <pagination v-show="total > 0" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize"
  104. :total="total" @pagination="getTaskList" />
  105. </el-card>
  106. <!-- 递交文档对话框 -->
  107. <el-dialog v-model="submitDialog.visible" :title="submitDialog.title" width="500px" append-to-body>
  108. <el-alert title="仅支持上传 PDF 格式文件" type="info" :closable="false" style="margin-bottom: 20px" />
  109. <el-form ref="submitFormRef" :model="submitForm" :rules="submitRules" label-width="100px">
  110. <el-form-item label="文件" prop="document">
  111. <fileUpload v-model="submitForm.document" :limit="1" :file-type="['pdf']" :is-show-tip="false" />
  112. </el-form-item>
  113. </el-form>
  114. <template #footer>
  115. <div class="dialog-footer">
  116. <el-button @click="cancelSubmit">取消</el-button>
  117. <el-button type="primary" :loading="submitButtonLoading" @click="submitSubmitForm">确认</el-button>
  118. </div>
  119. </template>
  120. </el-dialog>
  121. <!-- 审核弹窗 -->
  122. <el-dialog v-model="auditDialog.visible" title="审核" width="500px" append-to-body>
  123. <el-form ref="auditFormRef" :model="auditForm" :rules="auditRules" label-width="100px">
  124. <el-form-item label="结果" prop="result">
  125. <el-radio-group v-model="auditForm.result">
  126. <el-radio :label="1">通过</el-radio>
  127. <el-radio :label="2">驳回</el-radio>
  128. </el-radio-group>
  129. </el-form-item>
  130. <template v-if="auditForm.result === 2">
  131. <el-form-item label="问题类型" prop="rejectionType">
  132. <el-select v-model="auditForm.rejectionType" placeholder="请选择问题类型" style="width: 100%">
  133. <el-option
  134. v-for="dict in qc_question_type"
  135. :key="dict.value"
  136. :label="parseI18nName(dict.label)"
  137. :value="dict.value"
  138. />
  139. </el-select>
  140. </el-form-item>
  141. <el-form-item label="意见" prop="opinion">
  142. <el-input
  143. v-model="auditForm.opinion"
  144. type="textarea"
  145. :rows="3"
  146. placeholder="请输入意见"
  147. />
  148. </el-form-item>
  149. <el-form-item label="指定处理人" prop="designatedDealer">
  150. <el-select
  151. v-model="auditForm.designatedDealer"
  152. filterable
  153. remote
  154. reserve-keyword
  155. placeholder="请输入成员昵称搜索"
  156. :remote-method="searchDealers"
  157. :loading="dealerSearchLoading"
  158. style="width: 100%"
  159. >
  160. <el-option
  161. v-for="dealer in dealerOptions"
  162. :key="dealer.id"
  163. :label="`${dealer.name} / ${dealer.dept} --- ${dealer.phoneNumber}`"
  164. :value="dealer.id"
  165. />
  166. </el-select>
  167. </el-form-item>
  168. <el-form-item label="截止日期" prop="deadline">
  169. <el-date-picker
  170. v-model="auditForm.deadline"
  171. type="date"
  172. value-format="YYYY-MM-DD"
  173. placeholder="请选择截止日期"
  174. style="width: 100%"
  175. />
  176. </el-form-item>
  177. </template>
  178. </el-form>
  179. <template #footer>
  180. <div class="dialog-footer">
  181. <el-button @click="cancelAudit">取消</el-button>
  182. <el-button type="primary" :loading="auditLoading" @click="submitAudit">确认</el-button>
  183. </div>
  184. </template>
  185. </el-dialog>
  186. <!-- 审核记录弹窗 -->
  187. <QcLogDialog v-model="logDialogVisible" :detail-id="currentLogDetailId" />
  188. </div>
  189. </template>
  190. <script setup lang="ts">
  191. import { ref, reactive, onMounted, nextTick, computed, toRefs, getCurrentInstance } from 'vue';
  192. import { ElMessage } from 'element-plus';
  193. import type { FormInstance } from 'element-plus';
  194. import { listQcTasks, submitQcTask, auditQcTask } from '@/api/home/taskCenter/qc';
  195. import { QcTaskVO, QcTaskQuery, QcTaskSubmitForm, QcTaskAuditForm } from '@/api/home/taskCenter/qc/types';
  196. import { queryMemberNotInCenter } from '@/api/project/management';
  197. import { MemberNotInCenterVO, MemberNotInCenterQuery } from '@/api/project/management/types';
  198. import { parseI18nName } from '@/utils/i18n';
  199. import { formatDocumentName } from '@/utils/ruoyi';
  200. import fileUpload from '@/components/FileUpload/index.vue';
  201. import QcLogDialog from '@/components/QcLogDialog/index.vue';
  202. const { proxy } = getCurrentInstance() as any;
  203. const { qc_question_type } = toRefs<any>(proxy?.useDict('qc_question_type'));
  204. const loading = ref(false);
  205. const submitButtonLoading = ref(false);
  206. const auditLoading = ref(false);
  207. // 当前任务
  208. const currentTask = ref<QcTaskVO | null>(null);
  209. // 查询参数
  210. const queryParams = reactive<QcTaskQuery>({
  211. taskName: '',
  212. projectName: '',
  213. documentName: '',
  214. status: undefined,
  215. pageNum: 1,
  216. pageSize: 10
  217. });
  218. // 任务列表数据
  219. const taskList = ref<QcTaskVO[]>([]);
  220. const total = ref(0);
  221. // 递交对话框
  222. const submitDialog = reactive({
  223. visible: false,
  224. title: ''
  225. });
  226. // 递交表单ref
  227. const submitFormRef = ref<FormInstance>();
  228. // 递交表单数据
  229. const submitForm = ref({
  230. document: ''
  231. });
  232. // 递交表单验证规则
  233. const submitRules = {
  234. document: [{ required: true, message: '请上传文件', trigger: 'change' }]
  235. };
  236. // 审核相关
  237. const auditFormRef = ref<FormInstance>();
  238. const auditDialog = reactive({
  239. visible: false
  240. });
  241. // 审核表单初始数据
  242. const initAuditForm: QcTaskAuditForm = {
  243. taskId: 0,
  244. detailId: 0,
  245. result: 1,
  246. rejectionType: undefined,
  247. opinion: undefined,
  248. designatedDealer: undefined,
  249. deadline: undefined
  250. };
  251. const auditForm = ref<QcTaskAuditForm>({ ...initAuditForm });
  252. // 审核表单验证规则
  253. const auditRules = computed(() => ({
  254. result: [{ required: true, message: '请选择审核结果', trigger: 'change' }],
  255. rejectionType: auditForm.value.result === 2 ? [{ required: true, message: '请选择问题类型', trigger: 'change' }] : [],
  256. opinion: auditForm.value.result === 2 ? [{ required: true, message: '请输入意见', trigger: 'blur' }] : [],
  257. designatedDealer: auditForm.value.result === 2 ? [{ required: true, message: '请选择指定处理人', trigger: 'change' }] : [],
  258. deadline: auditForm.value.result === 2 ? [{ required: true, message: '请选择截止日期', trigger: 'change' }] : []
  259. }));
  260. // 处理人搜索相关
  261. const dealerSearchLoading = ref(false);
  262. const dealerOptions = ref<MemberNotInCenterVO[]>([]);
  263. let dealerSearchTimer: NodeJS.Timeout | null = null;
  264. // 审核记录相关
  265. const logDialogVisible = ref(false);
  266. const currentLogDetailId = ref<number | undefined>(undefined);
  267. /**
  268. * 获取任务列表
  269. */
  270. const getTaskList = async () => {
  271. try {
  272. loading.value = true;
  273. const res = await listQcTasks(queryParams);
  274. if (res.code === 200) {
  275. taskList.value = res.rows || [];
  276. total.value = res.total || 0;
  277. } else {
  278. ElMessage.error(res.msg || '获取任务列表失败');
  279. }
  280. } catch (error) {
  281. console.error('获取任务列表失败:', error);
  282. ElMessage.error('获取任务列表失败');
  283. } finally {
  284. loading.value = false;
  285. }
  286. };
  287. /**
  288. * 处理搜索
  289. */
  290. const handleQuery = () => {
  291. queryParams.pageNum = 1;
  292. getTaskList();
  293. };
  294. /**
  295. * 重置搜索
  296. */
  297. const resetQuery = () => {
  298. queryParams.taskName = '';
  299. queryParams.projectName = '';
  300. queryParams.documentName = '';
  301. queryParams.status = undefined;
  302. handleQuery();
  303. };
  304. /**
  305. * 处理递交
  306. */
  307. const handleSubmit = (row: QcTaskVO) => {
  308. currentTask.value = row;
  309. submitForm.value = {
  310. document: ''
  311. };
  312. submitDialog.visible = true;
  313. submitDialog.title = '递交文档';
  314. nextTick(() => {
  315. submitFormRef.value?.clearValidate();
  316. });
  317. };
  318. /**
  319. * 取消递交
  320. */
  321. const cancelSubmit = () => {
  322. submitDialog.visible = false;
  323. submitForm.value = {
  324. document: ''
  325. };
  326. currentTask.value = null;
  327. };
  328. /**
  329. * 提交递交表单
  330. */
  331. const submitSubmitForm = () => {
  332. submitFormRef.value?.validate(async (valid: boolean) => {
  333. if (valid && currentTask.value) {
  334. submitButtonLoading.value = true;
  335. try {
  336. const submitData: QcTaskSubmitForm = {
  337. detailId: currentTask.value.id,
  338. document: Number(submitForm.value.document)
  339. };
  340. await submitQcTask(submitData);
  341. ElMessage.success('递交成功');
  342. submitDialog.visible = false;
  343. await getTaskList();
  344. } catch (error) {
  345. console.error('递交失败:', error);
  346. ElMessage.error('递交失败');
  347. } finally {
  348. submitButtonLoading.value = false;
  349. }
  350. }
  351. });
  352. };
  353. /** 审核任务 */
  354. const handleAudit = (row: QcTaskVO) => {
  355. currentTask.value = row;
  356. auditForm.value = {
  357. ...initAuditForm,
  358. taskId: row.taskId || 0,
  359. detailId: row.id
  360. };
  361. dealerOptions.value = [];
  362. auditDialog.visible = true;
  363. };
  364. /** 搜索处理人 */
  365. const searchDealers = async (query: string) => {
  366. if (!query || query.trim() === '') {
  367. dealerOptions.value = [];
  368. return;
  369. }
  370. if (dealerSearchTimer) {
  371. clearTimeout(dealerSearchTimer);
  372. }
  373. dealerSearchTimer = setTimeout(async () => {
  374. dealerSearchLoading.value = true;
  375. try {
  376. const queryParams: MemberNotInCenterQuery = {
  377. pageNum: 1,
  378. pageSize: 10,
  379. projectId: currentTask.value?.projectId || 0,
  380. folderId: 0,
  381. name: query
  382. };
  383. const res = await queryMemberNotInCenter(queryParams);
  384. dealerOptions.value = res.rows || [];
  385. } catch (error) {
  386. console.error('搜索处理人失败:', error);
  387. ElMessage.error('搜索处理人失败');
  388. } finally {
  389. dealerSearchLoading.value = false;
  390. }
  391. }, 300);
  392. };
  393. /** 取消审核 */
  394. const cancelAudit = () => {
  395. auditDialog.visible = false;
  396. auditFormRef.value?.resetFields();
  397. currentTask.value = null;
  398. };
  399. /** 提交审核 */
  400. const submitAudit = () => {
  401. auditFormRef.value?.validate(async (valid: boolean) => {
  402. if (valid) {
  403. auditLoading.value = true;
  404. try {
  405. const submitData: QcTaskAuditForm = {
  406. taskId: auditForm.value.taskId,
  407. detailId: auditForm.value.detailId,
  408. result: auditForm.value.result
  409. };
  410. // 驳回时添加额外字段
  411. if (auditForm.value.result === 2) {
  412. submitData.rejectionType = auditForm.value.rejectionType;
  413. submitData.opinion = auditForm.value.opinion;
  414. submitData.designatedDealer = auditForm.value.designatedDealer;
  415. submitData.deadline = auditForm.value.deadline;
  416. }
  417. await auditQcTask(submitData);
  418. ElMessage.success('审核成功');
  419. auditDialog.visible = false;
  420. await getTaskList();
  421. } catch (error) {
  422. console.error('审核失败:', error);
  423. } finally {
  424. auditLoading.value = false;
  425. }
  426. }
  427. });
  428. };
  429. /** 查看审核记录 */
  430. const handleViewLog = (row: QcTaskVO) => {
  431. currentLogDetailId.value = row.id;
  432. logDialogVisible.value = true;
  433. };
  434. onMounted(() => {
  435. getTaskList();
  436. });
  437. </script>
  438. <style scoped lang="scss">
  439. .qc-task-container {
  440. padding: 20px;
  441. .flex {
  442. display: flex;
  443. }
  444. .justify-between {
  445. justify-content: space-between;
  446. }
  447. .items-center {
  448. align-items: center;
  449. }
  450. .text-lg {
  451. font-size: 1.125rem;
  452. }
  453. .font-bold {
  454. font-weight: 700;
  455. }
  456. .search-form {
  457. margin-bottom: 15px;
  458. }
  459. :deep(.el-table) {
  460. .cell {
  461. white-space: nowrap;
  462. }
  463. }
  464. }
  465. </style>