From 9259c2235e31708f954a3578bde3c6a7ab9753e8 Mon Sep 17 00:00:00 2001 From: houzhongjian <houzhongyi@126.com> Date: 星期一, 30 十二月 2024 15:51:40 +0800 Subject: [PATCH] 1、工作流相关组件更新 2、偶尔出现退出登录时路由报错的bug导致无法回到登录页面 3、全局配置文件修改,移除VITE_UPLOAD_URL配置等 --- src/views/bpm/model/index.vue | 507 ++++++++++++++++++------------------------------------- 1 files changed, 169 insertions(+), 338 deletions(-) diff --git a/src/views/bpm/model/index.vue b/src/views/bpm/model/index.vue index a20ea4e..bf43d29 100644 --- a/src/views/bpm/model/index.vue +++ b/src/views/bpm/model/index.vue @@ -1,275 +1,124 @@ <template> <ContentWrap> - <!-- 搜索工作栏 --> - <el-form - class="-mb-15px" - :model="queryParams" - ref="queryFormRef" - :inline="true" - label-width="68px" - > - <el-form-item label="流程标识" prop="key"> - <el-input - v-model="queryParams.key" - placeholder="请输入流程标识" - clearable - @keyup.enter="handleQuery" - class="!w-240px" - /> - </el-form-item> - <el-form-item label="流程名称" prop="name"> - <el-input - v-model="queryParams.name" - placeholder="请输入流程名称" - clearable - @keyup.enter="handleQuery" - class="!w-240px" - /> - </el-form-item> - <el-form-item label="流程分类" prop="category"> - <el-select - v-model="queryParams.category" - placeholder="请选择流程分类" - clearable - class="!w-240px" - > - <el-option - v-for="category in categoryList" - :key="category.code" - :label="category.name" - :value="category.code" - /> - </el-select> - </el-form-item> - <el-form-item> - <el-button @click="handleQuery"><Icon icon="ep:search" class="mr-5px" /> 搜索</el-button> - <el-button @click="resetQuery"><Icon icon="ep:refresh" class="mr-5px" /> 重置</el-button> - <el-button - type="primary" - plain - @click="openForm('create')" - v-hasPermi="['bpm:model:create']" - > - <Icon icon="ep:plus" class="mr-5px" /> 新建流程 - </el-button> - <el-button type="success" plain @click="openImportForm" v-hasPermi="['bpm:model:import']"> - <Icon icon="ep:upload" class="mr-5px" /> 导入流程 - </el-button> - </el-form-item> - </el-form> - </ContentWrap> + <div class="flex justify-between pl-20px items-center"> + <h3 class="font-extrabold">流程模型</h3> + <!-- 搜索工作栏 --> + <el-form + v-if="!isCategorySorting" + class="-mb-15px flex mr-10px" + :model="queryParams" + ref="queryFormRef" + :inline="true" + label-width="68px" + @submit.prevent + > + <el-form-item prop="name" class="ml-auto"> + <el-input + v-model="queryParams.name" + placeholder="搜索流程" + clearable + @keyup.enter="handleQuery" + class="!w-240px" + > + <template #prefix> + <Icon icon="ep:search" class="mx-10px" /> + </template> + </el-input> + </el-form-item> + <!-- 右上角:新建模型、更多操作 --> + <el-form-item> + <el-button type="primary" @click="openForm('create')" v-hasPermi="['bpm:model:create']"> + <Icon icon="ep:plus" class="mr-5px" /> 新建模型 + </el-button> + </el-form-item> + <el-form-item> + <el-dropdown @command="(command) => handleCommand(command)" placement="bottom-end"> + <el-button class="w-30px" plain> + <Icon icon="ep:setting" /> + </el-button> + <template #dropdown> + <el-dropdown-menu> + <el-dropdown-item command="handleCategoryAdd"> + <Icon icon="ep:circle-plus" :size="13" class="mr-5px" /> + 新建分类 + </el-dropdown-item> + <el-dropdown-item command="handleCategorySort"> + <Icon icon="fa:sort-amount-desc" :size="13" class="mr-5px" /> + 分类排序 + </el-dropdown-item> + </el-dropdown-menu> + </template> + </el-dropdown> + </el-form-item> + </el-form> + <div class="mr-20px" v-else> + <el-button @click="handleCategorySortCancel"> 取 消 </el-button> + <el-button type="primary" @click="handleCategorySortSubmit"> 保存排序 </el-button> + </div> + </div> - <!-- 列表 --> - <ContentWrap> - <el-table v-loading="loading" :data="list"> - <el-table-column label="流程标识" align="center" prop="key" width="200" /> - <el-table-column label="流程名称" align="center" prop="name" width="200"> - <template #default="scope"> - <el-button type="primary" link @click="handleBpmnDetail(scope.row)"> - <span>{{ scope.row.name }}</span> - </el-button> - </template> - </el-table-column> - <el-table-column label="流程图标" align="center" prop="icon" width="100"> - <template #default="scope"> - <el-image :src="scope.row.icon" class="w-32px h-32px" /> - </template> - </el-table-column> - <el-table-column label="流程分类" align="center" prop="categoryName" width="100" /> - <el-table-column label="表单信息" align="center" prop="formType" width="200"> - <template #default="scope"> - <el-button - v-if="scope.row.formType === 10" - type="primary" - link - @click="handleFormDetail(scope.row)" + <el-divider /> + + <!-- 按照分类,展示其所属的模型列表 --> + <div class="px-15px"> + <draggable + :disabled="!isCategorySorting" + v-model="categoryGroup" + item-key="id" + :animation="400" + > + <template #item="{ element }"> + <ContentWrap + class="rounded-lg transition-all duration-300 ease-in-out hover:shadow-xl" + v-loading="loading" + :body-style="{ padding: 0 }" + :key="element.id" > - <span>{{ scope.row.formName }}</span> - </el-button> - <el-button - v-else-if="scope.row.formType === 20" - type="primary" - link - @click="handleFormDetail(scope.row)" - > - <span>{{ scope.row.formCustomCreatePath }}</span> - </el-button> - <label v-else>暂无表单</label> - </template> - </el-table-column> - <el-table-column - label="创建时间" - align="center" - prop="createTime" - width="180" - :formatter="dateFormatter" - /> - <el-table-column label="最新部署的流程定义" align="center"> - <el-table-column - label="流程版本" - align="center" - prop="processDefinition.version" - width="100" - > - <template #default="scope"> - <el-tag v-if="scope.row.processDefinition"> - v{{ scope.row.processDefinition.version }} - </el-tag> - <el-tag v-else type="warning">未部署</el-tag> - </template> - </el-table-column> - <el-table-column - label="激活状态" - align="center" - prop="processDefinition.version" - width="85" - > - <template #default="scope"> - <el-switch - v-if="scope.row.processDefinition" - v-model="scope.row.processDefinition.suspensionState" - :active-value="1" - :inactive-value="2" - @change="handleChangeState(scope.row)" + <CategoryDraggableModel + :isCategorySorting="isCategorySorting" + :categoryInfo="element" + @success="getList" /> - </template> - </el-table-column> - <el-table-column label="部署时间" align="center" prop="deploymentTime" width="180"> - <template #default="scope"> - <span v-if="scope.row.processDefinition"> - {{ formatDate(scope.row.processDefinition.deploymentTime) }} - </span> - </template> - </el-table-column> - </el-table-column> - <el-table-column label="操作" align="center" width="240" fixed="right"> - <template #default="scope"> - <el-button - link - type="primary" - @click="openForm('update', scope.row.id)" - v-hasPermi="['bpm:model:update']" - > - 修改流程 - </el-button> - <el-button - link - type="primary" - @click="handleDesign(scope.row)" - v-hasPermi="['bpm:model:update']" - > - 设计流程 - </el-button> - <el-button - link - type="primary" - @click="handleDeploy(scope.row)" - v-hasPermi="['bpm:model:deploy']" - > - 发布流程 - </el-button> - <el-button - link - type="primary" - v-hasPermi="['bpm:process-definition:query']" - @click="handleDefinitionList(scope.row)" - > - 流程定义 - </el-button> - <el-button - link - type="danger" - @click="handleDelete(scope.row.id)" - v-hasPermi="['bpm:model:delete']" - > - 删除 - </el-button> + </ContentWrap> </template> - </el-table-column> - </el-table> - <!-- 分页 --> - <Pagination - :total="total" - v-model:page="queryParams.pageNo" - v-model:limit="queryParams.pageSize" - @pagination="getList" - /> + </draggable> + </div> </ContentWrap> <!-- 表单弹窗:添加/修改流程 --> <ModelForm ref="formRef" @success="getList" /> - - <!-- 表单弹窗:导入流程 --> - <ModelImportForm ref="importFormRef" @success="getList" /> - + <!-- 表单弹窗:添加分类 --> + <CategoryForm ref="categoryFormRef" @success="getList" /> <!-- 弹窗:表单详情 --> <Dialog title="表单详情" v-model="formDetailVisible" width="800"> <form-create :rule="formDetailPreview.rule" :option="formDetailPreview.option" /> </Dialog> - - <!-- 弹窗:流程模型图的预览 --> - <Dialog title="流程图" v-model="bpmnDetailVisible" width="800"> - <MyProcessViewer - key="designer" - v-model="bpmnXML" - :value="bpmnXML as any" - v-bind="bpmnControlForm" - :prefix="bpmnControlForm.prefix" - /> - </Dialog> </template> <script lang="ts" setup> -import { dateFormatter, formatDate } from '@/utils/formatTime' -import { MyProcessViewer } from '@/components/bpmnProcessDesigner/package' -import * as ModelApi from '@/api/bpm/model' -import * as FormApi from '@/api/bpm/form' -import ModelForm from './ModelForm.vue' -import ModelImportForm from '@/views/bpm/model/ModelImportForm.vue' -import { setConfAndFields2 } from '@/utils/formCreate' +import draggable from 'vuedraggable' import { CategoryApi } from '@/api/bpm/category' +import * as ModelApi from '@/api/bpm/model' +import ModelForm from './ModelForm.vue' +import CategoryForm from '../category/CategoryForm.vue' +import { cloneDeep } from 'lodash-es' +import CategoryDraggableModel from './CategoryDraggableModel.vue' defineOptions({ name: 'BpmModel' }) const message = useMessage() // 消息弹窗 -const { t } = useI18n() // 国际化 -const { push } = useRouter() // 路由 - const loading = ref(true) // 列表的加载中 -const total = ref(0) // 列表的总页数 -const list = ref([]) // 列表的数据 +const isCategorySorting = ref(false) // 是否 category 正处于排序状态 const queryParams = reactive({ - pageNo: 1, - pageSize: 10, - key: undefined, - name: undefined, - category: undefined + name: undefined }) const queryFormRef = ref() // 搜索的表单 -const categoryList = ref([]) // 流程分类列表 - -/** 查询列表 */ -const getList = async () => { - loading.value = true - try { - const data = await ModelApi.getModelPage(queryParams) - list.value = data.list - total.value = data.total - } finally { - loading.value = false - } -} +const categoryGroup: any = ref([]) // 按照 category 分组的数据 +const originalData: any = ref([]) // 原始数据 /** 搜索按钮操作 */ const handleQuery = () => { - queryParams.pageNo = 1 getList() -} - -/** 重置按钮操作 */ -const resetQuery = () => { - queryFormRef.value.resetFields() - handleQuery() } /** 添加/修改操作 */ @@ -278,113 +127,95 @@ formRef.value.open(type, id) } -/** 添加/修改操作 */ -const importFormRef = ref() -const openImportForm = () => { - importFormRef.value.open() -} - -/** 删除按钮操作 */ -const handleDelete = async (id: number) => { - try { - // 删除的二次确认 - await message.delConfirm() - // 发起删除 - await ModelApi.deleteModel(id) - message.success(t('common.delSuccess')) - // 刷新列表 - await getList() - } catch {} -} - -/** 更新状态操作 */ -const handleChangeState = async (row) => { - const state = row.processDefinition.suspensionState - try { - // 修改状态的二次确认 - const id = row.id - const statusState = state === 1 ? '激活' : '挂起' - const content = '是否确认' + statusState + '流程名字为"' + row.name + '"的数据项?' - await message.confirm(content) - // 发起修改状态 - await ModelApi.updateModelState(id, state) - // 刷新列表 - await getList() - } catch { - // 取消后,进行恢复按钮 - row.processDefinition.suspensionState = state === 1 ? 2 : 1 - } -} - -/** 设计流程 */ -const handleDesign = (row) => { - push({ - name: 'BpmModelEditor', - query: { - modelId: row.id - } - }) -} - -/** 发布流程 */ -const handleDeploy = async (row) => { - try { - // 删除的二次确认 - await message.confirm('是否部署该流程!!') - // 发起部署 - await ModelApi.deployModel(row.id) - message.success(t('部署成功')) - // 刷新列表 - await getList() - } catch {} -} - -/** 跳转到指定流程定义列表 */ -const handleDefinitionList = (row) => { - push({ - name: 'BpmProcessDefinition', - query: { - key: row.key - } - }) -} - /** 流程表单的详情按钮操作 */ const formDetailVisible = ref(false) const formDetailPreview = ref({ rule: [], option: {} }) -const handleFormDetail = async (row) => { - if (row.formType == 10) { - // 设置表单 - const data = await FormApi.getForm(row.formId) - setConfAndFields2(formDetailPreview, data.conf, data.fields) - // 弹窗打开 - formDetailVisible.value = true - } else { - await push({ - path: row.formCustomCreatePath - }) + +/** 右上角设置按钮 */ +const handleCommand = (command: string) => { + switch (command) { + case 'handleCategoryAdd': + handleCategoryAdd() + break + case 'handleCategorySort': + handleCategorySort() + break + default: + break } } -/** 流程图的详情按钮操作 */ -const bpmnDetailVisible = ref(false) -const bpmnXML = ref(null) -const bpmnControlForm = ref({ - prefix: 'flowable' -}) -const handleBpmnDetail = async (row) => { - const data = await ModelApi.getModel(row.id) - bpmnXML.value = data.bpmnXml || '' - bpmnDetailVisible.value = true +/** 新建分类 */ +const categoryFormRef = ref() +const handleCategoryAdd = () => { + categoryFormRef.value.open('create') +} + +/** 分类排序的提交 */ +const handleCategorySort = () => { + // 保存初始数据 + originalData.value = cloneDeep(categoryGroup.value) + isCategorySorting.value = true +} + +/** 分类排序的取消 */ +const handleCategorySortCancel = () => { + // 恢复初始数据 + categoryGroup.value = cloneDeep(originalData.value) + isCategorySorting.value = false +} + +/** 分类排序的保存 */ +const handleCategorySortSubmit = async () => { + // 保存排序 + const ids = categoryGroup.value.map((item: any) => item.id) + await CategoryApi.updateCategorySortBatch(ids) + // 刷新列表 + isCategorySorting.value = false + message.success('排序分类成功') + await getList() +} + +/** 加载数据 */ +const getList = async () => { + loading.value = true + try { + // 查询模型 + 分裂的列表 + const modelList = await ModelApi.getModelList(queryParams.name) + const categoryList = await CategoryApi.getCategorySimpleList() + // 按照 category 聚合 + // 注意:必须一次性赋值给 categoryGroup,否则每次操作后,列表会重新渲染,滚动条的位置会偏离!!! + categoryGroup.value = categoryList.map((category: any) => ({ + ...category, + modelList: modelList.filter((model: any) => model.categoryName == category.name) + })) + } finally { + loading.value = false + } } /** 初始化 **/ -onMounted(async () => { - await getList() - // 查询流程分类列表 - categoryList.value = await CategoryApi.getCategorySimpleList() +onMounted(() => { + getList() }) </script> + +<style lang="scss" scoped> +:deep() { + .el-table--fit .el-table__inner-wrapper:before { + height: 0; + } + .el-card { + border-radius: 8px; + } + .el-form--inline .el-form-item { + margin-right: 10px; + } + .el-divider--horizontal { + margin-top: 6px; + } +} +</style> -- Gitblit v1.9.3