feat: 添加报名模块

This commit is contained in:
lwh 2023-04-19 19:33:55 +08:00
parent aebe97891a
commit bf58e8406a
7 changed files with 686 additions and 40 deletions

View File

@ -25,15 +25,15 @@
<!-- 导入 -->
<el-upload class="upload-demo" :action="importExcel" :headers="headers" :on-success="handleExcelSuccess"
:on-progress="uploadLoading" :on-error="closeUploadLoading" style="margin-left: 10px" :show-file-list="false">
<el-button type="primary">导入</el-button>
</el-upload>
:on-progress="uploadLoading" :on-error="closeUploadLoading" style="margin-left: 10px" :show-file-list="false">
<el-button type="primary">导入</el-button>
</el-upload>
<!-- 下载导入模板 -->
<el-button icon="ElIconDownload" @click="downloadTemplate()">下载导入模板</el-button>
<!-- 下载导入模板 -->
<el-button icon="ElIconDownload" @click="downloadTemplate()">下载导入模板</el-button>
<!-- 导出 -->
<el-button icon="ElIconDocument" @click="exportExcel(params)">导出</el-button>
<!-- 导出 -->
<el-button icon="ElIconDocument" @click="exportExcel(params)">导出</el-button>
<!-- 下拉操作 -->
<el-dropdown v-if="selectionData.length">
@ -51,7 +51,7 @@
</el-space>
<!-- 数据表格 -->
<el-table v-loading="loading" :data="dataList" ref="tableRef" border highlight-current-row
:onSelectionChange="data => (selectionData = data)">
:onSelectionChange="data => (selectionData = data)" >
<el-table-column type="selection" width="50" align="center" />
<el-table-column prop="classes_name" width="200" label="班型名称" :show-overflow-tooltip="true"> </el-table-column>
<el-table-column prop="classes_desc" width="300" label="班型简介" :show-overflow-tooltip="true"> </el-table-column>
@ -70,7 +70,8 @@
</template>
</el-table-column>
</el-table>
<pagination :total="total" v-model:page="params.page" v-model:limit="params.limit" @pagination="getList" />
<pagination :total="total" v-model:page="params.page" v-model:limit="params.limit"
@pagination="getList" />
<!-- 添加班型 -->
<AddClassesDialog v-model="addClassesDialogVisible" :done="() => getList()"></AddClassesDialog>
@ -82,7 +83,7 @@
</template>
<script setup>
import { ArrowDown } from '@element-plus/icons-vue';
import { ref, reactive, watch } from 'vue';
import { ref, reactive,watch } from 'vue';
import { useLoginStore } from "~/store";
import { getClassesList, editClasses, deleteClasses, exportExcel, downloadTemplate, importExcel } from '~/service/classes';
import AddClassesDialog from './components/AddClassesDialog.vue';
@ -160,8 +161,8 @@ watch(drag, () => {
drag.value.drop((info) => {
let targetIdx = [...document.getElementsByClassName('row1')].indexOf(info.el)
let sourceIdx = [...document.getElementsByClassName('row1')].indexOf(info.info.source.el)
let orderIdx = dataList.value[sourceIdx]['works_type_order']
dataList.value[targetIdx]['works_type_order'] = orderIdx
let orderIdx = dataList.value[sourceIdx]['classes_sort']
dataList.value[targetIdx]['classes_sort'] = orderIdx
handleEditOrder(dataList.value[targetIdx])
})
})

View File

