This commit is contained in:
吴红兵
2026-03-07 12:35:45 +08:00
parent 271710e870
commit b997b3ba48
423 changed files with 79612 additions and 91574 deletions

View File

@@ -1,249 +1,230 @@
<template>
<el-dialog v-model="dialogVisible" title="编辑教师资格证" width="600px" append-to-body :close-on-click-modal="false" destroy-on-close>
<div v-if="showForm">
<el-form
ref="formRef"
:model="dataForm"
:rules="formRules"
label-width="120px"
>
<el-form-item label="类型" prop="certificateConfId">
<el-select
v-model="dataForm.certificateConfId"
placeholder="请选择类型"
style="width: 100%"
>
<el-option
v-for="item in teacherCertificateList"
:key="item.id"
:label="item.cretificateName"
:value="item.id"
/>
</el-select>
</el-form-item>
<el-dialog v-model="dialogVisible" title="编辑教师资格证" width="600px" append-to-body :close-on-click-modal="false" destroy-on-close>
<div v-if="showForm">
<el-form ref="formRef" :model="dataForm" :rules="formRules" label-width="120px">
<el-form-item label="类型" prop="certificateConfId">
<el-select v-model="dataForm.certificateConfId" placeholder="请选择类型" style="width: 100%">
<el-option v-for="item in teacherCertificateList" :key="item.id" :label="item.cretificateName" :value="item.id" />
</el-select>
</el-form-item>
<el-form-item label="证书编号" prop="certificateNumber">
<el-input
v-model="dataForm.certificateNumber"
placeholder="请输入证书编号(仅支持英文和数字)"
show-word-limit
maxlength="64"
@input="handleCertificateNumberInput"
/>
</el-form-item>
<el-form-item label="证书编号" prop="certificateNumber">
<el-input
v-model="dataForm.certificateNumber"
placeholder="请输入证书编号(仅支持英文和数字)"
show-word-limit
maxlength="64"
@input="handleCertificateNumberInput"
/>
</el-form-item>
<el-form-item label="证明材料" prop="evidenceA" required>
<el-upload
:headers="headers"
:limit="1"
:action="url"
:file-list="fileList"
:on-success="materiaUploadSuccess"
:accept="'.jpg,.jpeg,.png,.pdf'"
>
<el-button size="small" type="primary">点击上传</el-button>
<template #tip>
<div style="margin-top: 8px;">
<el-tag>仅支持jpg,jpeg,png,pdf后缀的文件上传</el-tag>
</div>
</template>
</el-upload>
</el-form-item>
</el-form>
</div>
<template #footer>
<div class="dialog-footer">
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="dialogSubmit" :loading="submitLoading">确定</el-button>
</div>
</template>
</el-dialog>
<el-form-item label="证明材料" prop="evidenceA" required>
<el-upload
:headers="headers"
:limit="1"
:action="url"
:file-list="fileList"
:on-success="materiaUploadSuccess"
:accept="'.jpg,.jpeg,.png,.pdf'"
>
<el-button size="small" type="primary">点击上传</el-button>
<template #tip>
<div style="margin-top: 8px">
<el-tag>仅支持jpg,jpeg,png,pdf后缀的文件上传</el-tag>
</div>
</template>
</el-upload>
</el-form-item>
</el-form>
</div>
<template #footer>
<div class="dialog-footer">
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="dialogSubmit" :loading="submitLoading">确定</el-button>
</div>
</template>
</el-dialog>
</template>
<script setup lang="ts">
import { ref, reactive, computed } from 'vue'
import { Session } from '/@/utils/storage'
import { useMessage } from '/@/hooks/message'
import { addObj } from '/@/api/professional/professionaluser/professionalteachercertificaterelation'
import { getTeacherCertificateList } from '/@/api/professional/rsbase/professionalteachercertificateconf'
import { checkLocked } from '/@/api/professional/professionalstatuslock'
import { ref, reactive, computed } from 'vue';
import { Session } from '/@/utils/storage';
import { useMessage } from '/@/hooks/message';
import { addObj } from '/@/api/professional/professionaluser/professionalteachercertificaterelation';
import { getTeacherCertificateList } from '/@/api/professional/rsbase/professionalteachercertificateconf';
import { checkLocked } from '/@/api/professional/professionalstatuslock';
// Emits
const emit = defineEmits<{
(e: 'refreshData'): void
}>()
(e: 'refreshData'): void;
}>();
// 消息提示
const message = useMessage()
const message = useMessage();
// 表单引用
const formRef = ref()
const submitLoading = ref(false)
const formRef = ref();
const submitLoading = ref(false);
// 对话框显示状态(内部管理)
const dialogVisible = ref(false)
const dialogVisible = ref(false);
// 表单数据
const dataForm = reactive({
certificateConfId: '',
certificateNumber: '',
evidenceA: '',
state: '',
id: ''
})
certificateConfId: '',
certificateNumber: '',
evidenceA: '',
state: '',
id: '',
});
// 表单验证规则
const formRules = {
certificateConfId: [
{ required: true, message: '请选择类型', trigger: 'change' }
],
certificateNumber: [
{ required: true, message: '请输入证书编号', trigger: 'blur' },
{ pattern: /^[A-Za-z0-9]+$/, message: '证书编号只能包含英文和数字', trigger: 'blur' }
],
evidenceA: [
{ required: true, message: '请上传证明材料', trigger: 'change' }
]
}
certificateConfId: [{ required: true, message: '请选择类型', trigger: 'change' }],
certificateNumber: [
{ required: true, message: '请输入证书编号', trigger: 'blur' },
{ pattern: /^[A-Za-z0-9]+$/, message: '证书编号只能包含英文和数字', trigger: 'blur' },
],
evidenceA: [{ required: true, message: '请上传证明材料', trigger: 'change' }],
};
// 教师资格证列表
const teacherCertificateList = ref<any[]>([])
const teacherCertificateList = ref<any[]>([]);
// 基础信息(简化,只保留需要的)
// 上传相关
const url = ref('')
const fileList = ref<any[]>([])
const url = ref('');
const fileList = ref<any[]>([]);
// 请求头
const headers = computed(() => {
return {
"Authorization": 'Bearer ' + Session.getToken()
}
})
return {
Authorization: 'Bearer ' + Session.getToken(),
};
});
// 显示表单标志
const showForm = ref(false)
const showForm = ref(false);
// 初始化字典数据
const initDicData = async () => {
try {
const res = await getTeacherCertificateList()
teacherCertificateList.value = res.data || []
} catch (error) {
// 获取字典数据失败
}
}
try {
const res = await getTeacherCertificateList();
teacherCertificateList.value = res.data || [];
} catch (error) {
// 获取字典数据失败
}
};
// 证书编号输入处理(只允许英文和数字)
const handleCertificateNumberInput = (value: string) => {
dataForm.certificateNumber = value.replace(/[^A-Za-z0-9]/g, '')
}
dataForm.certificateNumber = value.replace(/[^A-Za-z0-9]/g, '');
};
// 文件上传成功
const materiaUploadSuccess = (response: any) => {
if (response.data && response.data.code === "-1") {
message.error("当前不允许上传文件")
return
}
dataForm.evidenceA = response.data.url
// 清除验证错误
formRef.value?.clearValidate('evidenceA')
}
if (response.data && response.data.code === '-1') {
message.error('当前不允许上传文件');
return;
}
dataForm.evidenceA = response.data.url;
// 清除验证错误
formRef.value?.clearValidate('evidenceA');
};
// 打开对话框
const openDialog = async (row?: any) => {
// 新增模式:先检查是否锁定
if (!row || !row.id) {
try {
const lockResponse = await checkLocked('teacherTitle')
if (lockResponse.data) {
message.warning("新增功能已锁定,暂不允许操作")
return
}
} catch (error) {
// 错误处理已在数据请求层统一处理,此处不需要提示
return
}
}
// 公共设置
url.value = `/professional/file/teacherAboutInfoUpload?type=0`
fileList.value = []
// 根据是否有 row 设置不同的表单数据
if (row && row.id) {
// 编辑模式:使用传入的数据
Object.assign(dataForm, {
certificateConfId: row.certificateConfId || '',
certificateNumber: row.certificateNumber || '',
evidenceA: row.evidenceA || '',
state: row.state || '',
id: row.id || ''
})
} else {
// 新增模式:重置为空
Object.assign(dataForm, {
certificateConfId: '',
certificateNumber: '',
evidenceA: '',
state: '',
id: ''
})
}
// 公共操作:加载字典数据并显示对话框
await initDicData()
showForm.value = true
dialogVisible.value = true
}
// 新增模式:先检查是否锁定
if (!row || !row.id) {
try {
const lockResponse = await checkLocked('teacherTitle');
if (lockResponse.data) {
message.warning('新增功能已锁定,暂不允许操作');
return;
}
} catch (error) {
// 错误处理已在数据请求层统一处理,此处不需要提示
return;
}
}
// 公共设置
url.value = `/professional/file/teacherAboutInfoUpload?type=0`;
fileList.value = [];
// 根据是否有 row 设置不同的表单数据
if (row && row.id) {
// 编辑模式:使用传入的数据
Object.assign(dataForm, {
certificateConfId: row.certificateConfId || '',
certificateNumber: row.certificateNumber || '',
evidenceA: row.evidenceA || '',
state: row.state || '',
id: row.id || '',
});
} else {
// 新增模式:重置为空
Object.assign(dataForm, {
certificateConfId: '',
certificateNumber: '',
evidenceA: '',
state: '',
id: '',
});
}
// 公共操作:加载字典数据并显示对话框
await initDicData();
showForm.value = true;
dialogVisible.value = true;
};
// 提交表单
const dialogSubmit = async () => {
if (!formRef.value) return
await formRef.value.validate(async (valid: boolean) => {
if (valid) {
submitLoading.value = true
try {
// 统一使用 addObj 接口(新增和编辑都使用)
const submitData: any = {
certificateConfId: dataForm.certificateConfId,
certificateNumber: dataForm.certificateNumber,
evidenceA: dataForm.evidenceA,
state: '' // 待审核
}
// 编辑时需要传递 id
if (dataForm.id) {
submitData.id = dataForm.id
}
await addObj(submitData)
message.success(dataForm.id ? "修改成功" : "提交成功")
dialogVisible.value = false
emit('refreshData')
} catch (error: any) {
// 错误处理已在数据请求层统一处理,此处不需要提示
} finally {
submitLoading.value = false
}
}
})
}
if (!formRef.value) return;
await formRef.value.validate(async (valid: boolean) => {
if (valid) {
submitLoading.value = true;
try {
// 统一使用 addObj 接口(新增和编辑都使用)
const submitData: any = {
certificateConfId: dataForm.certificateConfId,
certificateNumber: dataForm.certificateNumber,
evidenceA: dataForm.evidenceA,
state: '', // 待审核
};
// 编辑时需要传递 id
if (dataForm.id) {
submitData.id = dataForm.id;
}
await addObj(submitData);
message.success(dataForm.id ? '修改成功' : '提交成功');
dialogVisible.value = false;
emit('refreshData');
} catch (error: any) {
// 错误处理已在数据请求层统一处理,此处不需要提示
} finally {
submitLoading.value = false;
}
}
});
};
// 初始化方法(保持兼容性)
const init = () => {
// 新版本不再需要初始化配置
}
// 新版本不再需要初始化配置
};
// 暴露方法
defineExpose({
openDialog,
init
})
openDialog,
init,
});
</script>
<style scoped>
</style>
<style scoped></style>

