From 1220f5ca98b10b735a47c37a81fbfc554b01e2fe Mon Sep 17 00:00:00 2001 From: liriming <1343021927@qq.com> Date: 星期一, 20 一月 2025 14:41:35 +0800 Subject: [PATCH] Merge remote-tracking branch 'origin/master' --- src/views/bpm/processInstance/create/ProcessDefinitionDetail.vue | 298 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 files changed, 298 insertions(+), 0 deletions(-) diff --git a/src/views/bpm/processInstance/create/ProcessDefinitionDetail.vue b/src/views/bpm/processInstance/create/ProcessDefinitionDetail.vue new file mode 100644 index 0000000..7eaf0f4 --- /dev/null +++ b/src/views/bpm/processInstance/create/ProcessDefinitionDetail.vue @@ -0,0 +1,298 @@ +<template> + <ContentWrap :bodyStyle="{ padding: '10px 20px 0' }"> + <div class="processInstance-wrap-main"> + <el-scrollbar> + <div class="text-#878c93 h-15px">流程:{{ selectProcessDefinition.name }}</div> + <el-divider class="!my-8px" /> + + <!-- 中间主要内容 tab 栏 --> + <el-tabs v-model="activeTab"> + <!-- 表单信息 --> + <el-tab-pane label="表单填写" name="form"> + <div class="form-scroll-area" v-loading="processInstanceStartLoading"> + <el-scrollbar> + <el-row> + <el-col :span="17"> + <form-create + :rule="detailForm.rule" + v-model:api="fApi" + v-model="detailForm.value" + :option="detailForm.option" + @submit="submitForm" + /> + </el-col> + + <el-col :span="6" :offset="1"> + <!-- 流程时间线 --> + <ProcessInstanceTimeline + ref="timelineRef" + :activity-nodes="activityNodes" + :show-status-icon="false" + @select-user-confirm="selectUserConfirm" + /> + </el-col> + </el-row> + </el-scrollbar> + </div> + </el-tab-pane> + <!-- 流程图 --> + <el-tab-pane label="流程图" name="diagram"> + <div class="form-scroll-area"> + <!-- BPMN 流程图预览 --> + <ProcessInstanceBpmnViewer + :bpmn-xml="bpmnXML" + v-if="BpmModelType.BPMN === selectProcessDefinition.modelType" + /> + + <!-- Simple 流程图预览 --> + <ProcessInstanceSimpleViewer + :simple-json="simpleJson" + v-if="BpmModelType.SIMPLE === selectProcessDefinition.modelType" + /> + </div> + </el-tab-pane> + </el-tabs> + + <!-- 底部操作栏 --> + <div class="b-t-solid border-t-1px border-[var(--el-border-color)]"> + <!-- 操作栏按钮 --> + <div + v-if="activeTab === 'form'" + class="h-50px bottom-10 text-14px flex items-center color-#32373c dark:color-#fff font-bold btn-container" + > + <el-button plain type="success" @click="submitForm"> + <Icon icon="ep:select" /> 发起 + </el-button> + <el-button plain type="danger" @click="handleCancel"> + <Icon icon="ep:close" /> 取消 + </el-button> + </div> + </div> + </el-scrollbar> + </div> + </ContentWrap> +</template> +<script lang="ts" setup> +import { decodeFields, setConfAndFields2 } from '@/utils/formCreate' +import { BpmModelType } from '@/utils/constants' +import { + CandidateStrategy, + NodeId, + FieldPermissionType +} from '@/components/SimpleProcessDesignerV2/src/consts' +import ProcessInstanceBpmnViewer from '../detail/ProcessInstanceBpmnViewer.vue' +import ProcessInstanceSimpleViewer from '../detail/ProcessInstanceSimpleViewer.vue' +import ProcessInstanceTimeline from '../detail/ProcessInstanceTimeline.vue' +import type { ApiAttrs } from '@form-create/element-ui/types/config' +import { useTagsViewStore } from '@/store/modules/tagsView' +import * as ProcessInstanceApi from '@/api/bpm/processInstance' +import * as DefinitionApi from '@/api/bpm/definition' +import { ApprovalNodeInfo } from '@/api/bpm/processInstance' + +defineOptions({ name: 'ProcessDefinitionDetail' }) +const props = defineProps<{ + selectProcessDefinition: any +}>() +const emit = defineEmits(['cancel']) +const processInstanceStartLoading = ref(false) // 流程实例发起中 +const { push, currentRoute } = useRouter() // 路由 +const message = useMessage() // 消息弹窗 +const { delView } = useTagsViewStore() // 视图操作 + +const detailForm: any = ref({ + rule: [], + option: {}, + value: {} +}) // 流程表单详情 +const fApi = ref<ApiAttrs>() +// 指定审批人 +const startUserSelectTasks: any = ref([]) // 发起人需要选择审批人或抄送人的任务列表 +const startUserSelectAssignees = ref({}) // 发起人选择审批人的数据 +const bpmnXML: any = ref(null) // BPMN 数据 +const simpleJson = ref<string | undefined>() // Simple 设计器数据 json 格式 + +const activeTab = ref('form') // 当前的 Tab +const activityNodes = ref<ProcessInstanceApi.ApprovalNodeInfo[]>([]) // 审批节点信息 + +/** 设置表单信息、获取流程图数据 **/ +const initProcessInfo = async (row: any, formVariables?: any) => { + // 重置指定审批人 + startUserSelectTasks.value = [] + startUserSelectAssignees.value = {} + + // 情况一:流程表单 + if (row.formType == 10) { + // 设置表单 + // 注意:需要从 formVariables 中,移除不在 row.formFields 的值。 + // 原因是:后端返回的 formVariables 里面,会有一些非表单的信息。例如说,某个流程节点的审批人。 + // 这样,就可能导致一个流程被审批不通过后,重新发起时,会直接后端报错!!! + const allowedFields = decodeFields(row.formFields).map((fieldObj: any) => fieldObj.field) + for (const key in formVariables) { + if (!allowedFields.includes(key)) { + delete formVariables[key] + } + } + setConfAndFields2(detailForm, row.formConf, row.formFields, formVariables) + + await nextTick() + fApi.value?.btn.show(false) // 隐藏提交按钮 + + // 获取流程审批信息 + await getApprovalDetail(row) + + // 加载流程图 + const processDefinitionDetail = await DefinitionApi.getProcessDefinition(row.id) + if (processDefinitionDetail) { + bpmnXML.value = processDefinitionDetail.bpmnXml + simpleJson.value = processDefinitionDetail.simpleModel + } + // 情况二:业务表单 + } else if (row.formCustomCreatePath) { + await push({ + path: row.formCustomCreatePath + }) + // 这里暂时无需加载流程图,因为跳出到另外个 Tab; + } +} + +/** 获取审批详情 */ +const getApprovalDetail = async (row: any) => { + try { + // TODO 获取审批详情,设置 activityId 为发起人节点(为了获取字段权限。暂时只对 Simple 设计器有效) + const data = await ProcessInstanceApi.getApprovalDetail({ + processDefinitionId: row.id, + activityId: NodeId.START_USER_NODE_ID + }) + + if (!data) { + message.error('查询不到审批详情信息!') + return + } + + // 获取发起人自选的任务 + startUserSelectTasks.value = data.activityNodes?.filter( + (node: ApprovalNodeInfo) => CandidateStrategy.START_USER_SELECT === node.candidateStrategy + ) + if (startUserSelectTasks.value?.length > 0) { + for (const node of startUserSelectTasks.value) { + startUserSelectAssignees.value[node.id] = [] + } + } + + // 获取审批节点,显示 Timeline 的数据 + activityNodes.value = data.activityNodes + // 获取表单字段权限 + const formFieldsPermission = data.formFieldsPermission + // 设置表单字段权限 + if (formFieldsPermission) { + Object.keys(formFieldsPermission).forEach((item) => { + setFieldPermission(item, formFieldsPermission[item]) + }) + } + } finally { + } +} + +/** + * 设置表单权限 + */ +const setFieldPermission = (field: string, permission: string) => { + if (permission === FieldPermissionType.READ) { + //@ts-ignore + fApi.value?.disabled(true, field) + } + if (permission === FieldPermissionType.WRITE) { + //@ts-ignore + fApi.value?.disabled(false, field) + } + if (permission === FieldPermissionType.NONE) { + //@ts-ignore + fApi.value?.hidden(true, field) + } +} + +/** 提交按钮 */ +const submitForm = async () => { + if (!fApi.value || !props.selectProcessDefinition) { + return + } + // 流程表单校验 + await fApi.value.validate() + // 如果有指定审批人,需要校验 + if (startUserSelectTasks.value?.length > 0) { + for (const userTask of startUserSelectTasks.value) { + if ( + Array.isArray(startUserSelectAssignees.value[userTask.id]) && + startUserSelectAssignees.value[userTask.id].length === 0 + ) + return message.warning(`请选择${userTask.name}的候选人`) + } + } + + // 提交请求 + processInstanceStartLoading.value = true + try { + await ProcessInstanceApi.createProcessInstance({ + processDefinitionId: props.selectProcessDefinition.id, + variables: detailForm.value.value, + startUserSelectAssignees: startUserSelectAssignees.value + }) + // 提示 + message.success('发起流程成功') + // 跳转回去 + delView(unref(currentRoute)) + await push({ + name: 'BpmProcessInstanceMy' + }) + } finally { + processInstanceStartLoading.value = false + } +} + +/** 取消发起审批 */ +const handleCancel = () => { + emit('cancel') +} + +/** 选择发起人 */ +const selectUserConfirm = (id: string, userList: any[]) => { + startUserSelectAssignees.value[id] = userList?.map((item: any) => item.id) +} + +defineExpose({ initProcessInfo }) +</script> + +<style lang="scss" scoped> +$wrap-padding-height: 20px; +$wrap-margin-height: 15px; +$button-height: 51px; +$process-header-height: 105px; + +.processInstance-wrap-main { + height: calc( + 100vh - var(--top-tool-height) - var(--tags-view-height) - var(--app-footer-height) - 35px + ); + max-height: calc( + 100vh - var(--top-tool-height) - var(--tags-view-height) - var(--app-footer-height) - 35px + ); + overflow: auto; + + .form-scroll-area { + height: calc( + 100vh - var(--top-tool-height) - var(--tags-view-height) - var(--app-footer-height) - 35px - + $process-header-height - 40px + ); + max-height: calc( + 100vh - var(--top-tool-height) - var(--tags-view-height) - var(--app-footer-height) - 35px - + $process-header-height - 40px + ); + overflow: auto; + } +} + +.form-box { + :deep(.el-card) { + border: none; + } +} +</style> -- Gitblit v1.9.3