@ -0,0 +1,148 @@
<template>
<el-dialog v-model="dialogVisible" title="添加报名流程" width="900px" @closed="closeDialog" @open="openDialog">
<el-form ref="formRef" :model="formData" :rules="rules">
<el-row>
<el-col :span="12">
<el-form-item :label-width="labelWidth" label="标题" prop="sign_up_process_title">
<el-input v-model='formData.sign_up_process_title' type="text" placeholder='请输入标题'></el-input>
</el-form-item>
</el-col>
<el-col :span='12'>
<el-form-item :label-width='labelWidth' label='图标' prop='sign_up_process_icon'>
<UploadImage ref='uploadRef' v-model='formData.sign_up_process_icon' :data=uoloadData :limit='1' :fileSize='5'
:drag='true' :isShowTip='false' />
</el-form-item>
</el-col>
<el-col :span="24">
<el-form-item :label-width="labelWidth" label="介绍" prop="sign_up_process_intro">
<el-input v-model='formData.sign_up_process_intro' type="textarea" :rows="5" placeholder='请输入介绍'></el-input>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label-width="labelWidth" label="排序" prop="sign_up_process_sort">
<el-input-number v-model='formData.sign_up_process_sort' controls-position='right'></el-input-number>
</el-form-item>
</el-col>
</el-row>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button type="primary" @click="handleAddClick(formRef)">添加</el-button>
<el-button @click="handleResetClick(formRef)">重置</el-button>
</span>
</template>
</el-dialog>
</template>
<script setup>
import { reactive, ref, watch } from "vue";
import { addSignUpProcess } from "~/service/sign_up_process";
import { useLoginStore } from "~/store";
// --
// --
// --
const store = useLoginStore();
const headers = {
Accept: "application/json",
...store.headers,
};
const formRef = ref();
const labelWidth = 90;
const props = defineProps({
modelValue: Boolean,
done: Function,
});
const emits = defineEmits(["update:modelValue"]);
const dialogVisible = ref(props.modelValue);
const formData = reactive({});
const uoloadData = ref({
dirName: "SignUpProcess"
})
watch(props, (v) => {
dialogVisible.value = v.modelValue;
});
const rules = reactive({
sign_up_process_title: [
{
required: true,
message: '标题不能为空'
}
],
sign_up_process_icon: [
{
required: true,
message: '图标不能为空'
}
],
sign_up_process_intro: [
{
required: true,
message: '介绍不能为空'
}
],
sign_up_process_sort: [
{
required: true,
message: '排序不能为空'
}
],
});
// --
//
const openDialog = () => {
};
const closeDialog = () => {
handleResetClick(formRef.value);
dialogVisible.value = false;
emits("update:modelValue", false);
};
const handleAddClick = async (formEl) => {
console.log(formData);
if (!formEl) return;
formEl.validate(async (valid) => {
if (!valid) {
return;
}
const { code } = await addSignUpProcess(formData);
if (code == 0) {
closeDialog();
props.done();
}
});
};
const handleResetClick = async (formEl) => {
if (!formEl) return;
formEl.resetFields();
};
</script>
<style lang="less" scoped>
</style>

View File

@ -0,0 +1,84 @@
<template>
<el-dialog v-model="props.modelValue" title="报名流程详情" width="900px" @closed="closeDialog" @open="openDialog">
<el-form ref="formRef" :model="formData" :disabled="true">
<el-row>
<el-col :span="12">
<el-form-item :label-width="labelWidth" label="标题" prop="sign_up_process_title">
<el-input v-model='formData.sign_up_process_title' type="text" placeholder='请输入标题'></el-input>
</el-form-item>
</el-col>
<el-col :span='12'>
<el-form-item :label-width='labelWidth' label='图标' prop='sign_up_process_icon'>
<UploadImage ref='uploadRef' v-model='formData.sign_up_process_icon' :data=uoloadData :limit='1' :fileSize='5'
:drag='true' :isShowTip='false' />
</el-form-item>
</el-col>
<el-col :span="24">
<el-form-item :label-width="labelWidth" label="介绍" prop="sign_up_process_intro">
<el-input v-model='formData.sign_up_process_intro' type="textarea" :rows="5" placeholder='请输入介绍'></el-input>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label-width="labelWidth" label="排序" prop="sign_up_process_sort">
<el-input-number v-model='formData.sign_up_process_sort' controls-position='right'></el-input-number>
</el-form-item>
</el-col>
</el-row>
</el-form>
</el-dialog>
</template>
<script setup>
import { reactive, ref, watch } from "vue";
import { isEmptyObject } from "~/utils/index";
// --
// --
// --
const formRef = ref();
const labelWidth = 100;
const props = defineProps({
modelValue: Boolean,
data: Object,
done: Function,
});
const emits = defineEmits(["update:modelValue"]);
const formData = ref({
...props.data,
});
// --
watch(props, (v) => {
formData.value = v.data;
});
//
const openDialog = () => {
};
const closeDialog = () => {
emits("update:modelValue", false);
};
</script>
<style lang="less" scoped>
</style>

View File

