This commit is contained in:
ljx
2025-09-09 10:00:56 +08:00
29 changed files with 997 additions and 923 deletions

View File

@ -7,12 +7,13 @@ VITE_APP_ENV = 'development'
# 开发环境 # 开发环境
VITE_APP_BASE_API = 'http://192.168.110.209:8899' VITE_APP_BASE_API = 'http://192.168.110.209:8899'
# 李陈杰 209 # 李陈杰 209
# VITE_APP_BASE_API = 'http://192.168.110.209:8899'
# 曾涛 # 曾涛
# VITE_APP_BASE_API = 'http://192.168.110.180:8899' # VITE_APP_BASE_API = 'http://192.168.110.180:8899'
# 罗成 # 罗成
# VITE_APP_BASE_API = 'http://192.168.110.188:8899' # VITE_APP_BASE_API = 'http://192.168.110.188:8899'
# 朱银 # 朱银
# VITE_APP_BASE_API = 'http://192.168.110.180:8899' # VITE_APP_BASE_API = 'http://192.168.110.149:8899'
#曾涛 #曾涛
# VITE_APP_BASE_API = 'http://192.168.110.171:8899' # VITE_APP_BASE_API = 'http://192.168.110.171:8899'

View File

@ -8,10 +8,11 @@ import { ProgressCategoryVO, ProgressCategoryForm, ProgressCategoryQuery } from
* @returns {*} * @returns {*}
*/ */
export const listProgressCategory = (id?: string | number): AxiosPromise<any[]> => { export const listProgressCategory = (parentId?: string | number,name?:string): AxiosPromise<any[]> => {
return request({ return request({
url: '/progress/progressCategory/listByParent/' + id, url: '/progress/progressCategory/listByParent',
method: 'get' method: 'get',
params:{parentId,name}
}); });
}; };

View File