View File

@@ -1,413 +1,389 @@
<template>
<div class="layout-padding">
<div class="layout-padding-auto layout-padding-view">
<!-- 搜索表单 -->
<search-form
v-show="showSearch"
:model="search"
ref="searchFormRef"
@keyup-enter="handleFilter"
>
<template #default="{ visible }">
<template v-if="visible">
<el-form-item label="审核状态" prop="state">
<el-select
v-model="search.state"
clearable
placeholder="请选择审核状态"
>
<el-option
v-for="item in professionalState"
:key="item.value"
:label="item.label"
:value="item.value"
/>
</el-select>
</el-form-item>
<div class="layout-padding">
<div class="layout-padding-auto layout-padding-view">
<!-- 搜索表单 -->
<search-form v-show="showSearch" :model="search" ref="searchFormRef" @keyup-enter="handleFilter">
<template #default="{ visible }">
<template v-if="visible">
<el-form-item label="审核状态" prop="state">
<el-select v-model="search.state" clearable placeholder="请选择审核状态">
<el-option v-for="item in professionalState" :key="item.value" :label="item.label" :value="item.value" />
</el-select>
</el-form-item>
<el-form-item label="工号" prop="teacherNo">
<el-input
v-model="search.teacherNo"
clearable
placeholder="请输入工号"
/>
</el-form-item>
<el-form-item label="工号" prop="teacherNo">
<el-input v-model="search.teacherNo" clearable placeholder="请输入工号" />
</el-form-item>
<el-form-item label="姓名" prop="realName">
<el-input
v-model="search.realName"
clearable
placeholder="请输入姓名"
/>
</el-form-item>
</template>
</template>
<el-form-item label="姓名" prop="realName">
<el-input v-model="search.realName" clearable placeholder="请输入姓名" />
</el-form-item>
</template>
</template>
<!-- 操作按钮 -->
<template #actions>
<el-form-item>
<el-button type="primary" @click="handleFilter" icon="Search">查询</el-button>
<el-button @click="resetQuery" icon="Refresh">重置</el-button>
</el-form-item>
</template>
</search-form>
<!-- 操作按钮 -->
<template #actions>
<el-form-item>
<el-button type="primary" @click="handleFilter" icon="Search">查询</el-button>
<el-button @click="resetQuery" icon="Refresh">重置</el-button>
</el-form-item>
</template>
</search-form>
<!-- 操作按钮 -->
<el-row>
<div class="mb15" style="width: 100%;">
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_add')"
type="primary"
icon="FolderAdd"
@click="handleAdd">
</el-button>
<el-button
v-if="hasAuth('professional_teacherbase_export')"
type="warning"
plain
icon="Download"
@click="handleDownLoadWord"
:loading="exportLoading">导出信息
</el-button>
<el-button
v-auth="'professional_teacherinfo_import'"
type="primary"
plain
icon="UploadFilled"
:loading="exportLoading"
@click="handleImportDialog">导入信息
</el-button>
</div>
</el-row>
<!-- 操作按钮 -->
<el-row>
<div class="mb15" style="width: 100%">
<el-button v-if="hasAuth('professional_professionalteachercertificaterelation_add')" type="primary" icon="FolderAdd" @click="handleAdd"
>
</el-button>
<el-button
v-if="hasAuth('professional_teacherbase_export')"
type="warning"
plain
icon="Download"
@click="handleDownLoadWord"
:loading="exportLoading"
>导出信息
</el-button>
<el-button
v-auth="'professional_teacherinfo_import'"
type="primary"
plain
icon="UploadFilled"
:loading="exportLoading"
@click="handleImportDialog"
>导入信息
</el-button>
</div>
</el-row>
<!-- 表格 -->
<el-table
ref="tableRef"
:data="state.dataList"
row-key="id"
v-loading="state.loading"
border
:cell-style="tableStyle.cellStyle"
:header-cell-style="tableStyle.headerCellStyle"
>
<el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="state" label="审核状态" width="120" align="center">
<template #default="scope">
<AuditState :state="scope.row.state" :options="auditStateOptions" />
</template>
</el-table-column>
<el-table-column label="姓名/工号" min-width="150" align="center">
<template #default="scope">
<TeacherNameNo :name="scope.row.realName" :no="scope.row.teacherNo" />
</template>
</el-table-column>
<el-table-column prop="certificateConfId" label="关联资格证书" min-width="150" align="center" show-overflow-tooltip>
<template #default="scope">
{{ getCertificateName(scope.row.certificateConfId) }}
</template>
</el-table-column>
<el-table-column prop="certificateNumber" label="证书编号" min-width="150" align="center" show-overflow-tooltip />
<el-table-column prop="createTime" label="创建时间" width="180" align="center" />
<el-table-column label="证明材料" width="120" align="center">
<template #default="scope">
<el-button
v-if="scope.row.evidenceA && scope.row.evidenceA !== ''"
type="primary"
link
icon="Document"
@click="handlePreview(scope.row.srcList)">查看
</el-button>
<span v-else>-</span>
</template>
</el-table-column>
<el-table-column prop="backReason" label="驳回理由" min-width="150" align="center" show-overflow-tooltip />
<el-table-column label="操作" width="280" align="center" fixed="right">
<template #default="scope">
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_edit') && (scope.row.state === '0' || scope.row.state === '-2')"
type="primary"
link
icon="edit-pen"
@click="handleEdit(scope.row)">编辑
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_exam') && scope.row.canExam"
type="success"
link
icon="CircleCheck"
@click="changeState(scope.row, 1)">通过
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_exam') && scope.row.canDeptExam"
type="success"
link
icon="CircleCheck"
@click="changeState(scope.row, 1)">部门通过
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_exam') && scope.row.canBack"
type="danger"
link
icon="CircleClose"
@click="changeState(scope.row, -2)">驳回
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_exam') && scope.row.canDeptBack"
type="danger"
link
icon="CircleClose"
@click="changeState(scope.row, -2)">部门驳回
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_del')"
type="danger"
link
icon="delete"
style="margin-left: 12px"
@click="handleDel(scope.row)">删除
</el-button>
</template>
</el-table-column>
</el-table>
<!-- 表格 -->
<el-table
ref="tableRef"
:data="state.dataList"
row-key="id"
v-loading="state.loading"
border
:cell-style="tableStyle.cellStyle"
:header-cell-style="tableStyle.headerCellStyle"
>
<el-table-column type="index" label="序号" width="60" align="center" />
<!-- 分页 -->
<pagination
v-bind="state.pagination"
@current-change="currentChangeHandle"
@size-change="sizeChangeHandle"
/>
<el-table-column prop="state" label="审核状态" width="120" align="center">
<template #default="scope">
<AuditState :state="scope.row.state" :options="auditStateOptions" />
</template>
</el-table-column>
<!-- 材料预览图片直接显示PDF 在组件内部 dialog 中显示 -->
<preview-file
v-for="src in imgUrl"
:key="src.title"
:authSrc="src.url"
dialog-title="教师资格材料"
/>
<el-table-column label="姓名/工号" min-width="150" align="center">
<template #default="scope">
<TeacherNameNo :name="scope.row.realName" :no="scope.row.teacherNo" />
</template>
</el-table-column>
<!-- 子组件 -->
<DataForm ref="dataFormRef" @refreshData="handleFilter" />
<ProfessionalBackResaon ref="backReasonRef" @refreshData="handleFilter" />
<import-teacher-other-info ref="importTeacherOtherInfoRef" @refreshData="handleFilter" />
</div>
</div>
<el-table-column prop="certificateConfId" label="关联资格证书" min-width="150" align="center" show-overflow-tooltip>
<template #default="scope">
{{ getCertificateName(scope.row.certificateConfId) }}
</template>
</el-table-column>
<el-table-column prop="certificateNumber" label="证书编号" min-width="150" align="center" show-overflow-tooltip />
<el-table-column prop="createTime" label="创建时间" width="180" align="center" />
<el-table-column label="证明材料" width="120" align="center">
<template #default="scope">
<el-button
v-if="scope.row.evidenceA && scope.row.evidenceA !== ''"
type="primary"
link
icon="Document"
@click="handlePreview(scope.row.srcList)"
>查看
</el-button>
<span v-else>-</span>
</template>
</el-table-column>
<el-table-column prop="backReason" label="驳回理由" min-width="150" align="center" show-overflow-tooltip />
<el-table-column label="操作" width="280" align="center" fixed="right">
<template #default="scope">
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_edit') && (scope.row.state === '0' || scope.row.state === '-2')"
type="primary"
link
icon="edit-pen"
@click="handleEdit(scope.row)"
>编辑
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_exam') && scope.row.canExam"
type="success"
link
icon="CircleCheck"
@click="changeState(scope.row, 1)"
>通过
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_exam') && scope.row.canDeptExam"
type="success"
link
icon="CircleCheck"
@click="changeState(scope.row, 1)"
>部门通过
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_exam') && scope.row.canBack"
type="danger"
link
icon="CircleClose"
@click="changeState(scope.row, -2)"
>驳回
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_exam') && scope.row.canDeptBack"
type="danger"
link
icon="CircleClose"
@click="changeState(scope.row, -2)"
>部门驳回
</el-button>
<el-button
v-if="hasAuth('professional_professionalteachercertificaterelation_del')"
type="danger"
link
icon="delete"
style="margin-left: 12px"
@click="handleDel(scope.row)"
>删除
</el-button>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<pagination v-bind="state.pagination" @current-change="currentChangeHandle" @size-change="sizeChangeHandle" />
<!-- 材料预览图片直接显示PDF 在组件内部 dialog 中显示 -->
<preview-file v-for="src in imgUrl" :key="src.title" :authSrc="src.url" dialog-title="教师资格材料" />
<!-- 子组件 -->
<DataForm ref="dataFormRef" @refreshData="handleFilter" />
<ProfessionalBackResaon ref="backReasonRef" @refreshData="handleFilter" />
<import-teacher-other-info ref="importTeacherOtherInfoRef" @refreshData="handleFilter" />
</div>
</div>
</template>
<script setup lang="ts">
import { ref, reactive, onMounted, nextTick } from 'vue'
import { BasicTableProps, useTable } from '/@/hooks/table'
import { useAuth } from '/@/hooks/auth'
import { useMessage } from '/@/hooks/message'
import { useMessageBox } from '/@/hooks/message'
import { useDict } from '/@/hooks/dict'
import { getTeacherCertificateList } from '/@/api/professional/rsbase/professionalteachercertificateconf'
import {
fetchList,
examObj,
delObj
} from '/@/api/professional/professionaluser/professionalteachercertificaterelation'
import { ref, reactive, onMounted, nextTick } from 'vue';
import { BasicTableProps, useTable } from '/@/hooks/table';
import { useAuth } from '/@/hooks/auth';
import { useMessage } from '/@/hooks/message';
import { useMessageBox } from '/@/hooks/message';
import { useDict } from '/@/hooks/dict';
import { getTeacherCertificateList } from '/@/api/professional/rsbase/professionalteachercertificateconf';
import { fetchList, examObj, delObj } from '/@/api/professional/professionaluser/professionalteachercertificaterelation';
import { makeExportTeacherInfoByTypeTask } from '/@/api/professional/professionalfile';
import { PROFESSIONAL_AUDIT_STATE_OPTIONS } from '/@/config/global'
import { defineAsyncComponent } from 'vue'
const TeacherNameNo = defineAsyncComponent(() => import('/@/components/TeacherNameNo/index.vue'))
const AuditState = defineAsyncComponent(() => import('/@/components/AuditState/index.vue'))
const DataForm = defineAsyncComponent(() => import('./form.vue'))
const ProfessionalBackResaon = defineAsyncComponent(() => import('/@/views/professional/common/professional-back-resaon.vue'))
const previewFile = defineAsyncComponent(() => import('/@/components/tools/preview-file.vue'))
const ImportTeacherOtherInfo = defineAsyncComponent(() => import('/@/views/professional/common/import-teacher-other-info.vue'))
import { PROFESSIONAL_AUDIT_STATE_OPTIONS } from '/@/config/global';
import { defineAsyncComponent } from 'vue';
const TeacherNameNo = defineAsyncComponent(() => import('/@/components/TeacherNameNo/index.vue'));
const AuditState = defineAsyncComponent(() => import('/@/components/AuditState/index.vue'));
const DataForm = defineAsyncComponent(() => import('./form.vue'));
const ProfessionalBackResaon = defineAsyncComponent(() => import('/@/views/professional/common/professional-back-resaon.vue'));
const previewFile = defineAsyncComponent(() => import('/@/components/tools/preview-file.vue'));
const ImportTeacherOtherInfo = defineAsyncComponent(() => import('/@/views/professional/common/import-teacher-other-info.vue'));
// 审核状态选项(独立定义,防止其他页面修改时被波及)
import type { StateOption } from '/@/components/AuditState/index.vue'
const auditStateOptions: StateOption[] = PROFESSIONAL_AUDIT_STATE_OPTIONS
import type { StateOption } from '/@/components/AuditState/index.vue';
const auditStateOptions: StateOption[] = PROFESSIONAL_AUDIT_STATE_OPTIONS;
// 无权限即无节点
const { hasAuth } = useAuth()
const { hasAuth } = useAuth();
// 消息提示 hooks
const message = useMessage()
const messageBox = useMessageBox()
const message = useMessage();
const messageBox = useMessageBox();
// 字典数据
const { professional_state: professionalState } = useDict('professional_state')
const { professional_state: professionalState } = useDict('professional_state');
// 表格引用
const tableRef = ref()
const searchFormRef = ref()
const dataFormRef = ref()
const backReasonRef = ref()
const showSearch = ref(true)
const tableRef = ref();
const searchFormRef = ref();
const dataFormRef = ref();
const backReasonRef = ref();
const showSearch = ref(true);
// 搜索表单数据
const search = reactive({
state: '',
teacherNo: '',
realName: ''
})
state: '',
teacherNo: '',
realName: '',
});
// 材料预览
const imgUrl = ref<Array<{ title: string; url: string }>>([])
const imgUrl = ref<Array<{ title: string; url: string }>>([]);
// 导出加载状态
const exportLoading = ref(false)
const importTeacherOtherInfoRef = ref()
const exportLoading = ref(false);
const importTeacherOtherInfoRef = ref();
// 证书列表
const certificateList = ref<any[]>([])
const certificateList = ref<any[]>([]);
// 配置 useTable
const state: BasicTableProps = reactive<BasicTableProps>({
pageList: async (params: any) => {
const response = await fetchList(params)
const records = response.data.records || []
// 处理证明材料列表
records.forEach((v: any) => {
v.srcList = []
if (v.evidenceA != null) {
v.srcList.push(v.evidenceA)
}
})
return {
data: {
records,
total: response.data.total || 0
}
}
},
queryForm: search
})
pageList: async (params: any) => {
const response = await fetchList(params);
const records = response.data.records || [];
// 处理证明材料列表
records.forEach((v: any) => {
v.srcList = [];
if (v.evidenceA != null) {
v.srcList.push(v.evidenceA);
}
});
return {
data: {
records,
total: response.data.total || 0,
},
};
},
queryForm: search,
});
const { getDataList, currentChangeHandle, sizeChangeHandle, tableStyle } = useTable(state)
const { getDataList, currentChangeHandle, sizeChangeHandle, tableStyle } = useTable(state);
// 预览材料
const handlePreview = (list: string[]) => {
imgUrl.value = []
nextTick(() => {
list.forEach(v => {
imgUrl.value.push({
title: '',
url: v
})
})
})
}
imgUrl.value = [];
nextTick(() => {
list.forEach((v) => {
imgUrl.value.push({
title: '',
url: v,
});
});
});
};
// 审核状态变更
const changeState = (row: any, val: number) => {
if (val === 1) {
// 通过
const str = '通过'
messageBox.confirm(`是否确认${str}${row.realName}的申请`).then(async () => {
try {
await examObj({
id: row.id,
state: val
})
message.success('操作成功')
getDataList()
} catch (error: any) {
// 处理业务错误
message.error(error.msg)
}
}).catch(() => {})
} else if (val === -2) {
// 驳回
backReasonRef.value?.init(row, 'cer')
}
}
if (val === 1) {
// 通过
const str = '通过';
messageBox
.confirm(`是否确认${str}${row.realName}的申请`)
.then(async () => {
try {
await examObj({
id: row.id,
state: val,
});
message.success('操作成功');
getDataList();
} catch (error: any) {
// 处理业务错误
message.error(error.msg);
}
})
.catch(() => {});
} else if (val === -2) {
// 驳回
backReasonRef.value?.init(row, 'cer');
}
};
// 查询
const handleFilter = () => {
getDataList()
}
getDataList();
};
// 重置查询
const resetQuery = () => {
Object.assign(search, {
state: '',
teacherNo: '',
realName: ''
})
getDataList()
}
Object.assign(search, {
state: '',
teacherNo: '',
realName: '',
});
getDataList();
};
// 打开新增窗口
const handleAdd = () => {
dataFormRef.value?.openDialog()
}
dataFormRef.value?.openDialog();
};
// 打开编辑窗口
const handleEdit = (row: any) => {
dataFormRef.value?.openDialog(row)
}
dataFormRef.value?.openDialog(row);
};
// 删除
const handleDel = (row: any) => {
messageBox.confirm('是否确认删除该条记录').then(async () => {
try {
await delObj(row.id)
message.success('删除成功')
// 如果当前页只剩一条数据,且不是第一页,则跳转到上一页
if (state.pagination && state.dataList && state.dataList.length === 1 && state.pagination.current && state.pagination.current > 1) {
state.pagination.current = state.pagination.current - 1
}
getDataList()
} catch (error: any) {
// 处理业务错误
message.error(error.msg)
}
}).catch(() => {})
}
messageBox
.confirm('是否确认删除该条记录')
.then(async () => {
try {
await delObj(row.id);
message.success('删除成功');
// 如果当前页只剩一条数据,且不是第一页,则跳转到上一页
if (state.pagination && state.dataList && state.dataList.length === 1 && state.pagination.current && state.pagination.current > 1) {
state.pagination.current = state.pagination.current - 1;
}
getDataList();
} catch (error: any) {
// 处理业务错误
message.error(error.msg);
}
})
.catch(() => {});
};
// 导出
const handleDownLoadWord = async () => {
exportLoading.value = true;
let params = Object.assign(search, { type: 'P20004' });
makeExportTeacherInfoByTypeTask(params).then((res: any) => {
message.success('后台下载进行中,请稍后查看任务列表');
});
setTimeout(() => {
exportLoading.value = false;
}, 3000); // 5分钟后自动关闭
exportLoading.value = true;
let params = Object.assign(search, { type: 'P20004' });
makeExportTeacherInfoByTypeTask(params).then((res: any) => {
message.success('后台下载进行中,请稍后查看任务列表');
});
setTimeout(() => {
exportLoading.value = false;
}, 3000); // 5分钟后自动关闭
};
// 获取证书名称
const getCertificateName = (id: string | number) => {
const item = certificateList.value.find((item: any) => item.id === id)
return item ? item.cretificateName : '-'
}
const item = certificateList.value.find((item: any) => item.id === id);
return item ? item.cretificateName : '-';
};
// 打开导入弹窗
const handleImportDialog = () => {
importTeacherOtherInfoRef.value?.init('cerRelation')
}
importTeacherOtherInfoRef.value?.init('cerRelation');
};
// 加载字典数据
const loadDictData = async () => {
try {
// 加载证书列表
const certRes = await getTeacherCertificateList()
if (certRes && certRes.data) {
certificateList.value = certRes.data
}
} catch (error) {
// Failed to load dict data
}
}
try {
// 加载证书列表
const certRes = await getTeacherCertificateList();
if (certRes && certRes.data) {
certificateList.value = certRes.data;
}
} catch (error) {
// Failed to load dict data
}
};
// 初始化:仅加载下拉字典,表格数据由 useTablecreatedIsNeed 默认 true自动请求
onMounted(async () => {
await loadDictData()
})
await loadDictData();
});
</script>
<style lang="scss" scoped>
</style>
<style lang="scss" scoped></style>