@ -0,0 +1,146 @@
<template>
<el-dialog v-model="props.modelValue" title="编辑报名流程" width="900px" @closed="closeDialog" @open="openDialog">
<el-form ref="formRef" :model="formData" :rules="rules">
<el-row>
<el-col :span="12">
<el-form-item :label-width="labelWidth" label="标题" prop="sign_up_process_title">
<el-input v-model='formData.sign_up_process_title' type="text" placeholder='请输入标题'></el-input>
</el-form-item>
</el-col>
<el-col :span='12'>
<el-form-item :label-width='labelWidth' label='图标' prop='sign_up_process_icon'>
<UploadImage ref='uploadRef' v-model='formData.sign_up_process_icon' :data=uoloadData :limit='1' :fileSize='5'
:drag='true' :isShowTip='false' />
</el-form-item>
</el-col>
<el-col :span="24">
<el-form-item :label-width="labelWidth" label="介绍" prop="sign_up_process_intro">
<el-input v-model='formData.sign_up_process_intro' type="textarea" :rows="5" placeholder='请输入介绍'></el-input>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item :label-width="labelWidth" label="排序" prop="sign_up_process_sort">
<el-input-number v-model='formData.sign_up_process_sort' controls-position='right'></el-input-number>
</el-form-item>
</el-col>
</el-row>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button type="primary" @click="handleEditClick(formRef)">编辑</el-button>
<el-button @click="handleResetClick(formRef)">重置</el-button>
</span>
</template>
</el-dialog>
</template>
<script setup>
import { reactive, ref, watch } from "vue";
import { editSignUpProcess } from "~/service/sign_up_process";
import { useLoginStore } from "~/store";
// --
// --
// --
const store = useLoginStore();
const headers = {
Accept: "application/json",
...store.headers,
};
const formRef = ref();
const labelWidth = 100;
const props = defineProps({
modelValue: Boolean,
data: Object,
done: Function,
});
const emits = defineEmits(["update:modelValue"]);
const formData = ref({
...props.data,
});
const uoloadData = ref({
dirName: "SignUpProcess"
})
// --
watch(props, (v) => {
formData.value = v.data;
});
//
const openDialog = () => {
};
const closeDialog = () => {
props.done();
emits("update:modelValue", false);
};
const rules = reactive({
sign_up_process_title: [
{
required: true,
message: '标题不能为空'
}
],
sign_up_process_icon: [
{
required: true,
message: '图标不能为空'
}
],
sign_up_process_intro: [
{
required: true,
message: '介绍不能为空'
}
],
sign_up_process_sort: [
{
required: true,
message: '排序不能为空'
}
],
});
const handleEditClick = async (formEl) => {
console.log(formData.value);
if (!formEl) return;
formEl.validate(async (valid) => {
if (!valid) {
return;
}
const { code } = await editSignUpProcess(formData.value);
if (code == 0) {
closeDialog();
props.done();
}
});
};
const handleResetClick = async (formEl) => {
if (!formEl) return;
formEl.resetFields();
};
</script>
<style lang="less" scoped>
</style>

View File