@ -78,9 +78,10 @@ export const getTabList = (id: string) => {
* @param parentId * @param parentId
* @returns {*} * @returns {*}
*/ */
export const listProgressCategoryTemplateByParent = (parentId: string | number): AxiosPromise<ProgressCategoryTemplateVO[]> => { export const listProgressCategoryTemplateByParent = (parentId: string | number,name:string): AxiosPromise<ProgressCategoryTemplateVO[]> => {
return request({ return request({
url: '/progress/progressCategoryTemplate/listByParent/' + parentId, url: '/progress/progressCategoryTemplate/listByParent' ,
method: 'get' method: 'get',
params:{parentId,name}
}); });
}; };

View File

@ -164,7 +164,7 @@ export const updateConstructionUserSalary = (data: ConstructionUserSalaryForm) =
*/ */
export const getConstructionUserExit = (query: ConstructionUserExitForm) => { export const getConstructionUserExit = (query: ConstructionUserExitForm) => {
return request({ return request({
url: '/contractor/constructionUserExit/list', url: '/project/constructionUserExit/list',
method: 'get', method: 'get',
params: query params: query
}); });
@ -204,7 +204,6 @@ export const importConstructionUserInfo = (file: string) => {
}); });
}; };
// 获取项目列表 // 获取项目列表
export const ProjectList = (query) => { export const ProjectList = (query) => {
return request({ return request({
@ -216,7 +215,7 @@ export const ProjectList = (query) => {
// 获取班组列表 // 获取班组列表
export const TeamList = (query) => { export const TeamList = (query) => {
return request({ return request({
url: '/contractor/constructionUser/teamList', url: '/contractor/constructionUser/teamList',
method: 'get', method: 'get',
params: query params: query
@ -230,4 +229,4 @@ export const TeamDistribution = (data) => {
method: 'post', method: 'post',
data: data data: data
}); });
}; };

View File

@ -10,7 +10,7 @@ import { ConstructionUserFileVO, ConstructionUserFileForm, ConstructionUserFileQ
export const listConstructionUserFile = (query?: ConstructionUserFileQuery): AxiosPromise<ConstructionUserFileVO[]> => { export const listConstructionUserFile = (query?: ConstructionUserFileQuery): AxiosPromise<ConstructionUserFileVO[]> => {
return request({ return request({
url: '/project/constructionUserFile/list', url: '/contractor/constructionUserFile/list',
method: 'get', method: 'get',
params: query params: query
}); });
@ -22,7 +22,7 @@ export const listConstructionUserFile = (query?: ConstructionUserFileQuery): Axi
*/ */
export const setConstructionUserFile = (data: ConstructionUserFileForm): AxiosPromise<string | number> => { export const setConstructionUserFile = (data: ConstructionUserFileForm): AxiosPromise<string | number> => {
return request({ return request({
url: '/project/constructionUserFile/save', url: '/contractor/constructionUserFile/save',
method: 'post', method: 'post',
data data
}); });
@ -34,7 +34,7 @@ export const setConstructionUserFile = (data: ConstructionUserFileForm): AxiosPr
*/ */
export const delConstructionUserFile = (id: string | number | Array<string | number>) => { export const delConstructionUserFile = (id: string | number | Array<string | number>) => {
return request({ return request({
url: '/project/constructionUserFile/' + id, url: '/contractor/constructionUserFile/' + id,
method: 'delete' method: 'delete'
}); });
}; };

View File

@ -1,6 +1,6 @@
import request from '@/utils/request'; import request from '@/utils/request';
import { AxiosPromise } from 'axios'; import { AxiosPromise } from 'axios';
import { QuestionBankVO, QuestionBankForm, QuestionBankQuery } from '@/api/safety/questionBank/types'; import { QuestionBankVO, QuestionBankForm, QuestionBankQuery } from '@/api/safety/wgzQuestionBank/types';
/** /**
* 查询题库列表 * 查询题库列表
@ -10,7 +10,7 @@ import { QuestionBankVO, QuestionBankForm, QuestionBankQuery } from '@/api/safet
export const listQuestionBank = (query?: QuestionBankQuery): AxiosPromise<QuestionBankVO[]> => { export const listQuestionBank = (query?: QuestionBankQuery): AxiosPromise<QuestionBankVO[]> => {
return request({ return request({
url: '/safety/questionBank/list', url: '/safety/wgzQuestionBank/list',
method: 'get', method: 'get',
params: query params: query
}); });
@ -22,7 +22,7 @@ export const listQuestionBank = (query?: QuestionBankQuery): AxiosPromise<Questi
*/ */
export const getQuestionBank = (id: string | number): AxiosPromise<QuestionBankVO> => { export const getQuestionBank = (id: string | number): AxiosPromise<QuestionBankVO> => {
return request({ return request({
url: '/safety/questionBank/' + id, url: '/safety/wgzQuestionBank/' + id,
method: 'get' method: 'get'
}); });
}; };
@ -33,7 +33,7 @@ export const getQuestionBank = (id: string | number): AxiosPromise<QuestionBankV
*/ */
export const addQuestionBank = (data: QuestionBankForm) => { export const addQuestionBank = (data: QuestionBankForm) => {
return request({ return request({
url: '/safety/questionBank', url: '/safety/wgzQuestionBank',
method: 'post', method: 'post',
data: data data: data
}); });
@ -45,7 +45,7 @@ export const addQuestionBank = (data: QuestionBankForm) => {
*/ */
export const updateQuestionBank = (data: QuestionBankForm) => { export const updateQuestionBank = (data: QuestionBankForm) => {
return request({ return request({
url: '/safety/questionBank', url: '/safety/wgzQuestionBank',
method: 'put', method: 'put',
data: data data: data
}); });
@ -57,7 +57,7 @@ export const updateQuestionBank = (data: QuestionBankForm) => {
*/ */
export const delQuestionBank = (id: string | number | Array<string | number>) => { export const delQuestionBank = (id: string | number | Array<string | number>) => {
return request({ return request({
url: '/safety/questionBank/' + id, url: '/safety/wgzQuestionBank/' + id,
method: 'delete' method: 'delete'
}); });
}; };

View File

@ -1,6 +1,6 @@
import request from '@/utils/request'; import request from '@/utils/request';
import { AxiosPromise } from 'axios'; import { AxiosPromise } from 'axios';
import { QuestionUserAnswerVO, QuestionUserAnswerForm, QuestionUserAnswerQuery } from '@/api/safety/questionUserAnswer/types'; import { QuestionUserAnswerVO, QuestionUserAnswerForm, QuestionUserAnswerQuery } from '@/api/safety/wgzQuestionSave/types';
/** /**
* 查询用户试卷存储列表 * 查询用户试卷存储列表
@ -10,7 +10,7 @@ import { QuestionUserAnswerVO, QuestionUserAnswerForm, QuestionUserAnswerQuery }
export const listQuestionUserAnswer = (query?: QuestionUserAnswerQuery): AxiosPromise<QuestionUserAnswerVO[]> => { export const listQuestionUserAnswer = (query?: QuestionUserAnswerQuery): AxiosPromise<QuestionUserAnswerVO[]> => {
return request({ return request({
url: '/safety/questionUserAnswer/list', url: '/safety/wgzQuestionSave/list',
method: 'get', method: 'get',
params: query params: query
}); });
@ -22,7 +22,7 @@ export const listQuestionUserAnswer = (query?: QuestionUserAnswerQuery): AxiosPr
*/ */
export const getQuestionUserAnswer = (id: string | number): AxiosPromise<QuestionUserAnswerVO> => { export const getQuestionUserAnswer = (id: string | number): AxiosPromise<QuestionUserAnswerVO> => {
return request({ return request({
url: '/safety/questionUserAnswer/' + id, url: '/safety/wgzQuestionSave/' + id,
method: 'get' method: 'get'
}); });
}; };
@ -33,7 +33,7 @@ export const getQuestionUserAnswer = (id: string | number): AxiosPromise<Questio
*/ */
export const addQuestionUserAnswer = (data: QuestionUserAnswerForm) => { export const addQuestionUserAnswer = (data: QuestionUserAnswerForm) => {
return request({ return request({
url: '/safety/questionUserAnswer', url: '/safety/wgzQuestionSave',
method: 'post', method: 'post',
data: data data: data
}); });
@ -45,7 +45,7 @@ export const addQuestionUserAnswer = (data: QuestionUserAnswerForm) => {
*/ */
export const updateQuestionUserAnswer = (data: QuestionUserAnswerForm) => { export const updateQuestionUserAnswer = (data: QuestionUserAnswerForm) => {
return request({ return request({
url: '/safety/questionUserAnswer', url: '/safety/wgzQuestionSave',
method: 'put', method: 'put',
data: data data: data
}); });
@ -57,7 +57,7 @@ export const updateQuestionUserAnswer = (data: QuestionUserAnswerForm) => {
*/ */
export const delQuestionUserAnswer = (id: string | number | Array<string | number>) => { export const delQuestionUserAnswer = (id: string | number | Array<string | number>) => {
return request({ return request({
url: '/safety/questionUserAnswer/' + id, url: '/safety/wgzQuestionSave/' + id,
method: 'delete' method: 'delete'
}); });
}; };
@ -68,7 +68,7 @@ export const delQuestionUserAnswer = (id: string | number | Array<string | numbe
*/ */
export const uploadQuestionUserAnswer = (data: any) => { export const uploadQuestionUserAnswer = (data: any) => {
return request({ return request({
url: '/safety/questionUserAnswer/upload/zip', url: '/safety/wgzQuestionSave/upload/zip',
method: 'post', method: 'post',
data: data data: data
}); });

View File

@ -10,7 +10,7 @@ import { QuestionsCategoryVO, QuestionsCategoryForm, QuestionsCategoryQuery } fr
export const listQuestionsCategory = (query?: QuestionsCategoryQuery): AxiosPromise<QuestionsCategoryVO[]> => { export const listQuestionsCategory = (query?: QuestionsCategoryQuery): AxiosPromise<QuestionsCategoryVO[]> => {
return request({ return request({
url: '/safety/questionsCategory/list', url: '/safety/wzgQuestionCategory/list',
method: 'get', method: 'get',
params: query params: query
}); });
@ -22,7 +22,7 @@ export const listQuestionsCategory = (query?: QuestionsCategoryQuery): AxiosProm
*/ */
export const getQuestionsCategory = (id: string | number): AxiosPromise<QuestionsCategoryVO> => { export const getQuestionsCategory = (id: string | number): AxiosPromise<QuestionsCategoryVO> => {
return request({ return request({
url: '/safety/questionsCategory/' + id, url: '/safety/wzgQuestionCategory' + id,
method: 'get' method: 'get'
}); });
}; };
@ -33,7 +33,7 @@ export const getQuestionsCategory = (id: string | number): AxiosPromise<Question
*/ */
export const addQuestionsCategory = (data: QuestionsCategoryForm) => { export const addQuestionsCategory = (data: QuestionsCategoryForm) => {
return request({ return request({
url: '/safety/questionsCategory', url: '/safety/wzgQuestionCategory',
method: 'post', method: 'post',
data: data data: data
}); });
@ -45,7 +45,7 @@ export const addQuestionsCategory = (data: QuestionsCategoryForm) => {
*/ */
export const updateQuestionsCategory = (data: QuestionsCategoryForm) => { export const updateQuestionsCategory = (data: QuestionsCategoryForm) => {
return request({ return request({
url: '/safety/questionsCategory', url: '/safety/wzgQuestionCategory',
method: 'put', method: 'put',
data: data data: data
}); });
@ -57,7 +57,7 @@ export const updateQuestionsCategory = (data: QuestionsCategoryForm) => {
*/ */
export const delQuestionsCategory = (id: string | number | Array<string | number>) => { export const delQuestionsCategory = (id: string | number | Array<string | number>) => {
return request({ return request({
url: '/safety/questionsCategory/' + id, url: '/safety/wzgQuestionCategory' + id,
method: 'delete' method: 'delete'
}); });
}; };

View File

@ -22,7 +22,7 @@ export const listQuestionsConfig = (query?: QuestionsConfigQuery): AxiosPromise<
*/ */
export const getQuestionsConfig = (id: string | number): AxiosPromise<QuestionsConfigVO> => { export const getQuestionsConfig = (id: string | number): AxiosPromise<QuestionsConfigVO> => {
return request({ return request({
url: '/safety/questionsConfig/' + id, url: '/safety/wzgQuestionsConfiguration/' + id,
method: 'get' method: 'get'
}); });
}; };
@ -45,7 +45,7 @@ export const addQuestionsConfig = (data: QuestionsConfigForm) => {
*/ */
export const updateQuestionsConfig = (data: QuestionsConfigForm) => { export const updateQuestionsConfig = (data: QuestionsConfigForm) => {
return request({ return request({
url: '/safety/questionsConfig', url: '/safety/wzgQuestionsConfiguration',
method: 'put', method: 'put',
data: data data: data
}); });

View File

@ -203,6 +203,7 @@ watch(
}); });
} else { } else {
fileList.value = []; fileList.value = [];
return []; return [];
} }
}, },
@ -293,7 +294,7 @@ const handleChange = (file: any, filelist: any) => {
} }
} }
// 记录 status = 'ready' 的文件 // 记录 status = 'ready' 的文件
if (file.status === 'ready') { if (file.status === 'ready' && !props.isConstruction) {
pendingFiles.value.push(file); pendingFiles.value.push(file);
fileList.value = pendingFiles.value; fileList.value = pendingFiles.value;
} }

View File

@ -170,8 +170,6 @@ onMounted(() => {
//分页 //分页
const getWaitingList = () => { const getWaitingList = () => {
pageByTaskWait({ pageNum: 1, pageSize: 10 }).then((resp) => { pageByTaskWait({ pageNum: 1, pageSize: 10 }).then((resp) => {
console.log(resp);
total.value = resp.total; total.value = resp.total;
}); });
}; };

View File

@ -17,7 +17,7 @@
<template v-if="item.meta" #title> <template v-if="item.meta" #title>
<svg-icon :icon-class="item.meta ? item.meta.icon : ''" /> <svg-icon :icon-class="item.meta ? item.meta.icon : ''" />
<span class="menu-title" :title="hasTitle(item.meta?.title)">{{ item.meta?.title }}</span> <span class="menu-title" :title="hasTitle(item.meta?.title)">{{ item.meta?.title }}</span>
<!-- <span class="bage" v-if="item.meta?.title == '我的任务' && total >= 0">{{ total }}</span> --> <span class="bage" v-if="item.meta?.title == '我的任务' && total > 0">{{ total }}</span>
</template> </template>
<sidebar-item <sidebar-item
@ -59,10 +59,10 @@ const props = defineProps({
const total = ref(0); const total = ref(0);
const totalChao = ref(0); const totalChao = ref(0);
onMounted(() => { onMounted(() => {
if (onlyOneChild.value.meta?.title == '我的待办') { if (onlyOneChild.value.meta?.title == '我的待办' || props.item.meta?.title == '我的任务') {
getWaitingList(); getWaitingList();
} }
if (onlyOneChild.value.meta?.title == '我的抄送') { if (onlyOneChild.value.meta?.title == '我的抄送') {
getChaoList(); getChaoList();
} }
}); });
@ -130,9 +130,7 @@ const hasTitle = (title: string | undefined): string => {
watch( watch(
() => noticeStore.state.value.notices, () => noticeStore.state.value.notices,
(newVal) => { (newVal) => {
if (onlyOneChild.value.meta?.title == '我的待办') { if (onlyOneChild.value.meta?.title == '我的待办' || props.item.meta?.title == '我的任务') {
console.log(121212121);
let time = setTimeout(() => { let time = setTimeout(() => {
getWaitingList(); getWaitingList();
clearTimeout(time); clearTimeout(time);

View File

@ -47,9 +47,9 @@
<el-row :gutter="8" class="mb-3 font-medium text-gray-700 whitespace-nowrap"> <el-row :gutter="8" class="mb-3 font-medium text-gray-700 whitespace-nowrap">
<el-col :span="4">专业</el-col> <el-col :span="4">专业</el-col>
<el-col :span="5">设计人员可多选</el-col> <el-col :span="5">设计人员可多选</el-col>
<el-col :span="5">校审人员</el-col> <el-col :span="5">校审人员</el-col> <el-col :span="4">审核人员</el-col>
<el-col :span="5">审定人员</el-col> <el-col :span="5">审定人员</el-col>
<el-col :span="4">审核人员</el-col>
<el-col :span="3"></el-col> <el-col :span="3"></el-col>
</el-row> </el-row>
@ -209,61 +209,7 @@
</el-col> </el-col>
<!-- 4. 审定人员 --> <!-- 4. 审定人员 -->
<el-col :span="5" class="mb-4 sm:mb-0">
<div class="pl-2 border-l-2 border-orange-200 py-2">
<div class="space-y-3">
<div
v-for="(person, personIndex) in majorConfig.approvedPersons"
:key="`approved-${configIndex}-${personIndex}`"
class="flex items-center"
>
<el-form-item
:prop="`approved.${configIndex}.persons.${personIndex}.userId`"
:rules="{ required: true, message: '请选择审定人员', trigger: 'change' }"
class="flex-1 mr-2 mb-0"
label="审定"
label-width="50px"
>
<el-select
filterable
v-model="person.userId"
placeholder="选择人员"
class="w-full transition-all duration-300 border-gray-300"
@change="() => checkDuplicate(person, 'approved', configIndex, personIndex)"
>
<el-option v-for="item in userList" :key="`user-${item.userId}`" :label="item.nickName" :value="item.userId" />
</el-select>
</el-form-item>
<!-- <div class="flex gap-1">
<el-button
type="danger"
size="small"
@click="removePerson('approved', configIndex, personIndex)"
class="transition-all duration-300 hover:bg-red-600"
:disabled="majorConfig.approvedPersons.length <= 1 || disabledForm"
>
<el-icon :size="14"><Delete /></el-icon>
</el-button>
<el-button
type="success"
size="small"
@click="addPerson('approved', configIndex)"
class="transition-all duration-300 transform hover:scale-105"
:disabled="!form.designers[configIndex].userMajor || disabledForm"
>
<el-icon :size="14"><Plus /></el-icon>
</el-button>
</div> -->
</div>
</div>
<div
v-if="majorConfig.approvedPersons.length == 0"
class="text-gray-500 text-xs py-2 bg-gray-100 rounded border border-dashed border-gray-200"
>
点击"添加"
</div>
</div>
</el-col>
<!-- 5. 审核人员 --> <!-- 5. 审核人员 -->
<el-col :span="5" class="mb-4 sm:mb-0"> <el-col :span="5" class="mb-4 sm:mb-0">
@ -321,6 +267,61 @@
</div> </div>
</div> </div>
</el-col> </el-col>
<el-col :span="5" class="mb-4 sm:mb-0">
<div class="pl-2 border-l-2 border-orange-200 py-2">
<div class="space-y-3">
<div
v-for="(person, personIndex) in majorConfig.approvedPersons"
:key="`approved-${configIndex}-${personIndex}`"
class="flex items-center"
>
<el-form-item
:prop="`approved.${configIndex}.persons.${personIndex}.userId`"
:rules="{ required: true, message: '请选择审定人员', trigger: 'change' }"
class="flex-1 mr-2 mb-0"
label="审定"
label-width="50px"
>
<el-select
filterable
v-model="person.userId"
placeholder="选择人员"
class="w-full transition-all duration-300 border-gray-300"
@change="() => checkDuplicate(person, 'approved', configIndex, personIndex)"
>
<el-option v-for="item in userList" :key="`user-${item.userId}`" :label="item.nickName" :value="item.userId" />
</el-select>
</el-form-item>
<!-- <div class="flex gap-1">
<el-button
type="danger"
size="small"
@click="removePerson('approved', configIndex, personIndex)"
class="transition-all duration-300 hover:bg-red-600"
:disabled="majorConfig.approvedPersons.length <= 1 || disabledForm"
>
<el-icon :size="14"><Delete /></el-icon>
</el-button>
<el-button
type="success"
size="small"
@click="addPerson('approved', configIndex)"
class="transition-all duration-300 transform hover:scale-105"
:disabled="!form.designers[configIndex].userMajor || disabledForm"
>
<el-icon :size="14"><Plus /></el-icon>
</el-button>
</div> -->
</div>
</div>
<div
v-if="majorConfig.approvedPersons.length == 0"
class="text-gray-500 text-xs py-2 bg-gray-100 rounded border border-dashed border-gray-200"
>
点击"添加"
</div>
</div>
</el-col>
<!-- 操作列 --> <!-- 操作列 -->
<el-col :span="2" class="pr-4 mt-2 text-right"> <el-col :span="2" class="pr-4 mt-2 text-right">
@ -343,13 +344,17 @@
type="primary" type="primary"
size="large" size="large"
v-hasPermi="['design:user:batch']" v-hasPermi="['design:user:batch']"
icon="Check"
@click="submitForm" @click="submitForm"
class="px-8 py-2.5 transition-all duration-300 transform hover:scale-105 bg-blue-500 hover:bg-blue-600 text-white font-medium" class="px-8 py-2.5 transition-all duration-300 transform hover:scale-105 bg-blue-500 hover:bg-blue-600 text-white font-medium"
> >确认提交
<i class="el-icon-check mr-2"></i>确认提交
</el-button> </el-button>
<el-button size="large" @click="resetForm" class="px-8 py-2.5 transition-all duration-300 border-gray-300 hover:bg-gray-100 font-medium"> <el-button
<i class="el-icon-refresh mr-2"></i>重置 size="large"
icon="Refresh"
@click="resetForm"
class="px-8 py-2.5 transition-all duration-300 border-gray-300 hover:bg-gray-100 font-medium"
>重置
</el-button> </el-button>
</div> </div>
</el-form> </el-form>

View File

@ -62,13 +62,13 @@
:accept="'.xlsx,.xls'" :accept="'.xlsx,.xls'"
:limit="1" :limit="1"
> >
<el-button type="primary" icon="Upload">导入文件</el-button> <el-button type="primary" icon="Download">导入文件</el-button>
</el-upload> </el-upload>
<el-button <el-button
v-if="!form.id || form.status == 'draft'" v-if="!form.id || form.status == 'draft'"
type="primary" type="primary"
style="margin-left: 20px" style="margin-left: 20px"
icon="Download" icon="Upload"
@click="exportTemplate" @click="exportTemplate"
class="transition-all hover:bg-blue-600" class="transition-all hover:bg-blue-600"
> >

View File

@ -1,9 +1,7 @@
<template> <template>
<formalitiesAreConsolidated ref="formalitiesAreConsolidatedRef" class="overlay" <formalitiesAreConsolidated ref="formalitiesAreConsolidatedRef" class="overlay" v-if="showFormalitiesAreConsolidated" />
v-if="showFormalitiesAreConsolidated" />
<div class="p-2" v-else> <div class="p-2" v-else>
<transition :enter-active-class="proxy?.animate.searchAnimate.enter" <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
:leave-active-class="proxy?.animate.searchAnimate.leave">
<div v-show="showSearch" class="mb-[10px]"> <div v-show="showSearch" class="mb-[10px]">
<el-card shadow="hover"> <el-card shadow="hover">
<el-form ref="queryFormRef" :model="queryParams" :inline="true"> <el-form ref="queryFormRef" :model="queryParams" :inline="true">
@ -23,12 +21,12 @@
<template #header> <template #header>
<el-row :gutter="10" class="mb8"> <el-row :gutter="10" class="mb8">
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="primary" plain icon="Plus" @click="handleAdd" <el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['formalities:listOfFormalities:add']">新增</el-button>
v-hasPermi="['formalities:listOfFormalities:add']">新增</el-button>
</el-col> </el-col>
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="primary" :disabled="multiple" plain icon="Plus" @click="handleOk" <el-button type="primary" :disabled="multiple" plain icon="Plus" @click="handleOk" v-hasPermi="['formalities:listOfFormalities:add']"
v-hasPermi="['formalities:listOfFormalities:add']">确认</el-button> >确认</el-button
>
</el-col> </el-col>
<right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar> <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
</el-row> </el-row>
@ -44,8 +42,7 @@
<el-table-column type="selection" width="55" align="center" /> <el-table-column type="selection" width="55" align="center" />
<el-table-column label="名称" prop="name" /> <el-table-column label="名称" prop="name" />
</el-table> </el-table>
<pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" <pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
v-model:limit="queryParams.pageSize" @pagination="getList" />
</el-card> </el-card>
<!-- 添加或修改手续办理清单模板对话框 --> <!-- 添加或修改手续办理清单模板对话框 -->

File diff suppressed because it is too large Load Diff

View File

@ -85,7 +85,7 @@ const handleQuery = () => {
/** 重置按钮操作 */ /** 重置按钮操作 */
const resetQuery = () => { const resetQuery = () => {
queryParams.value.month = ''; resetMonth();
handleQuery(); handleQuery();
}; };
// 获取列表 // 获取列表
@ -102,14 +102,17 @@ const getList = async () => {
total.value = res.total; total.value = res.total;
} }
}; };
onMounted(() => { const resetMonth=()=>{
const currentDate = new Date(); const currentDate = new Date();
const year = currentDate.getFullYear(); const year = currentDate.getFullYear();
const month = currentDate.getMonth() + 1; // 月份从0开始所以需要加1 const month = currentDate.getMonth() + 1; // 月份从0开始所以需要加1
// 形成"YYYY-M"格式 // 形成"YYYY-M"格式
const formattedDate = `${year}-${String(month).padStart(2, '0')}`; const formattedDate = `${year}-${String(month).padStart(2, '0')}`;
queryParams.value.month = formattedDate; queryParams.value.month = formattedDate;
}
onMounted(() => {
resetMonth();
getList(); getList();
}); });

View File

@ -120,17 +120,17 @@
<!-- <el-input v-model="form.projectStructure" placeholder="请输入对应项目结构" /> --> <!-- <el-input v-model="form.projectStructure" placeholder="请输入对应项目结构" /> -->
</el-form-item> </el-form-item>
<el-form-item label="预计开始时间" prop="planStartDate"> <el-form-item label="预计开始时间" prop="planStartDate">
<el-date-picker clearable v-model="form.planStartDate" type="datetime" value-format="YYYY-MM-DD HH:mm:ss" placeholder="选择预计开始时间" /> <el-date-picker clearable v-model="form.planStartDate" type="date" value-format="YYYY-MM-DD" placeholder="选择预计开始时间" />
</el-form-item> </el-form-item>
<el-form-item label="预计结束时间" prop="planEndDate"> <el-form-item label="预计结束时间" prop="planEndDate">
<el-date-picker clearable v-model="form.planEndDate" type="datetime" value-format="YYYY-MM-DD HH:mm:ss" placeholder="选择预计结束时间" /> <el-date-picker clearable v-model="form.planEndDate" type="date" value-format="YYYY-MM-DD" placeholder="选择预计结束时间" />
</el-form-item> </el-form-item>
<el-form-item label="实际开始时间" prop="practicalStartDate"> <el-form-item label="实际开始时间" prop="practicalStartDate">
<el-date-picker <el-date-picker
clearable clearable
v-model="form.practicalStartDate" v-model="form.practicalStartDate"
type="datetime" type="date"
value-format="YYYY-MM-DD HH:mm:ss" value-format="YYYY-MM-DD"
placeholder="选择实际开始时间" placeholder="选择实际开始时间"
/> />
</el-form-item> </el-form-item>
@ -138,8 +138,8 @@
<el-date-picker <el-date-picker
clearable clearable
v-model="form.practicalEndDate" v-model="form.practicalEndDate"
type="datetime" type="date"
value-format="YYYY-MM-DD HH:mm:ss" value-format="YYYY-MM-DD"
placeholder="选择实际结束时间" placeholder="选择实际结束时间"
/> />
</el-form-item> </el-form-item>

View File

@ -216,7 +216,7 @@ const { queryParams, form, rules } = toRefs(data);
/** 查询进度类别模版列表 */ /** 查询进度类别模版列表 */
const getList = async () => { const getList = async () => {
loading.value = true; loading.value = true;
const res = await listProgressCategoryTemplateByParent(activeTab.value); const res = await listProgressCategoryTemplateByParent(activeTab.value,queryParams.value.name);
const data = proxy?.handleTree<ProgressCategoryTemplateVO>(res.data, 'id', 'parentId'); const data = proxy?.handleTree<ProgressCategoryTemplateVO>(res.data, 'id', 'parentId');
if (data) { if (data) {
progressCategoryTemplateList.value = data; progressCategoryTemplateList.value = data;

View File

@ -44,11 +44,11 @@
{{ userDetail?.sfzNumber }} {{ userDetail?.sfzNumber }}
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <!-- <el-col :span="12">
<el-form-item label="身份证号码"> <el-form-item label="身份证号码">
{{ userDetail?.sfzNumber }} {{ userDetail?.sfzNumber }}
</el-form-item> </el-form-item>
</el-col> </el-col> -->
<el-col :span="12"> <el-col :span="12">
<el-form-item label="身份证有效开始期"> <el-form-item label="身份证有效开始期">
{{ dayjs(userDetail?.sfzStart).format('YYYY 年 MM 月 DD 日') }} {{ dayjs(userDetail?.sfzStart).format('YYYY 年 MM 月 DD 日') }}
@ -59,7 +59,7 @@
{{ dayjs(userDetail?.sfzEnd).format('YYYY 年 MM 月 DD 日') }} {{ dayjs(userDetail?.sfzEnd).format('YYYY 年 MM 月 DD 日') }}
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="24">
<el-form-item label="身份证地址"> <el-form-item label="身份证地址">
{{ userDetail?.sfzSite }} {{ userDetail?.sfzSite }}
</el-form-item> </el-form-item>

View File

@ -44,14 +44,14 @@
{{ userDetail?.sfzNumber }} {{ userDetail?.sfzNumber }}
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <!-- <el-col :span="12">
<el-form-item label="身份证号码"> <el-form-item label="身份证号码">
{{ userDetail?.sfzNumber }} {{ userDetail?.sfzNumber }}
</el-form-item> </el-form-item>
</el-col> </el-col> -->
<el-col :span="12"> <el-col :span="12">
<el-form-item label="身份证有效开始期"> <el-form-item label="身份证有效开始期">
{{ dayjs(userDetail?.sfzStart).format('YYYY 年 MM 月 DD 日') }} {{ dayjs(userDetail?.sfzStart).format('YYYY年 MM 月 DD 日') }}
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">

View File

@ -373,7 +373,7 @@
<div class="image_upload" v-for="(item, index) in uploadPath" :key="item.value"> <div class="image_upload" v-for="(item, index) in uploadPath" :key="item.value">
<div class="title">{{ item.label }}</div> <div class="title">{{ item.label }}</div>
<div class="file_upload_all" v-if="item.value != 7"> <div class="file_upload_all" v-if="item.value != 7">
<file-upload v-model="item.path" isConstruction :isShowTip="false" :limit="10" :file-type="['pdf']" :file-size="50" /> <file-upload v-model="item.path" isConstruction show-file-list :isShowTip="false" :limit="10" :file-type="['pdf']" :file-size="50" />
</div> </div>
</div> </div>
<template #footer> <template #footer>
@ -485,7 +485,7 @@
<el-select <el-select
v-model="personnelAllocationObject.teamId" v-model="personnelAllocationObject.teamId"
:disabled="!personnelAllocationObject.projectId" :disabled="!personnelAllocationObject.projectId"
placeholder="请选择分包单位" placeholder="请选择班组"
style="width: 240px" style="width: 240px"
> >
<el-option v-for="item in teamList" :key="item.id" :label="item.teamName" :value="item.id" /> <el-option v-for="item in teamList" :key="item.id" :label="item.teamName" :value="item.id" />
@ -1027,7 +1027,7 @@ const handleExit = async (row: ConstructionUserVO) => {
//上传按钮 //上传按钮
const handleUpload = async (row: ConstructionUserVO) => { const handleUpload = async (row: ConstructionUserVO) => {
const _id = row?.id || ids.value[0]; const _id = row?.sysUserId;
currentUserId.value = _id; currentUserId.value = _id;
const res = await listConstructionUserFile({ userId: _id }); const res = await listConstructionUserFile({ userId: _id });
fileList.value = res.data; fileList.value = res.data;
@ -1163,7 +1163,11 @@ const listeningProject = watch(
const handleAssign = async (row: ConstructionUserVO) => { const handleAssign = async (row: ConstructionUserVO) => {
const _id = row?.id || ids.value[0]; const _id = row?.id || ids.value[0];
currentUserId.value = _id; currentUserId.value = _id;
personnelAllocationObject.projectId = '';
personnelAllocationObject.postId = '';
personnelAllocationObject.teamId = '';
personnelAllocationObject.memberId = row?.sysUserId; personnelAllocationObject.memberId = row?.sysUserId;
skipName.value = row?.userName;
personnelAllocation.value = true; personnelAllocation.value = true;
}; };
// 选择项目1 // 选择项目1

View File

@ -14,15 +14,6 @@
<el-option v-for="dict in safety_question_type" :key="dict.value" :label="dict.label" :value="dict.value" /> <el-option v-for="dict in safety_question_type" :key="dict.value" :label="dict.label" :value="dict.value" />
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item label="题目内容" prop="questionContent">
<el-input v-model="queryParams.questionContent" placeholder="请输入题目内容" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="选项" prop="options">
<el-input v-model="queryParams.options" placeholder="请输入选项" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="正确答案" prop="correctAnswer">
<el-input v-model="queryParams.correctAnswer" placeholder="请输入正确答案" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item> <el-form-item>
<el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button> <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
<el-button icon="Refresh" @click="resetQuery">重置</el-button> <el-button icon="Refresh" @click="resetQuery">重置</el-button>
@ -38,32 +29,18 @@
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['safety:questionBank:add']"> 新增 </el-button> <el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['safety:questionBank:add']"> 新增 </el-button>
</el-col> </el-col>
<el-col :span="1.5">
<el-button type="success" plain icon="Edit" :disabled="single" @click="handleUpdate()" v-hasPermi="['safety:questionBank:edit']"
>修改
</el-button>
</el-col>
<el-col :span="1.5">
<el-button type="danger" plain icon="Delete" :disabled="multiple" @click="handleDelete()" v-hasPermi="['safety:questionBank:remove']"
>删除
</el-button>
</el-col>
<el-col :span="1.5">
<el-button type="warning" plain icon="Download" @click="handleExport" v-hasPermi="['safety:questionBank:export']">导出 </el-button>
</el-col>
<right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar> <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
</el-row> </el-row>
</template> </template>
<el-table v-loading="loading" :data="questionBankList" @selection-change="handleSelectionChange"> <el-table v-loading="loading" :data="questionBankList">
<el-table-column type="selection" width="55" align="center" /> <el-table-column label="序号" align="center" type="index" width="80" />
<el-table-column label="主键id" align="center" prop="id" v-if="true" /> <el-table-column label="题目类别" align="center" width="100" prop="categoryType">
<el-table-column label="题目类别" align="center" prop="categoryType">
<template #default="scope"> <template #default="scope">
<dict-tag :options="safety_question_category_type" :value="scope.row.categoryType" /> <dict-tag :options="safety_question_category_type" :value="scope.row.categoryType" />
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="题目类型" align="center" prop="questionType"> <el-table-column label="题目类型" align="center" width="100" prop="questionType">
<template #default="scope"> <template #default="scope">
<dict-tag :options="safety_question_type" :value="scope.row.questionType" /> <dict-tag :options="safety_question_type" :value="scope.row.questionType" />
</template> </template>
@ -73,16 +50,11 @@
<el-table-column label="正确答案" align="center" prop="correctAnswer" /> <el-table-column label="正确答案" align="center" prop="correctAnswer" />
<el-table-column label="操作" align="center" class-name="small-padding fixed-width"> <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
<template #default="scope"> <template #default="scope">
<el-tooltip content="修改" placement="top"> <el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)" v-hasPermi="['safety:questionBank:edit']">修改</el-button>
<el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)" v-hasPermi="['safety:questionBank:edit']"></el-button> <el-button link type="primary" icon="Delete" @click="handleDelete(scope.row)" v-hasPermi="['safety:questionBank:remove']">删除</el-button>
</el-tooltip>
<el-tooltip content="删除" placement="top">
<el-button link type="primary" icon="Delete" @click="handleDelete(scope.row)" v-hasPermi="['safety:questionBank:remove']"></el-button>
</el-tooltip>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" /> <pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
</el-card> </el-card>
<!-- 添加或修改题库对话框 --> <!-- 添加或修改题库对话框 -->

View File

@ -1,7 +1,6 @@
<template> <template>
<div class="p-2"> <div class="p-2">
<transition :enter-active-class="proxy?.animate.searchAnimate.enter" <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
:leave-active-class="proxy?.animate.searchAnimate.leave">
<div v-show="showSearch" class="mb-[10px]"> <div v-show="showSearch" class="mb-[10px]">
<el-card shadow="hover"> <el-card shadow="hover">
<el-form ref="queryFormRef" :model="queryParams" :inline="true"> <el-form ref="queryFormRef" :model="queryParams" :inline="true">
@ -31,23 +30,27 @@
:上传压缩包内的文件夹名称需设置为姓名-身份证-满分-得分-及格分 <br /> :上传压缩包内的文件夹名称需设置为姓名-身份证-满分-得分-及格分 <br />
例如:小明-5130112333654X-100-59-60 例如:小明-5130112333654X-100-59-60
</template> </template>
<file-upload :limit="1" v-model:model-value="filePath" isImportInfo :fileType="['zip']" <file-upload
uploadUrl="/safety/questionUserAnswer/upload/zip" :file-size="5000" :limit="1"
:data="{ projectId: currentProject.id }"><el-button type="success" plain v-model:model-value="filePath"
icon="Upload">上传线下安全考试</el-button></file-upload> isImportInfo
:fileType="['zip']"
uploadUrl="/safety/wgzQuestionSave/upload/zip"
:file-size="5000"
:data="{ projectId: currentProject.id }"
><el-button type="success" plain icon="Upload">上传线下安全考试</el-button></file-upload
>
</el-tooltip> </el-tooltip>
</el-col> </el-col>
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="primary" plain icon="Download" :disabled="single" <el-button type="primary" plain icon="Download" :disabled="single" @click="handleDownload()">批量下载试卷</el-button>
@click="handleDownload()">批量下载试卷</el-button>
</el-col> </el-col>
<right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar> <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
</el-row> </el-row>
</template> </template>
<el-table v-loading="loading" :data="questionUserAnswerList" @selection-change="handleSelectionChange"> <el-table v-loading="loading" :data="questionUserAnswerList" @selection-change="handleSelectionChange">
<el-table-column type="selection" width="55" align="center" /> <el-table-column type="index" label="序号" width="80" align="center" />
<el-table-column label="主键id" align="center" prop="id" v-if="false" />
<el-table-column label="姓名" align="center" prop="userName" /> <el-table-column label="姓名" align="center" prop="userName" />
<el-table-column label="及格线/总分" align="center" prop="pass" /> <el-table-column label="及格线/总分" align="center" prop="pass" />
<el-table-column label="得分" align="center" prop="score" /> <el-table-column label="得分" align="center" prop="score" />
@ -59,21 +62,19 @@
<dict-tag :options="user_exam_type" :value="scope.row.examType" /> <dict-tag :options="user_exam_type" :value="scope.row.examType" />
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="操作" align="center" class-name="small-padding fixed-width"> <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
<template #default="scope"> <template #default="scope">
<el-link type="primary" :underline="false" :href="scope.row.fileUrl[0]" target="_blank"> <!-- <el-link type="primary" :underline="false" :href="scope.row.path[0]" target="_blank">
<el-button link type="primary" icon="View">预览试卷</el-button> <el-button link type="primary" icon="View">预览试卷</el-button>
</el-link> </el-link> -->
<el-button link type="primary" icon="Download" @click="downloadOssOne(scope.row)" v-hasPermi="['system:oss:download']"
<el-button link type="primary" icon="Download" @click="downloadOssOne(scope.row)" >下载试卷</el-button
v-hasPermi="['system:oss:download']">下载试卷</el-button> >
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" <pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
v-model:limit="queryParams.pageSize" @pagination="getList" />
</el-card> </el-card>
</div> </div>
</template> </template>
@ -196,35 +197,15 @@ const handleSelectionChange = (selection: QuestionUserAnswerVO[]) => {
multiple.value = !selection.length; multiple.value = !selection.length;
}; };
/** 修改按钮操作 */
// const handleUpdate = async (row?: QuestionUserAnswerVO) => {
// reset();
// const _id = row?.id || ids.value[0];
// const res = await getQuestionUserAnswer(_id);
// Object.assign(form.value, res.data);
// dialog.visible = true;
// dialog.title = '修改用户试卷存储';
// };
/** 批量下载按钮操作 */ /** 批量下载按钮操作 */
const handleDownload = async () => { const handleDownload = async () => {
const _ids = ids.value; const _ids = ids.value;
await downLoadOss({ idList: _ids }, '/safety/questionUserAnswer/exportFile', '安全考试.zip'); await downLoadOss({ idList: _ids }, '/safety/wgzQuestionSave/exportFile', '安全考试.zip');
}; };
/** 下载单个按钮操作 */ /** 下载单个按钮操作 */
const downloadOssOne = async (row?: QuestionUserAnswerVO) => { const downloadOssOne = async (row?: QuestionUserAnswerVO) => {
await download.oss(row?.file); await download.oss(row?.file);
}; };
// const fileWatch = watch(
// () => filePath.value,
// (nid, oid) => {
// uploadQuestionUserAnswer({ file: filePath.value, projectId: currentProject.value?.id }).then((res) => {
// console.log(res);
// });
// }
// );
//监听项目id刷新数据 //监听项目id刷新数据
const listeningProject = watch( const listeningProject = watch(
() => currentProject.value?.id, () => currentProject.value?.id,

View File

@ -22,38 +22,21 @@
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['safety:questionsCategory:add']">新增</el-button> <el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['safety:questionsCategory:add']">新增</el-button>
</el-col> </el-col>
<el-col :span="1.5">
<el-button type="success" plain icon="Edit" :disabled="single" @click="handleUpdate()" v-hasPermi="['safety:questionsCategory:edit']"
>修改</el-button
>
</el-col>
<el-col :span="1.5">
<el-button type="danger" plain icon="Delete" :disabled="multiple" @click="handleDelete()" v-hasPermi="['safety:questionsCategory:remove']"
>删除</el-button
>
</el-col>
<right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar> <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
</el-row> </el-row>
</template> </template>
<el-table v-loading="loading" :data="questionsCategoryList" @selection-change="handleSelectionChange"> <el-table v-loading="loading" :data="questionsCategoryList">
<el-table-column type="selection" width="55" align="center" />
<el-table-column label="序号" align="center" type="index" width="100" /> <el-table-column label="序号" align="center" type="index" width="100" />
<el-table-column label="题库类别" align="center" prop="categoryName" /> <el-table-column label="题库类别" align="center" prop="categoryName" />
<el-table-column label="操作" align="center" class-name="small-padding fixed-width"> <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
<template #default="scope"> <template #default="scope">
<el-tooltip content="修改" placement="top"> <el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)" v-hasPermi="['safety:questionsCategory:edit']"
<el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)" v-hasPermi="['safety:questionsCategory:edit']"></el-button> >修改</el-button
</el-tooltip> >
<el-tooltip content="删除" placement="top"> <el-button link type="primary" icon="Delete" @click="handleDelete(scope.row)" v-hasPermi="['safety:questionsCategory:remove']"
<el-button >删除</el-button
link >
type="primary"
icon="Delete"
@click="handleDelete(scope.row)"
v-hasPermi="['safety:questionsCategory:remove']"
></el-button>
</el-tooltip>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>

View File

@ -1,309 +1,220 @@
<template> <template>
<div class="p-2"> <div class="container p-4 md:p-6">
<transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave"> <el-card shadow="hover" class="config-card transition-all duration-300 hover:shadow-lg">
<div v-show="showSearch" class="mb-[10px]"> <div class="card-header mb-4">
<el-card shadow="hover"> <h2 class="text-xl font-semibold text-gray-800">题库配置</h2>
<el-form ref="queryFormRef" :model="queryParams" :inline="true"> <p class="text-gray-500 text-sm">设置各类题型数量分值及考试参数</p>
<el-form-item label="单选题" prop="singleChoice">
<el-input v-model="queryParams.singleChoice" placeholder="请输入单选题" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="单选分数" prop="singleScore">
<el-input v-model="queryParams.singleScore" placeholder="请输入单选分数" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="多选题" prop="multipleChoice">
<el-input v-model="queryParams.multipleChoice" placeholder="请输入多选题" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="多选分数" prop="multipleScore">
<el-input v-model="queryParams.multipleScore" placeholder="请输入多选分数" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="判断题" prop="estimate">
<el-input v-model="queryParams.estimate" placeholder="请输入判断题" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="判断分数" prop="estimateScore">
<el-input v-model="queryParams.estimateScore" placeholder="请输入判断分数" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="满分" prop="fullMark">
<el-input v-model="queryParams.fullMark" placeholder="请输入满分" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="及格线" prop="passScore">
<el-input v-model="queryParams.passScore" placeholder="请输入及格线" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="答题最大时间" prop="answerTime">
<el-input v-model="queryParams.answerTime" placeholder="请输入答题最大时间" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item>
<el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
<el-button icon="Refresh" @click="resetQuery">重置</el-button>
</el-form-item>
</el-form>
</el-card>
</div> </div>
</transition>
<el-card shadow="never"> <el-form :model="form" label-width="120px" ref="formRef" class="config-form">
<template #header> <el-row :gutter="20">
<el-row :gutter="10" class="mb8"> <!-- 单选题设置 -->
<el-col :span="1.5"> <el-col :span="12" class="form-item-col">
<el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['safety:questionsConfig:add']"> 新增 </el-button> <el-form-item label="单选题(道)" :rules="[{ required: true, message: '单选题数量不能为空', trigger: 'blur' }]">
<el-input type="number" v-model="form.singleChoice" min="0" class="transition-all" placeholder="请输入数量" />
</el-form-item>
</el-col> </el-col>
<el-col :span="1.5">
<el-button type="success" plain icon="Edit" :disabled="single" @click="handleUpdate()" v-hasPermi="['safety:questionsConfig:edit']" <el-col :span="12" class="form-item-col">
>修改 <el-form-item label="单选分数" :rules="[{ required: true, message: '单选分数不能为空', trigger: 'blur' }]">
</el-button> <el-input type="number" v-model="form.singleScore" min="0" class="transition-all" placeholder="每题分数" />
</el-form-item>
</el-col> </el-col>
<el-col :span="1.5">
<el-button type="danger" plain icon="Delete" :disabled="multiple" @click="handleDelete()" v-hasPermi="['safety:questionsConfig:remove']" <!-- 多选题设置 -->
>删除 <el-col :span="12" class="form-item-col">
</el-button> <el-form-item label="多选题(道)" :rules="[{ required: true, message: '多选题数量不能为空', trigger: 'blur' }]">
<el-input type="number" v-model="form.multipleChoice" min="0" class="transition-all" placeholder="请输入数量" />
</el-form-item>
</el-col> </el-col>
<el-col :span="1.5">
<el-button type="warning" plain icon="Download" @click="handleExport" v-hasPermi="['safety:questionsConfig:export']">导出 </el-button> <el-col :span="12" class="form-item-col">
<el-form-item label="多选分数" :rules="[{ required: true, message: '多选分数不能为空', trigger: 'blur' }]">
<el-input type="number" v-model="form.multipleScore" min="0" class="transition-all" placeholder="每题分数" />
</el-form-item>
</el-col>
<!-- 判断题设置 -->
<el-col :span="12" class="form-item-col">
<el-form-item label="判断题(道)" :rules="[{ required: true, message: '判断题数量不能为空', trigger: 'blur' }]">
<el-input type="number" v-model="form.estimate" min="0" class="transition-all" placeholder="请输入数量" />
</el-form-item>
</el-col>
<el-col :span="12" class="form-item-col">
<el-form-item label="判断分数" :rules="[{ required: true, message: '判断分数不能为空', trigger: 'blur' }]">
<el-input type="number" v-model="form.estimateScore" min="0" class="transition-all" placeholder="每题分数" />
</el-form-item>
</el-col>
<!-- 分数设置 -->
<el-col :span="12" class="form-item-col">
<el-form-item label="及格分数" :rules="[{ required: true, message: '及格分数不能为空', trigger: 'blur' }]">
<el-input
type="number"
v-model="form.passingScore"
min="0"
:max="form.fullMark"
@change="handlePassingScoreChange"
class="transition-all"
placeholder="请输入及格分数"
/>
</el-form-item>
</el-col>
<el-col :span="12" class="form-item-col">
<el-form-item label="满分">
<el-input type="number" v-model="form.fullMark" disabled class="bg-gray-50" />
</el-form-item>
</el-col>
<!-- 时间设置 -->
<el-col :span="12" class="form-item-col">
<el-form-item label="答题时间(分钟)" :rules="[{ required: true, message: '答题时间不能为空', trigger: 'blur' }]">
<el-input type="number" v-model="form.answerTime" min="0" class="transition-all" placeholder="请输入时间" />
</el-form-item>
</el-col> </el-col>
<right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
</el-row> </el-row>
</template>
<el-table v-loading="loading" :data="questionsConfigList" @selection-change="handleSelectionChange">
<el-table-column type="selection" width="55" align="center" />
<el-table-column label="主键id" align="center" prop="id" v-if="true" />
<el-table-column label="单选题" align="center" prop="singleChoice" />
<el-table-column label="单选分数" align="center" prop="singleScore" />
<el-table-column label="多选题" align="center" prop="multipleChoice" />
<el-table-column label="多选分数" align="center" prop="multipleScore" />
<el-table-column label="判断题" align="center" prop="estimate" />
<el-table-column label="判断分数" align="center" prop="estimateScore" />
<el-table-column label="满分" align="center" prop="fullMark" />
<el-table-column label="及格线" align="center" prop="passScore" />
<el-table-column label="答题最大时间" align="center" prop="answerTime" />
<el-table-column label="操作" align="center" class-name="small-padding fixed-width">
<template #default="scope">
<el-tooltip content="修改" placement="top">
<el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)" v-hasPermi="['safety:questionsConfig:edit']"></el-button>
</el-tooltip>
<el-tooltip content="删除" placement="top">
<el-button
link
type="primary"
icon="Delete"
@click="handleDelete(scope.row)"
v-hasPermi="['safety:questionsConfig:remove']"
></el-button>
</el-tooltip>
</template>
</el-table-column>
</el-table>
<pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
</el-card>
<!-- 添加或修改题库配置对话框 -->
<el-dialog :title="dialog.title" v-model="dialog.visible" width="500px" append-to-body>
<el-form ref="questionsConfigFormRef" :model="form" :rules="rules" label-width="80px">
<el-form-item label="单选题" prop="singleChoice">
<el-input v-model="form.singleChoice" placeholder="请输入单选题" />
</el-form-item>
<el-form-item label="单选分数" prop="singleScore">
<el-input v-model="form.singleScore" placeholder="请输入单选分数" />
</el-form-item>
<el-form-item label="多选题" prop="multipleChoice">
<el-input v-model="form.multipleChoice" placeholder="请输入多选题" />
</el-form-item>
<el-form-item label="多选分数" prop="multipleScore">
<el-input v-model="form.multipleScore" placeholder="请输入多选分数" />
</el-form-item>
<el-form-item label="判断题" prop="estimate">
<el-input v-model="form.estimate" placeholder="请输入判断题" />
</el-form-item>
<el-form-item label="判断分数" prop="estimateScore">
<el-input v-model="form.estimateScore" placeholder="请输入判断分数" />
</el-form-item>
<el-form-item label="满分" prop="fullMark">
<el-input v-model="form.fullMark" placeholder="请输入满分" />
</el-form-item>
<el-form-item label="及格线" prop="passScore">
<el-input v-model="form.passScore" placeholder="请输入及格线" />
</el-form-item>
<el-form-item label="答题最大时间" prop="answerTime">
<el-input v-model="form.answerTime" placeholder="请输入答题最大时间" />
</el-form-item>
</el-form> </el-form>
<template #footer>
<div class="dialog-footer"> <div class="form-actions flex justify-center mt-6">
<el-button :loading="buttonLoading" type="primary" @click="submitForm"> </el-button> <el-button type="primary" @click="onSubmit" :loading="buttonLoading" class="px-8 transition-all hover:scale-105"> 保存配置 </el-button>
<el-button @click="cancel"> </el-button> </div>
</div> </el-card>
</template>
</el-dialog>
</div> </div>
</template> </template>
<script setup name="QuestionsConfig" lang="ts"> <script setup name="QuestionsConfig" lang="ts">
import { addQuestionsConfig, delQuestionsConfig, getQuestionsConfig, listQuestionsConfig, updateQuestionsConfig } from '@/api/safety/questionsConfig'; import { ref, reactive, computed, onMounted, getCurrentInstance } from 'vue';
import { QuestionsConfigForm, QuestionsConfigQuery, QuestionsConfigVO } from '@/api/safety/questionsConfig/types'; import { ElFormInstance, ComponentInternalInstance } from 'element-plus';
import { getQuestionsConfig, updateQuestionsConfig } from '@/api/safety/questionsConfig';
import { QuestionsConfigForm} from '@/api/safety/questionsConfig/types';
import { useUserStoreHook } from '@/store/modules/user'; import { useUserStoreHook } from '@/store/modules/user';
const { proxy } = getCurrentInstance() as ComponentInternalInstance; // 获取用户信息
// 获取用户 store
const userStore = useUserStoreHook(); const userStore = useUserStoreHook();
// 从 store 中获取项目列表和当前选中的项目
const currentProject = computed(() => userStore.selectedProject); const currentProject = computed(() => userStore.selectedProject);
const questionsConfigList = ref<QuestionsConfigVO[]>([]);
// 表单相关
const formRef = ref<ElFormInstance>();
const buttonLoading = ref(false); const buttonLoading = ref(false);
const loading = ref(true); const form = reactive<QuestionsConfigForm>({
const showSearch = ref(true);
const ids = ref<Array<string | number>>([]);
const single = ref(true);
const multiple = ref(true);
const total = ref(0);
const queryFormRef = ref<ElFormInstance>();
const questionsConfigFormRef = ref<ElFormInstance>();
const dialog = reactive<DialogOption>({
visible: false,
title: ''
});
const initFormData: QuestionsConfigForm = {
id: undefined, id: undefined,
projectId: currentProject.value?.id, projectId: currentProject.value?.id,
singleChoice: undefined, singleChoice: 0,
singleScore: undefined, singleScore: 0,
multipleChoice: undefined, multipleChoice: 0,
multipleScore: undefined, multipleScore: 0,
estimate: undefined, estimate: 0,
estimateScore: undefined, estimateScore: 0,
fullMark: undefined, fullMark: 0,
passScore: undefined, passScore: 0,
answerTime: undefined answerTime: 0,
}; passingScore: 0
const data = reactive<PageData<QuestionsConfigForm, QuestionsConfigQuery>>({
form: { ...initFormData },
queryParams: {
pageNum: 1,
pageSize: 10,
projectId: currentProject.value?.id,
singleChoice: undefined,
singleScore: undefined,
multipleChoice: undefined,
multipleScore: undefined,
estimate: undefined,
estimateScore: undefined,
fullMark: undefined,
passScore: undefined,
answerTime: undefined,
params: {}
},
rules: {
id: [{ required: true, message: '主键id不能为空', trigger: 'blur' }],
singleChoice: [{ required: true, message: '单选题不能为空', trigger: 'blur' }],
singleScore: [{ required: true, message: '单选分数不能为空', trigger: 'blur' }],
multipleChoice: [{ required: true, message: '多选题不能为空', trigger: 'blur' }],
multipleScore: [{ required: true, message: '多选分数不能为空', trigger: 'blur' }],
estimate: [{ required: true, message: '判断题不能为空', trigger: 'blur' }],
estimateScore: [{ required: true, message: '判断分数不能为空', trigger: 'blur' }],
fullMark: [{ required: true, message: '满分不能为空', trigger: 'blur' }],
passScore: [{ required: true, message: '及格线不能为空', trigger: 'blur' }],
answerTime: [{ required: true, message: '答题最大时间不能为空', trigger: 'blur' }]
}
}); });
const { queryParams, form, rules } = toRefs(data); // 计算满分
const calculateFullMark = () => {
form.fullMark =
(form.singleChoice || 0) * (form.singleScore || 0) +
(form.multipleChoice || 0) * (form.multipleScore || 0) +
(form.estimate || 0) * (form.estimateScore || 0);
/** 查询题库配置列表 */ // 确保及格分不超过满分
const getList = async () => { if (form.passingScore > form.fullMark) {
loading.value = true; form.passingScore = form.fullMark;
const res = await listQuestionsConfig(queryParams.value); }
questionsConfigList.value = res.rows;
total.value = res.total;
loading.value = false;
}; };
/** 取消按钮 */ // 监听分数变化,自动计算满分
const cancel = () => { ['singleChoice', 'singleScore', 'multipleChoice', 'multipleScore', 'estimate', 'estimateScore'].forEach((field) => {
reset(); watch(() => form[field], calculateFullMark);
dialog.visible = false; });
// 处理及格分变化
const handlePassingScoreChange = (val: number) => {
if (val >= form.fullMark) {
form.passingScore = form.fullMark;
}
}; };
/** 表单重置 */ // 获取配置详情
const reset = () => { const getDetail = () => {
form.value = { ...initFormData }; getQuestionsConfig(1).then((res: any) => {
questionsConfigFormRef.value?.resetFields(); if (res.code == 200 && res.data) {
}; Object.assign(form, res.data);
calculateFullMark(); // 确保满分计算正确
/** 搜索按钮操作 */
const handleQuery = () => {
queryParams.value.pageNum = 1;
getList();
};
/** 重置按钮操作 */
const resetQuery = () => {
queryFormRef.value?.resetFields();
handleQuery();
};
/** 多选框选中数据 */
const handleSelectionChange = (selection: QuestionsConfigVO[]) => {
ids.value = selection.map((item) => item.id);
single.value = selection.length != 1;
multiple.value = !selection.length;
};
/** 新增按钮操作 */
const handleAdd = () => {
reset();
dialog.visible = true;
dialog.title = '添加题库配置';
};
/** 修改按钮操作 */
const handleUpdate = async (row?: QuestionsConfigVO) => {
reset();
const _id = row?.id || ids.value[0];
const res = await getQuestionsConfig(_id);
Object.assign(form.value, res.data);
dialog.visible = true;
dialog.title = '修改题库配置';
};
/** 提交按钮 */
const submitForm = () => {
questionsConfigFormRef.value?.validate(async (valid: boolean) => {
if (valid) {
buttonLoading.value = true;
if (form.value.id) {
await updateQuestionsConfig(form.value).finally(() => (buttonLoading.value = false));
} else {
await addQuestionsConfig(form.value).finally(() => (buttonLoading.value = false));
}
proxy?.$modal.msgSuccess('操作成功');
dialog.visible = false;
await getList();
} }
}); });
}; };
/** 删除按钮操作 */ // 提交表单
const handleDelete = async (row?: QuestionsConfigVO) => { const onSubmit = () => {
const _ids = row?.id || ids.value; formRef.value?.validate(async (valid: boolean) => {
await proxy?.$modal.confirm('是否确认删除题库配置编号为"' + _ids + '"的数据项?').finally(() => (loading.value = false)); if (valid) {
await delQuestionsConfig(_ids); buttonLoading.value = true;
proxy?.$modal.msgSuccess('删除成功'); try {
await getList(); await updateQuestionsConfig(form);
}; // 显示成功消息假设使用element-plus的message组件
ElMessage.success('配置保存成功');
/** 导出按钮操作 */ getDetail(); // 重新获取最新配置
const handleExport = () => { } catch (error) {
proxy?.download( ElMessage.error('保存失败,请重试');
'safety/questionsConfig/export', } finally {
{ buttonLoading.value = false;
...queryParams.value }
}, }
`questionsConfig_${new Date().getTime()}.xlsx` });
);
}; };
// 页面加载时获取数据
onMounted(() => { onMounted(() => {
getList(); getDetail();
}); });
</script> </script>
<style lang="scss" scoped>
.container {
max-width: 800px;
margin: 0 auto;
padding-top: 20px;
}
.config-card {
border-radius: 12px;
overflow: hidden;
}
.card-header {
border-bottom: 1px solid #f0f0f0;
padding-bottom: 12px;
}
.form-item-col {
margin-bottom: 16px;
}
.form-actions {
padding: 16px;
border-top: 1px solid #f0f0f0;
margin-top: 20px;
}
// 输入框聚焦效果
::v-deep .el-input__wrapper:focus-within {
box-shadow: 0 0 0 2px rgba(48, 163, 255, 0.2);
}
// 响应式调整
@media (max-width: 768px) {
.el-col {
&:span-12 {
width: 100% !important;
}
}
.card-header {
text-align: center;
}
}
</style>

View File

@ -192,12 +192,12 @@ async function handleDeptChange(value: number | string) {
]; ];
} }
const submitForm = () => { const submitForm = (cb) => {
userFormRef.value?.validate(async (valid: boolean) => { userFormRef.value?.validate(async (valid: boolean) => {
if (valid) { if (valid) {
form.value.userId ? await api.updateUser(form.value) : await api.addUser(form.value); // form.value.userId ? await api.updateUser(form.value) : await api.addUser(form.value);
proxy?.$modal.msgSuccess('操作成功'); // proxy?.$modal.msgSuccess('操作成功');
proxy?.$emit('submit', false); cb();
} }
}); });
}; };
@ -243,7 +243,7 @@ const getInfoForm = () => {
onMounted(() => { onMounted(() => {
getDeptTree(); getDeptTree();
}); });
defineExpose({ open, getInfoForm }); defineExpose({ open, getInfoForm, submitForm });
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
.editInfo { .editInfo {

View File

@ -219,10 +219,22 @@ const removeProject = (projectId: number | string) => {
// 提交表单 // 提交表单
const submitForm = async () => { const submitForm = async () => {
// 整理项目角色数据 // 整理项目角色数据
console.log(selectedProjects.value);
if (selectedProjects.value.length == 0) { if (selectedProjects.value.length == 0) {
proxy?.$modal.msgWarning('请选择项目角色'); proxy?.$modal.msgWarning('请选择项目角色');
return; return;
} }
let flag = false;
selectedProjects.value.forEach((project) => {
if (project.appRoles.length > 0 || project.webRoles.length > 0) {
flag = true;
}
});
if (!flag) {
proxy?.$modal.msgWarning('请选择项目角色');
return;
}
form.value.projectRoles = selectedProjects.value.map((project) => ({ form.value.projectRoles = selectedProjects.value.map((project) => ({
projectId: project.id, projectId: project.id,
roleIds: [...new Set(project.webRoles), ...new Set(project.appRoles)] roleIds: [...new Set(project.webRoles), ...new Set(project.appRoles)]

View File

@ -154,15 +154,24 @@
</el-row> </el-row>
<!-- 添加或修改用户配置对话框 --> <!-- 添加或修改用户配置对话框 -->
<el-dialog draggable ref="formDialogRef" style="background: rgb(249,250,251);" v-model="dialog.visible" :title="dialog.title" width="1300px" append-to-body @close="closeDialog"> <el-dialog
draggable
ref="formDialogRef"
style="background: rgb(249, 250, 251)"
v-model="dialog.visible"
:title="dialog.title"
width="1300px"
append-to-body
@close="closeDialog"
>
<div class="boxDetial"> <div class="boxDetial">
<div class="tab_info"> <div class="tab_info">
<div class="tab_item" @click="onTab(1)" :class="{ active: type == 1 }"> <div class="tab_item" @click="onTab(1)" :class="{ active: type == 1 }">
<Avatar style="width: 1em; height: 1em; margin-right: 8px" :style="{color: type == 1 ? '#1890ff' : '#000'}" /> <Avatar style="width: 1em; height: 1em; margin-right: 8px" :style="{ color: type == 1 ? '#1890ff' : '#000' }" />
<span>基本资料</span> <span>基本资料</span>
</div> </div>
<div class="tab_item" @click="onTab(2)" :class="{ active: type == 2 }"> <div class="tab_item" @click="onTab(2)" :class="{ active: type == 2 }">
<Key style="width: 1em; height: 1em; margin-right: 8px" :style="{color: type == 2 ? '#1890ff' : '#000'}" /> <Key style="width: 1em; height: 1em; margin-right: 8px" :style="{ color: type == 2 ? '#1890ff' : '#000' }" />
<span>角色信息</span> <span>角色信息</span>
</div> </div>
</div> </div>
@ -698,13 +707,20 @@ const uploadCert = async () => {
proxy?.$modal.msgSuccess('上传证书目录成功'); proxy?.$modal.msgSuccess('上传证书目录成功');
}; };
const onTab = (val: number) => { const onTab = (val: number) => {
type.value = val;
if (val == 2) { if (val == 2) {
let obj = editInfoRef.value?.getInfoForm(); // 查看基本信息
form.value = obj; editInfoRef.value?.submitForm((res) => {
nextTick(() => { type.value = val;
roleInfoRef.value?.open(form.value, deptIdRole.value); let obj = editInfoRef.value?.getInfoForm();
form.value = obj;
console.log(obj);
nextTick(() => {
roleInfoRef.value?.open(form.value, deptIdRole.value);
});
}); });
} else {
type.value = val;
} }
}; };
</script> </script>