修改打包报错问题
This commit is contained in:
624
src/views/stuwork/stuturnover/form.vue
Normal file
624
src/views/stuwork/stuturnover/form.vue
Normal file
@@ -0,0 +1,624 @@
|
||||
<template>
|
||||
<el-dialog
|
||||
:title="form.id ? '编辑' : '批量新增异动'"
|
||||
v-model="visible"
|
||||
:close-on-click-modal="false"
|
||||
draggable
|
||||
width="900px">
|
||||
<el-form
|
||||
ref="dataFormRef"
|
||||
:model="form"
|
||||
:rules="dataRules"
|
||||
label-width="120px"
|
||||
v-loading="loading">
|
||||
<el-row :gutter="24">
|
||||
<el-col :span="12" class="mb20">
|
||||
<el-form-item label="学年" prop="schoolYear">
|
||||
<el-select
|
||||
v-model="form.schoolYear"
|
||||
placeholder="请选择学年"
|
||||
clearable
|
||||
filterable
|
||||
style="width: 100%">
|
||||
<el-option
|
||||
v-for="item in schoolYearList"
|
||||
:key="item.year"
|
||||
:label="item.year"
|
||||
:value="item.year">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
|
||||
<el-col :span="12" class="mb20">
|
||||
<el-form-item label="学期" prop="schoolTerm">
|
||||
<el-select
|
||||
v-model="form.schoolTerm"
|
||||
placeholder="请选择学期"
|
||||
clearable
|
||||
style="width: 100%">
|
||||
<el-option
|
||||
v-for="item in schoolTermList"
|
||||
:key="item.value"
|
||||
:label="item.label"
|
||||
:value="item.value">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
|
||||
<el-col :span="12" class="mb20">
|
||||
<el-form-item label="原班级" prop="oldClassCode">
|
||||
<el-select
|
||||
v-model="form.oldClassCode"
|
||||
placeholder="请选择原班级"
|
||||
clearable
|
||||
filterable
|
||||
style="width: 100%"
|
||||
@change="handleOldClassChange">
|
||||
<el-option
|
||||
v-for="item in classList"
|
||||
:key="item.classCode"
|
||||
:label="item.classNo"
|
||||
:value="item.classCode">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
|
||||
<el-col :span="12" class="mb20">
|
||||
<el-form-item label="现班级" prop="newClassCode">
|
||||
<el-select
|
||||
v-model="form.newClassCode"
|
||||
placeholder="请选择现班级"
|
||||
clearable
|
||||
filterable
|
||||
style="width: 100%">
|
||||
<el-option
|
||||
v-for="item in classList"
|
||||
:key="item.classCode"
|
||||
:label="item.classNo"
|
||||
:value="item.classCode">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
|
||||
<el-col :span="12" class="mb20">
|
||||
<el-form-item label="异动类型" prop="turnoverType">
|
||||
<el-select
|
||||
v-model="form.turnoverType"
|
||||
placeholder="请选择异动类型"
|
||||
clearable
|
||||
style="width: 100%">
|
||||
<el-option
|
||||
v-for="item in turnoverTypeList"
|
||||
:key="item.value"
|
||||
:label="item.label"
|
||||
:value="item.value">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
|
||||
<el-col :span="12" class="mb20">
|
||||
<el-form-item label="转制类型" prop="turnYear">
|
||||
<el-select
|
||||
v-model="form.turnYear"
|
||||
placeholder="请选择转制类型"
|
||||
clearable
|
||||
style="width: 100%">
|
||||
<el-option
|
||||
v-for="item in turnYearList"
|
||||
:key="item.value"
|
||||
:label="item.label"
|
||||
:value="item.value">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
|
||||
<el-col :span="12" class="mb20">
|
||||
<el-form-item label="异动时间" prop="turnoverDate">
|
||||
<el-date-picker
|
||||
v-model="form.turnoverDate"
|
||||
type="datetime"
|
||||
placeholder="选择异动时间"
|
||||
format="YYYY-MM-DD HH:mm:ss"
|
||||
value-format="YYYY-MM-DD HH:mm:ss"
|
||||
style="width: 100%" />
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
|
||||
<el-col :span="12" class="mb20">
|
||||
<el-form-item label="选择学生" prop="selectedStudents">
|
||||
<el-input
|
||||
v-model="selectedStudentsText"
|
||||
placeholder="请选择学生"
|
||||
readonly
|
||||
style="width: 100%"
|
||||
@click="openStudentDialog">
|
||||
<template #suffix>
|
||||
<el-button
|
||||
icon="Search"
|
||||
circle
|
||||
@click.stop="openStudentDialog" />
|
||||
</template>
|
||||
</el-input>
|
||||
<span v-if="form.selectedStudents.length > 0" class="ml5" style="color: #409eff;">
|
||||
已选 {{ form.selectedStudents.length }} 人
|
||||
</span>
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
|
||||
<el-col :span="24" class="mb20">
|
||||
<el-form-item label="异动原因" prop="remarks">
|
||||
<el-input
|
||||
v-model="form.remarks"
|
||||
type="textarea"
|
||||
:rows="4"
|
||||
placeholder="请输入异动原因" />
|
||||
</el-form-item>
|
||||
</el-col>
|
||||
</el-row>
|
||||
</el-form>
|
||||
<template #footer>
|
||||
<span class="dialog-footer">
|
||||
<el-button @click="visible = false">取 消</el-button>
|
||||
<el-button type="primary" @click="onSubmit" :disabled="loading">确 认</el-button>
|
||||
</span>
|
||||
</template>
|
||||
</el-dialog>
|
||||
|
||||
<!-- 学生选择弹窗 -->
|
||||
<el-dialog
|
||||
title="选择学生"
|
||||
v-model="studentDialogVisible"
|
||||
:close-on-click-modal="false"
|
||||
draggable
|
||||
width="900px">
|
||||
<!-- 搜索表单 -->
|
||||
<el-form :model="studentSearchForm" :inline="true" @keyup.enter="handleStudentSearch">
|
||||
<el-form-item label="班号" prop="classNo">
|
||||
<el-input
|
||||
v-model="studentSearchForm.classNo"
|
||||
placeholder="请输入班号"
|
||||
clearable
|
||||
style="width: 200px" />
|
||||
</el-form-item>
|
||||
<el-form-item label="学号" prop="stuNo">
|
||||
<el-input
|
||||
v-model="studentSearchForm.stuNo"
|
||||
placeholder="请输入学号"
|
||||
clearable
|
||||
style="width: 200px" />
|
||||
</el-form-item>
|
||||
<el-form-item label="姓名" prop="realName">
|
||||
<el-input
|
||||
v-model="studentSearchForm.realName"
|
||||
placeholder="请输入姓名"
|
||||
clearable
|
||||
style="width: 200px" />
|
||||
</el-form-item>
|
||||
<el-form-item>
|
||||
<el-button type="primary" plain icon="Search" @click="handleStudentSearch">查询</el-button>
|
||||
<el-button icon="Refresh" @click="handleStudentReset">重置</el-button>
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
|
||||
<!-- 学生表格 -->
|
||||
<el-table
|
||||
ref="studentTableRef"
|
||||
:data="studentTableData"
|
||||
v-loading="studentTableLoading"
|
||||
border
|
||||
@selection-change="handleStudentSelectionChange"
|
||||
style="margin-top: 20px; max-height: 400px; overflow-y: auto;">
|
||||
<el-table-column type="selection" width="55" align="center" />
|
||||
<el-table-column type="index" label="序号" width="60" align="center" />
|
||||
<el-table-column prop="classNo" label="班号" show-overflow-tooltip align="center" />
|
||||
<el-table-column prop="stuNo" label="学号" show-overflow-tooltip align="center" />
|
||||
<el-table-column prop="realName" label="姓名" show-overflow-tooltip align="center" />
|
||||
</el-table>
|
||||
|
||||
<!-- 分页 -->
|
||||
<pagination
|
||||
@size-change="studentSizeChangeHandle"
|
||||
@current-change="studentCurrentChangeHandle"
|
||||
v-bind="studentPagination"
|
||||
style="margin-top: 20px" />
|
||||
|
||||
<template #footer>
|
||||
<span class="dialog-footer">
|
||||
<el-button @click="studentDialogVisible = false">取 消</el-button>
|
||||
<el-button type="primary" @click="confirmStudentSelection">确 认</el-button>
|
||||
</span>
|
||||
</template>
|
||||
</el-dialog>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts" name="StuTurnoverFormDialog">
|
||||
import { ref, reactive, nextTick, onMounted, computed } from 'vue'
|
||||
import { useMessage } from '/@/hooks/message'
|
||||
import { addObj, editObj, getDetail } from '/@/api/stuwork/stuturnover'
|
||||
import { queryAllSchoolYear } from '/@/api/basic/basicyear'
|
||||
import { getDicts } from '/@/api/admin/dict'
|
||||
import { list as getClassList } from '/@/api/basic/basicclass'
|
||||
import { fetchList as getStudentList } from '/@/api/basic/basicstudent'
|
||||
|
||||
const emit = defineEmits(['refresh'])
|
||||
|
||||
// 定义变量内容
|
||||
const dataFormRef = ref()
|
||||
const studentTableRef = ref()
|
||||
const visible = ref(false)
|
||||
const loading = ref(false)
|
||||
const operType = ref('add') // add 或 edit
|
||||
const schoolYearList = ref<any[]>([])
|
||||
const schoolTermList = ref<any[]>([])
|
||||
const classList = ref<any[]>([])
|
||||
const turnoverTypeList = ref<any[]>([])
|
||||
const turnYearList = ref<any[]>([])
|
||||
|
||||
// 学生选择弹窗相关
|
||||
const studentDialogVisible = ref(false)
|
||||
const studentTableLoading = ref(false)
|
||||
const studentTableData = ref<any[]>([])
|
||||
const selectedStudentRows = ref<any[]>([]) // 表格中选中的行
|
||||
const studentSearchForm = reactive({
|
||||
classCode: '',
|
||||
classNo: '',
|
||||
stuNo: '',
|
||||
realName: ''
|
||||
})
|
||||
const studentPagination = reactive({
|
||||
total: 0,
|
||||
pageSize: 10,
|
||||
currentPage: 1
|
||||
})
|
||||
|
||||
// 提交表单数据
|
||||
const form = reactive({
|
||||
id: '',
|
||||
schoolYear: '',
|
||||
schoolTerm: '',
|
||||
oldClassCode: '',
|
||||
newClassCode: '',
|
||||
turnoverType: '',
|
||||
turnYear: '',
|
||||
turnoverDate: '',
|
||||
remarks: '',
|
||||
selectedStudents: [] as any[] // 选中的学生对象数组
|
||||
})
|
||||
|
||||
// 已选学生文本显示
|
||||
const selectedStudentsText = computed(() => {
|
||||
if (!form.selectedStudents || form.selectedStudents.length === 0) {
|
||||
return ''
|
||||
}
|
||||
return `已选择 ${form.selectedStudents.length} 名学生`
|
||||
})
|
||||
|
||||
// 定义校验规则
|
||||
const dataRules = {
|
||||
schoolYear: [
|
||||
{ required: true, message: '请选择学年', trigger: 'change' }
|
||||
],
|
||||
schoolTerm: [
|
||||
{ required: true, message: '请选择学期', trigger: 'change' }
|
||||
],
|
||||
oldClassCode: [
|
||||
{ required: true, message: '请选择原班级', trigger: 'change' }
|
||||
],
|
||||
newClassCode: [
|
||||
{ required: true, message: '请选择现班级', trigger: 'change' }
|
||||
],
|
||||
turnoverType: [
|
||||
{ required: true, message: '请选择异动类型', trigger: 'change' }
|
||||
],
|
||||
turnoverDate: [
|
||||
{ required: true, message: '请选择异动时间', trigger: 'change' }
|
||||
],
|
||||
selectedStudents: [
|
||||
{ required: true, message: '请至少选择一个学生', trigger: 'change', type: 'array', min: 1 }
|
||||
]
|
||||
}
|
||||
|
||||
// 原班级变化时,更新学生搜索条件
|
||||
const handleOldClassChange = () => {
|
||||
studentSearchForm.classCode = form.oldClassCode
|
||||
// 清空已选学生
|
||||
form.selectedStudents = []
|
||||
}
|
||||
|
||||
// 打开学生选择弹窗
|
||||
const openStudentDialog = () => {
|
||||
if (!form.oldClassCode) {
|
||||
useMessage().warning('请先选择原班级')
|
||||
return
|
||||
}
|
||||
studentDialogVisible.value = true
|
||||
studentSearchForm.classCode = form.oldClassCode
|
||||
studentSearchForm.classNo = ''
|
||||
studentSearchForm.stuNo = ''
|
||||
studentSearchForm.realName = ''
|
||||
selectedStudentRows.value = []
|
||||
studentPagination.currentPage = 1
|
||||
handleStudentSearch()
|
||||
}
|
||||
|
||||
// 学生搜索
|
||||
const handleStudentSearch = async () => {
|
||||
studentTableLoading.value = true
|
||||
try {
|
||||
const params: any = {
|
||||
current: studentPagination.currentPage,
|
||||
size: studentPagination.pageSize,
|
||||
classCode: studentSearchForm.classCode || undefined,
|
||||
stuNo: studentSearchForm.stuNo || undefined,
|
||||
realName: studentSearchForm.realName || undefined
|
||||
}
|
||||
const res = await getStudentList(params)
|
||||
if (res.data && res.data.records) {
|
||||
studentTableData.value = res.data.records
|
||||
studentPagination.total = res.data.total || 0
|
||||
} else {
|
||||
studentTableData.value = []
|
||||
studentPagination.total = 0
|
||||
}
|
||||
|
||||
// 恢复之前选中的学生
|
||||
nextTick(() => {
|
||||
if (form.selectedStudents.length > 0) {
|
||||
studentTableData.value.forEach((row: any) => {
|
||||
const isSelected = form.selectedStudents.some((s: any) => s.stuNo === row.stuNo)
|
||||
if (isSelected) {
|
||||
studentTableRef.value?.toggleRowSelection(row, true)
|
||||
}
|
||||
})
|
||||
}
|
||||
})
|
||||
} catch (err) {
|
||||
console.error('获取学生列表失败', err)
|
||||
studentTableData.value = []
|
||||
} finally {
|
||||
studentTableLoading.value = false
|
||||
}
|
||||
}
|
||||
|
||||
// 学生搜索重置
|
||||
const handleStudentReset = () => {
|
||||
studentSearchForm.classNo = ''
|
||||
studentSearchForm.stuNo = ''
|
||||
studentSearchForm.realName = ''
|
||||
handleStudentSearch()
|
||||
}
|
||||
|
||||
// 学生表格选择变化
|
||||
const handleStudentSelectionChange = (selection: any[]) => {
|
||||
selectedStudentRows.value = selection
|
||||
}
|
||||
|
||||
// 确认学生选择
|
||||
const confirmStudentSelection = () => {
|
||||
if (selectedStudentRows.value.length === 0) {
|
||||
useMessage().warning('请至少选择一个学生')
|
||||
return
|
||||
}
|
||||
|
||||
// 合并已选学生(避免重复)
|
||||
const newStudents = selectedStudentRows.value.map((row: any) => ({
|
||||
stuNo: row.stuNo,
|
||||
realName: row.realName
|
||||
}))
|
||||
|
||||
// 合并到已选列表,去重
|
||||
const existingStuNos = form.selectedStudents.map((s: any) => s.stuNo)
|
||||
newStudents.forEach((student: any) => {
|
||||
if (!existingStuNos.includes(student.stuNo)) {
|
||||
form.selectedStudents.push(student)
|
||||
}
|
||||
})
|
||||
|
||||
studentDialogVisible.value = false
|
||||
useMessage().success(`已选择 ${form.selectedStudents.length} 名学生`)
|
||||
}
|
||||
|
||||
// 学生分页变化
|
||||
const studentSizeChangeHandle = (size: number) => {
|
||||
studentPagination.pageSize = size
|
||||
studentPagination.currentPage = 1
|
||||
handleStudentSearch()
|
||||
}
|
||||
|
||||
const studentCurrentChangeHandle = (current: number) => {
|
||||
studentPagination.currentPage = current
|
||||
handleStudentSearch()
|
||||
}
|
||||
|
||||
// 打开弹窗
|
||||
const openDialog = async (type: string = 'add', row?: any) => {
|
||||
visible.value = true
|
||||
operType.value = type
|
||||
|
||||
// 重置表单数据
|
||||
nextTick(() => {
|
||||
dataFormRef.value?.resetFields()
|
||||
form.id = ''
|
||||
form.schoolYear = ''
|
||||
form.schoolTerm = ''
|
||||
form.oldClassCode = ''
|
||||
form.newClassCode = ''
|
||||
form.turnoverType = ''
|
||||
form.turnYear = ''
|
||||
form.turnoverDate = ''
|
||||
form.remarks = ''
|
||||
form.selectedStudents = []
|
||||
|
||||
// 编辑时填充数据
|
||||
if (type === 'edit' && row) {
|
||||
form.id = row.id
|
||||
form.schoolYear = row.schoolYear || ''
|
||||
form.schoolTerm = row.schoolTerm || ''
|
||||
form.oldClassCode = row.oldClassCode || ''
|
||||
form.newClassCode = row.newClassCode || ''
|
||||
form.turnoverType = row.turnoverType || ''
|
||||
form.turnYear = row.turnYear || ''
|
||||
form.turnoverDate = row.turnoverDate || ''
|
||||
form.remarks = row.remarks || ''
|
||||
|
||||
// 编辑时,学生列表只有当前学生
|
||||
if (row.stuNo && row.realName) {
|
||||
form.selectedStudents = [{
|
||||
stuNo: row.stuNo,
|
||||
realName: row.realName
|
||||
}]
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
// 提交表单
|
||||
const onSubmit = async () => {
|
||||
if (!dataFormRef.value) return
|
||||
|
||||
await dataFormRef.value.validate(async (valid: boolean) => {
|
||||
if (!valid) return
|
||||
|
||||
if (!form.selectedStudents || form.selectedStudents.length === 0) {
|
||||
useMessage().error('请至少选择一个学生')
|
||||
return
|
||||
}
|
||||
|
||||
loading.value = true
|
||||
try {
|
||||
const submitData: any = {
|
||||
schoolYear: form.schoolYear,
|
||||
schoolTerm: form.schoolTerm,
|
||||
oldClassCode: form.oldClassCode,
|
||||
newClassCode: form.newClassCode,
|
||||
turnoverType: form.turnoverType,
|
||||
turnYear: form.turnYear || '',
|
||||
turnoverDate: form.turnoverDate,
|
||||
remarks: form.remarks || '',
|
||||
stuList: form.selectedStudents.map((s: any) => ({
|
||||
stuNo: s.stuNo,
|
||||
realName: s.realName
|
||||
}))
|
||||
}
|
||||
|
||||
// 编辑时需要包含id和单个学生信息
|
||||
if (operType.value === 'edit') {
|
||||
submitData.id = form.id
|
||||
submitData.stuNo = form.selectedStudents[0]?.stuNo
|
||||
submitData.realName = form.selectedStudents[0]?.realName
|
||||
// 编辑时不需要stuList
|
||||
delete submitData.stuList
|
||||
}
|
||||
|
||||
if (operType.value === 'add') {
|
||||
await addObj(submitData)
|
||||
useMessage().success('批量新增成功')
|
||||
} else {
|
||||
await editObj(submitData)
|
||||
useMessage().success('编辑成功')
|
||||
}
|
||||
visible.value = false
|
||||
emit('refresh')
|
||||
} catch (err: any) {
|
||||
useMessage().error(err.msg || (operType.value === 'add' ? '批量新增失败' : '编辑失败'))
|
||||
} finally {
|
||||
loading.value = false
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
// 获取学年列表
|
||||
const getSchoolYearList = async () => {
|
||||
try {
|
||||
const res = await queryAllSchoolYear()
|
||||
if (res.data) {
|
||||
schoolYearList.value = Array.isArray(res.data) ? res.data : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取学年列表失败', err)
|
||||
schoolYearList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取学期字典
|
||||
const getSchoolTermDict = async () => {
|
||||
try {
|
||||
const res = await getDicts('school_term')
|
||||
if (res.data) {
|
||||
schoolTermList.value = Array.isArray(res.data) ? res.data.map((item: any) => ({
|
||||
label: item.label || item.dictLabel || item.name,
|
||||
value: item.value || item.dictValue || item.code
|
||||
})) : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取学期字典失败', err)
|
||||
schoolTermList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取班级列表
|
||||
const getClassListData = async () => {
|
||||
try {
|
||||
const res = await getClassList()
|
||||
if (res.data) {
|
||||
classList.value = Array.isArray(res.data) ? res.data : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取班级列表失败', err)
|
||||
classList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取异动类型字典
|
||||
const getTurnoverTypeDict = async () => {
|
||||
try {
|
||||
const res = await getDicts('turnover_type')
|
||||
if (res.data) {
|
||||
turnoverTypeList.value = Array.isArray(res.data) ? res.data.map((item: any) => ({
|
||||
label: item.label || item.dictLabel || item.name,
|
||||
value: item.value || item.dictValue || item.code
|
||||
})) : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取异动类型字典失败', err)
|
||||
turnoverTypeList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取转制类型字典
|
||||
const getTurnYearDict = async () => {
|
||||
try {
|
||||
const res = await getDicts('turn_year_type')
|
||||
if (res.data) {
|
||||
turnYearList.value = Array.isArray(res.data) ? res.data.map((item: any) => ({
|
||||
label: item.label || item.dictLabel || item.name,
|
||||
value: item.value || item.dictValue || item.code
|
||||
})) : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取转制类型字典失败', err)
|
||||
turnYearList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 初始化
|
||||
onMounted(() => {
|
||||
getSchoolYearList()
|
||||
getSchoolTermDict()
|
||||
getClassListData()
|
||||
getTurnoverTypeDict()
|
||||
getTurnYearDict()
|
||||
})
|
||||
|
||||
// 暴露方法给父组件
|
||||
defineExpose({
|
||||
openDialog
|
||||
})
|
||||
</script>
|
||||
|
||||
433
src/views/stuwork/stuturnover/index.vue
Normal file
433
src/views/stuwork/stuturnover/index.vue
Normal file
@@ -0,0 +1,433 @@
|
||||
<template>
|
||||
<div class="layout-padding">
|
||||
<div class="layout-padding-auto layout-padding-view">
|
||||
<!-- 搜索表单 -->
|
||||
<el-row v-show="showSearch">
|
||||
<el-form :model="searchForm" ref="searchFormRef" :inline="true" @keyup.enter="handleSearch">
|
||||
<el-form-item label="学年" prop="schoolYear">
|
||||
<el-select
|
||||
v-model="searchForm.schoolYear"
|
||||
placeholder="请选择学年"
|
||||
clearable
|
||||
filterable
|
||||
style="width: 200px">
|
||||
<el-option
|
||||
v-for="item in schoolYearList"
|
||||
:key="item.year"
|
||||
:label="item.year"
|
||||
:value="item.year">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
<el-form-item label="学期" prop="schoolTerm">
|
||||
<el-select
|
||||
v-model="searchForm.schoolTerm"
|
||||
placeholder="请选择学期"
|
||||
clearable
|
||||
style="width: 200px">
|
||||
<el-option
|
||||
v-for="item in schoolTermList"
|
||||
:key="item.value"
|
||||
:label="item.label"
|
||||
:value="item.value">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
<el-form-item label="学院" prop="deptCode">
|
||||
<el-select
|
||||
v-model="searchForm.deptCode"
|
||||
placeholder="请选择学院"
|
||||
clearable
|
||||
filterable
|
||||
style="width: 200px">
|
||||
<el-option
|
||||
v-for="item in deptList"
|
||||
:key="item.deptCode"
|
||||
:label="item.deptName"
|
||||
:value="item.deptCode">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
<el-form-item label="原班级" prop="oldClassCode">
|
||||
<el-select
|
||||
v-model="searchForm.oldClassCode"
|
||||
placeholder="请选择原班级"
|
||||
clearable
|
||||
filterable
|
||||
style="width: 200px">
|
||||
<el-option
|
||||
v-for="item in classList"
|
||||
:key="item.classCode"
|
||||
:label="item.classNo"
|
||||
:value="item.classCode">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
<el-form-item label="学号" prop="stuNo">
|
||||
<el-input
|
||||
v-model="searchForm.stuNo"
|
||||
placeholder="请输入学号"
|
||||
clearable
|
||||
style="width: 200px" />
|
||||
</el-form-item>
|
||||
<el-form-item label="姓名" prop="realName">
|
||||
<el-input
|
||||
v-model="searchForm.realName"
|
||||
placeholder="请输入姓名"
|
||||
clearable
|
||||
style="width: 200px" />
|
||||
</el-form-item>
|
||||
<el-form-item label="异动类型" prop="turnoverType">
|
||||
<el-select
|
||||
v-model="searchForm.turnoverType"
|
||||
placeholder="请选择异动类型"
|
||||
clearable
|
||||
style="width: 200px">
|
||||
<el-option
|
||||
v-for="item in turnoverTypeList"
|
||||
:key="item.value"
|
||||
:label="item.label"
|
||||
:value="item.value">
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
<el-form-item>
|
||||
<el-button type="primary" plain icon="Search" @click="handleSearch">查询</el-button>
|
||||
<el-button icon="Refresh" @click="handleReset">重置</el-button>
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
</el-row>
|
||||
|
||||
<!-- 操作按钮 -->
|
||||
<el-row>
|
||||
<div class="mb8" style="width: 100%">
|
||||
<el-button
|
||||
icon="FolderAdd"
|
||||
type="primary"
|
||||
class="ml10"
|
||||
@click="formDialogRef.openDialog()">
|
||||
批量新增异动
|
||||
</el-button>
|
||||
<el-button
|
||||
icon="Download"
|
||||
type="success"
|
||||
class="ml10"
|
||||
@click="handleExport">
|
||||
导 出
|
||||
</el-button>
|
||||
<right-toolbar
|
||||
v-model:showSearch="showSearch"
|
||||
class="ml10 mr20"
|
||||
style="float: right;"
|
||||
@queryTable="getDataList">
|
||||
</right-toolbar>
|
||||
</div>
|
||||
</el-row>
|
||||
|
||||
<!-- 表格 -->
|
||||
<el-table
|
||||
:data="state.dataList"
|
||||
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="schoolYear" label="学年" show-overflow-tooltip align="center" />
|
||||
<el-table-column prop="schoolTerm" label="学期" show-overflow-tooltip align="center">
|
||||
<template #default="scope">
|
||||
<span>{{ formatSchoolTerm(scope.row.schoolTerm) }}</span>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column prop="deptName" label="学院" show-overflow-tooltip align="center" />
|
||||
<el-table-column prop="oldClassNo" label="原班级" show-overflow-tooltip align="center" />
|
||||
<el-table-column prop="newClassNo" label="现班级" show-overflow-tooltip align="center" />
|
||||
<el-table-column prop="stuNo" label="学号" show-overflow-tooltip align="center" />
|
||||
<el-table-column prop="realName" label="姓名" show-overflow-tooltip align="center" />
|
||||
<el-table-column prop="turnoverDate" label="异动时间" show-overflow-tooltip align="center" width="180">
|
||||
<template #default="scope">
|
||||
<span>{{ scope.row.turnoverDate ? formatDate(scope.row.turnoverDate) : '-' }}</span>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column prop="turnoverType" label="异动类型" show-overflow-tooltip align="center">
|
||||
<template #default="scope">
|
||||
<span>{{ formatTurnoverType(scope.row.turnoverType) }}</span>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column prop="turnYear" label="转制类型" show-overflow-tooltip align="center">
|
||||
<template #default="scope">
|
||||
<span>{{ formatTurnYear(scope.row.turnYear) }}</span>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column prop="remarks" label="异动原因" show-overflow-tooltip align="center" min-width="150" />
|
||||
<el-table-column label="操作" width="150" align="center" fixed="right">
|
||||
<template #default="scope">
|
||||
<el-button
|
||||
icon="Edit"
|
||||
text
|
||||
type="primary"
|
||||
@click="handleEdit(scope.row)">
|
||||
编辑
|
||||
</el-button>
|
||||
<el-button
|
||||
icon="Delete"
|
||||
text
|
||||
type="danger"
|
||||
@click="handleDelete(scope.row)">
|
||||
删除
|
||||
</el-button>
|
||||
</template>
|
||||
</el-table-column>
|
||||
</el-table>
|
||||
|
||||
<!-- 分页 -->
|
||||
<pagination
|
||||
@size-change="sizeChangeHandle"
|
||||
@current-change="currentChangeHandle"
|
||||
v-bind="state.pagination" />
|
||||
</div>
|
||||
|
||||
<!-- 批量新增/编辑表单弹窗 -->
|
||||
<form-dialog ref="formDialogRef" @refresh="getDataList" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts" name="StuTurnover">
|
||||
import { reactive, ref, onMounted } from 'vue'
|
||||
import { BasicTableProps, useTable } from "/@/hooks/table";
|
||||
import { fetchList, delObj, exportData } from "/@/api/stuwork/stuturnover";
|
||||
import { queryAllSchoolYear } from "/@/api/basic/basicyear";
|
||||
import { getDicts } from "/@/api/admin/dict";
|
||||
import { getDeptListByLevelTwo } from "/@/api/basic/basicdept";
|
||||
import { list as getClassList } from "/@/api/basic/basicclass";
|
||||
import { useMessage, useMessageBox } from "/@/hooks/message";
|
||||
import FormDialog from './form.vue'
|
||||
|
||||
// 定义变量内容
|
||||
const searchFormRef = ref()
|
||||
const showSearch = ref(true)
|
||||
const schoolYearList = ref<any[]>([])
|
||||
const schoolTermList = ref<any[]>([])
|
||||
const deptList = ref<any[]>([])
|
||||
const classList = ref<any[]>([])
|
||||
const turnoverTypeList = ref<any[]>([])
|
||||
const turnYearList = ref<any[]>([])
|
||||
const formDialogRef = ref()
|
||||
|
||||
// 搜索表单
|
||||
const searchForm = reactive({
|
||||
schoolYear: '',
|
||||
schoolTerm: '',
|
||||
deptCode: '',
|
||||
oldClassCode: '',
|
||||
stuNo: '',
|
||||
realName: '',
|
||||
turnoverType: ''
|
||||
})
|
||||
|
||||
// 配置 useTable
|
||||
const state: BasicTableProps = reactive<BasicTableProps>({
|
||||
queryForm: searchForm,
|
||||
pageList: fetchList,
|
||||
props: {
|
||||
item: 'records',
|
||||
totalCount: 'total'
|
||||
},
|
||||
createdIsNeed: true
|
||||
})
|
||||
|
||||
// table hook
|
||||
const {
|
||||
getDataList,
|
||||
currentChangeHandle,
|
||||
sizeChangeHandle,
|
||||
tableStyle
|
||||
} = useTable(state)
|
||||
|
||||
// 格式化学期
|
||||
const formatSchoolTerm = (value: string | number) => {
|
||||
if (value === null || value === undefined || value === '') {
|
||||
return '-'
|
||||
}
|
||||
const dictItem = schoolTermList.value.find(item => item.value == value)
|
||||
return dictItem ? dictItem.label : value
|
||||
}
|
||||
|
||||
// 格式化日期
|
||||
const formatDate = (dateStr: string) => {
|
||||
if (!dateStr) return '-'
|
||||
// 如果包含时间部分,只显示日期部分
|
||||
if (dateStr.includes(' ')) {
|
||||
return dateStr.split(' ')[0]
|
||||
}
|
||||
return dateStr
|
||||
}
|
||||
|
||||
// 格式化异动类型
|
||||
const formatTurnoverType = (value: string) => {
|
||||
if (!value) return '-'
|
||||
const item = turnoverTypeList.value.find((item: any) => item.value === value)
|
||||
return item ? item.label : value
|
||||
}
|
||||
|
||||
// 格式化转制类型
|
||||
const formatTurnYear = (value: string) => {
|
||||
if (!value) return '-'
|
||||
const item = turnYearList.value.find((item: any) => item.value === value)
|
||||
return item ? item.label : value
|
||||
}
|
||||
|
||||
// 查询
|
||||
const handleSearch = () => {
|
||||
getDataList()
|
||||
}
|
||||
|
||||
// 重置
|
||||
const handleReset = () => {
|
||||
searchFormRef.value?.resetFields()
|
||||
searchForm.schoolYear = ''
|
||||
searchForm.schoolTerm = ''
|
||||
searchForm.deptCode = ''
|
||||
searchForm.oldClassCode = ''
|
||||
searchForm.stuNo = ''
|
||||
searchForm.realName = ''
|
||||
searchForm.turnoverType = ''
|
||||
getDataList()
|
||||
}
|
||||
|
||||
// 编辑
|
||||
const handleEdit = (row: any) => {
|
||||
formDialogRef.value?.openDialog('edit', row)
|
||||
}
|
||||
|
||||
// 删除
|
||||
const handleDelete = async (row: any) => {
|
||||
try {
|
||||
await useMessageBox().confirm('确定要删除这条记录吗?')
|
||||
await delObj([row.id])
|
||||
useMessage().success('删除成功')
|
||||
getDataList()
|
||||
} catch (err: any) {
|
||||
if (err !== 'cancel') {
|
||||
useMessage().error(err.msg || '删除失败')
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 导出
|
||||
const handleExport = async () => {
|
||||
try {
|
||||
const res = await exportData(searchForm)
|
||||
// 处理返回的文件流
|
||||
const blob = new Blob([res.data])
|
||||
const elink = document.createElement('a')
|
||||
elink.download = '学籍异动.xlsx'
|
||||
elink.style.display = 'none'
|
||||
elink.href = URL.createObjectURL(blob)
|
||||
document.body.appendChild(elink)
|
||||
elink.click()
|
||||
URL.revokeObjectURL(elink.href)
|
||||
document.body.removeChild(elink)
|
||||
useMessage().success('导出成功')
|
||||
} catch (err: any) {
|
||||
useMessage().error(err.msg || '导出失败')
|
||||
}
|
||||
}
|
||||
|
||||
// 获取学年列表
|
||||
const getSchoolYearList = async () => {
|
||||
try {
|
||||
const res = await queryAllSchoolYear()
|
||||
if (res.data) {
|
||||
schoolYearList.value = Array.isArray(res.data) ? res.data : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取学年列表失败', err)
|
||||
schoolYearList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取学期字典
|
||||
const getSchoolTermDict = async () => {
|
||||
try {
|
||||
const res = await getDicts('school_term')
|
||||
if (res.data) {
|
||||
schoolTermList.value = Array.isArray(res.data) ? res.data.map((item: any) => ({
|
||||
label: item.label || item.dictLabel || item.name,
|
||||
value: item.value || item.dictValue || item.code
|
||||
})) : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取学期字典失败', err)
|
||||
schoolTermList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取学院列表
|
||||
const getDeptListData = async () => {
|
||||
try {
|
||||
const res = await getDeptListByLevelTwo()
|
||||
if (res.data) {
|
||||
deptList.value = Array.isArray(res.data) ? res.data : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取学院列表失败', err)
|
||||
deptList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取班级列表
|
||||
const getClassListData = async () => {
|
||||
try {
|
||||
const res = await getClassList()
|
||||
if (res.data) {
|
||||
classList.value = Array.isArray(res.data) ? res.data : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取班级列表失败', err)
|
||||
classList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取异动类型字典
|
||||
const getTurnoverTypeDict = async () => {
|
||||
try {
|
||||
const res = await getDicts('turnover_type')
|
||||
if (res.data) {
|
||||
turnoverTypeList.value = Array.isArray(res.data) ? res.data.map((item: any) => ({
|
||||
label: item.label || item.dictLabel || item.name,
|
||||
value: item.value || item.dictValue || item.code
|
||||
})) : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取异动类型字典失败', err)
|
||||
turnoverTypeList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 获取转制类型字典
|
||||
const getTurnYearDict = async () => {
|
||||
try {
|
||||
const res = await getDicts('turn_year_type')
|
||||
if (res.data) {
|
||||
turnYearList.value = Array.isArray(res.data) ? res.data.map((item: any) => ({
|
||||
label: item.label || item.dictLabel || item.name,
|
||||
value: item.value || item.dictValue || item.code
|
||||
})) : []
|
||||
}
|
||||
} catch (err) {
|
||||
console.error('获取转制类型字典失败', err)
|
||||
turnYearList.value = []
|
||||
}
|
||||
}
|
||||
|
||||
// 初始化
|
||||
onMounted(() => {
|
||||
getSchoolYearList()
|
||||
getSchoolTermDict()
|
||||
getDeptListData()
|
||||
getClassListData()
|
||||
getTurnoverTypeDict()
|
||||
getTurnYearDict()
|
||||
})
|
||||
</script>
|
||||
|
||||
Reference in New Issue
Block a user