@ -0,0 +1,204 @@
<template>
<!-- 面包屑 -->
<el-breadcrumb>
<el-breadcrumb-item>报名流程管理</el-breadcrumb-item>
<el-breadcrumb-item to="/sign_up_process/list">报名流程列表</el-breadcrumb-item>
</el-breadcrumb>
<!-- 搜索 -->
<el-form inline :model="params">
<el-form-item label="标题">
<el-input v-model='params.sign_up_process_title' placeholder='请输入标题' @keyup.enter.native="getList()"></el-input>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="getList()" icon="ElIconSearch">
搜索
</el-button>
</el-form-item>
</el-form>
<el-space style="margin-bottom: 10px;">
<!-- 添加报名流程 -->
<el-col :span="1">
<el-button type="primary" @click="addSignUpProcessDialogVisible = true"> 添加 </el-button>
</el-col>
<!-- 导出 -->
<el-button icon="ElIconDocument" @click="exportExcel(params)">导出</el-button>
<!-- 下拉操作 -->
<el-dropdown v-if="selectionData.length">
<el-button type="primary">
批量操作<el-icon class="el-icon--right"><arrow-down /></el-icon>
</el-button>
<template #dropdown>
<el-dropdown-menu>
<el-dropdown-item @click="handleDelete(selectionData)">
批量删除
</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
</el-space>
<!-- 数据表格 -->
<el-table v-loading="loading" :data="dataList" ref="tableRef" border highlight-current-row
:onSelectionChange="data => (selectionData = data)">
<el-table-column type="selection" width="50" align="center" />
<el-table-column prop="sign_up_process_title" width="250" label="标题" :show-overflow-tooltip="true"> </el-table-column>
<el-table-column prop="sign_up_process_intro" width="300" label="介绍" :show-overflow-tooltip="true"> </el-table-column>
<el-table-column prop="sign_up_process_icon" width="200" label="图标" :show-overflow-tooltip="true" align="center">
<template #default="scope">
<el-image v-if="scope.row.sign_up_process_icon" :src="scope.row.sign_up_process_icon.split(',')[0]" lazy
:preview-src-list="scope.row.sign_up_process_icon.split(',')" :preview-teleported="true"
:hide-on-click-modal="true" fit="contain" class="el-avatar"></el-image>
<template v-else>暂无图片</template>
</template>
</el-table-column>
<el-table-column prop="sign_up_process_sort" width="200" label="排序" sort>
<template #default="scope">
<el-input-number v-model='scope.row.sign_up_process_sort' controls-position="right"
@change="handleEditOrder(scope.row)"></el-input-number>
</template>
</el-table-column>
<el-table-column label="操作" align="center" width="250" fixed='right'>
<template #default="scope">
<el-button size="small" type="primary" @click="handleUpdate(scope.row)">编辑</el-button>
<el-button size="small" type="danger" @click="handleDelete([scope.row])">删除</el-button>
<el-button size="small" type="info" @click="handleDetail(scope.row)">查看</el-button>
</template>
</el-table-column>
</el-table>
<pagination :total="total" v-model:page="params.page" v-model:limit="params.limit" @pagination="getList" />
<!-- 添加报名流程 -->
<AddSignUpProcessDialog v-model="addSignUpProcessDialogVisible" :done="() => getList()">
</AddSignUpProcessDialog>
<!-- 编辑报名流程 -->
<EditSignUpProcessDialog v-model="EditSignUpProcessDialogVisible" :data="EditSignUpProcessDialogRow"
:done="() => getList()"></EditSignUpProcessDialog>
<!-- 报名流程详情 -->
<DetailSignUpProcessDialog v-model="DetailSignUpProcessDialogVisible" :data="DetailSignUpProcessDialogRow">
</DetailSignUpProcessDialog>
</template>
<script setup>
import { ArrowDown } from '@element-plus/icons-vue';
import { ref, reactive, watch } from 'vue';
import { useLoginStore } from "~/store";
import { getSignUpProcessList, editSignUpProcess, deleteSignUpProcess,exportExcel } from '~/service/sign_up_process';
import AddSignUpProcessDialog from './components/AddSignUpProcessDialog.vue';
import EditSignUpProcessDialog from './components/EditSignUpProcessDialog.vue';
import DetailSignUpProcessDialog from './components/DetailSignUpProcessDialog.vue';
import createDraw from 'hyw-drag'
const tableRef = ref();
const selectionData = ref([]);
const store = useLoginStore();
const addSignUpProcessDialogVisible = ref(false);
const EditSignUpProcessDialogVisible = ref(false);
const EditSignUpProcessDialogRow = ref({});
const DetailSignUpProcessDialogVisible = ref(false);
const DetailSignUpProcessDialogRow = ref({});
const headers = {
Accept: "application/json",
...store.headers,
};
const loading = ref(false)
const dataList = ref([])
//
let drag = ref(false)
//
let total = ref()
const params = reactive({
page: 1,
limit: 10,
sign_up_process_title: "",
});
const handleCommand = ({ type, row }) => {
switch (type) {
case "detail":
handleDetail(row);
break;
case 'delete':
handleDelete([row]);
break;
}
};
//
function getList() {
loading.value = true
getSignUpProcessList(params).then((res) => {
if (res.code == 0) {
dataList.value = res.data
setTimeout(() => {
[...document.getElementsByClassName('el-table__row')].map(item => { item.classList.add('row1') });
[...document.getElementsByClassName('el-table__row--level-1')].map(item => { item.classList.remove('row1') })
drag.value === false && (drag.value = createDraw(document.getElementsByClassName('el-table__row')[0].parentElement))
})
total.value = res.count
loading.value = false
}
})
}
watch(drag, () => {
drag.value.start(() => {
dataList.value.map(item => {
tableRef.value.toggleRowExpansion(item, false)
})
tableExpandAll.value = false
})
drag.value.drop((info) => {
let targetIdx = [...document.getElementsByClassName('row1')].indexOf(info.el)
let sourceIdx = [...document.getElementsByClassName('row1')].indexOf(info.info.source.el)
let orderIdx = dataList.value[sourceIdx]['sign_up_process_sort']
dataList.value[targetIdx]['sign_up_process_sort'] = orderIdx
handleEditOrder(dataList.value[targetIdx])
})
})
getList()
//
async function handleEditOrder(data) {
loading.value = true
await editSignUpProcess(data);
getList();
}
//
const handleDelete = data => {
ElMessageBox.confirm(`您确定要删除该报名流程吗?`).then(async () => {
const res = await deleteSignUpProcess({
sign_up_process_guid: data.map(v => v.sign_up_process_guid).join()
});
if (res) {
getList();
}
});
};
//
function handleUpdate(row) {
EditSignUpProcessDialogVisible.value = true
EditSignUpProcessDialogRow.value = row
}
//
function handleDetail(row) {
DetailSignUpProcessDialogVisible.value = true
DetailSignUpProcessDialogRow.value = row
}
</script>

View File

@ -1,30 +1,30 @@
import { api, downloadFile, createApiUrl} from '~/utils/axios';
import { api, downloadFile, createApiUrl } from '~/utils/axios';
/**
* 导出班型
* @param {Object} data
* @return {Promise} api
*/
export function exportExcel(data) {
downloadFile(createApiUrl('Enrol.Classes/exportExcel'), data);
}
/**
* 导出班型
* @param {Object} data
* @return {Promise} api
*/
export function exportExcel(data) {
downloadFile(createApiUrl('Enrol.Classes/exportExcel'), data);
}
/**
* 下载班型模板
* @param {Object} data
* @return {Promise} api
*/
export function downloadTemplate(data) {
downloadFile(createApiUrl('Enrol.Classes/downloadTemplate'), data);
}
/**
* 下载班型模板
* @param {Object} data
* @return {Promise} api
*/
export function downloadTemplate(data) {
downloadFile(createApiUrl('Enrol.Classes/downloadTemplate'), data);
}
/**
* 导入班型
* @param {Object} data
* @return {Promise} api
*/
export const importExcel = createApiUrl('Enrol.Classes/importExcel');
/**
* 导入班型
* @param {Object} data
* @return {Promise} api
*/
export const importExcel = createApiUrl('Enrol.Classes/importExcel');

View File

@ -0,0 +1,63 @@
import { api, downloadFile, createApiUrl} from '~/utils/axios';
/**
* 导出报名流程
* @param {Object} data
* @return {Promise} api
*/
export function exportExcel(data) {
downloadFile(createApiUrl('Enrol.SignUpProcess/exportExcel'), data);
}
/**
* 获取报名流程列表
* @param {Object} data
* @return {Promise} api
*/
export function getSignUpProcessList(data) {
return api.post('Enrol.SignUpProcess/getSignUpProcessList', data);
}
/**
* 删除报名流程
* @param {Object} data
* @return {Promise} api
*/
export function deleteSignUpProcess(data) {
return api.post('Enrol.SignUpProcess/deleteSignUpProcess', data, {
isTransformResponse: true,
isShowSuccessMessage: true,
errorMessageText: '删除失败'
});
}
/**
* 添加报名流程
* @param {Object} data
* @return {Promise} api
*/
export function addSignUpProcess(data) {
return api.post('Enrol.SignUpProcess/addSignUpProcess', data, {
isTransformResponse: true,
isShowSuccessMessage: true,
errorMessageText: '添加失败'
});
}
/**
* 编辑报名流程
* @param {Object} data
* @return {Promise} api
*/
export function editSignUpProcess(data) {
return api.post('Enrol.SignUpProcess/editSignUpProcess', data, {
isTransformResponse: true,
isShowSuccessMessage: true,
errorMessageText: '编辑失败'
});
}