!88 合并flowable工作流功能
* merge 合并dev * add 添加抄送查询 * add 添加我的已办 * add 添加抄送 * add 添加附件下载 * update 优化类型 * Merge remote-tracking branch 'origin/future/flowable' into future/flowable * fix 修复 流程设计器打包部署报错问题 * add 添加审批附件上传 * update 修复固定值未在xml显示问题 * update 修复跳转条件回显Object问题 * update 优化面板文件名称 * update 调整key * remove 移除旧设计器,添加xml保存 * update 增加任务面板提示 * update 优化xml预览和svg预览 * update 优化xml预览和svg预览 * update 优化模型设计 * update 优化设计器 * update 优化设计器 * update 优化设计器 * update 删除旧设计器 * update 删除console.log * add 添加模型接口 * update 优化预览xml和svg样式被修改问题 * update 优化属性面板,增加展开动画 * update 去除开发模式 * update 优化任务栏样式 * update 优化图标渲染样式 * update 增加BpmnFactory类型 * update 增加BpmnFactory * update 移除users和group * update 移除无用类型 * update 优化页面类型 * update 去除多余属性 * update 完善流程线 * update 增加复杂网关 * update 完善流程 * update 完善网关 * update 优化网关汉化 * update 优化过期时间选择 * update 支持多实例 * update 增加类容提示 * update 支持选择组 * update 新增角色api * update 优化roleSelect 选中未确定,再次打开还保留选中的问题 * update 优化userSelect 选中未确定,再次打开还保留选中的问题 * update 优化userSelect 选中未确定,再次打开还保留选中的问题 * update 去掉modeler store多余属性 bpmnModel * update 优化属性面板,当面板未选中时默认展示流程面板 * update 优化TaskPanel类型,去掉roles属性 * update 优化用户api * update 优化用户选择器 * update 优化执行监听器 * update 优化任务监听器 * update 优化usePanel方法 * update 选人优化 * update 增加扩展节点信息 * update 增加usePanel默认方法 * update 去除处理事件 * update 扩展flowable userinfo属性 * update 全局modeler 改为非响应式 * update 增加hooks方法 * update 修改命名 * update 修改面板formData来源 * update 重写用户任务面板选择逻辑 * update 重写用户任务面板选择逻辑 * update 修改用户选择组件获取数据逻辑 * update 修改枚举类型 * update 修改默认配置列 * update 增加修改节点方法 * update 调整预览窗口大小 * update 优化用户选择组件 返回值 * update 优化用户选择组件 * update 新增通过ids 获取用户信息 * update 重写task面板选人 未完成 * update 升级用户选择 支持多选配置 * update 升级bpmnjs依赖版本 * update 增加useDialog类型 * update 调整全局样式 * update 代码高亮设置 * update 优化领用,归还加载 * update 增加选择角色 * update 新增角色选择组件 * update 新增过期时间选择组件 * update 调整任务面板样式 * update 调整全局dialog header 增加分割线 * update 代码高亮设置 * update 调整面板位置 * update 封装用户选择组件 * update 移除所有的节点描述 * update 删除分类 * update 调整面板位置 * update 修改命名,增加自定义渲染 * update 修改命名,增加自定义渲染 * update 增加 Element类型定义 * update 调整样式 * update 移除bpmn panel依赖,升级bpmn.js依赖到最新,修改汉化包 * update 调整类型声明文件 * update 调整类型声明文件 * update 优化面板工具 * update 优化面板工具 * Merge remote-tracking branch 'origin/future/flowable' into future/flowable * update 优化面板工具 * Merge branch 'future/flowable' of https://gitee.com/JavaLionLi/plus-ui… * add 添加修改办理人 * update 优化面板工具 * update 初始化流程数据 * Merge remote-tracking branch 'origin/future/flowable' into future/flowable * add 流程设计面板 * update 调整初始化xml * add 任务面板 * add 新增bpmn.js * update 优化request请求类判断请求头方式 * update 流程定义预览 优化 * update 流程定义预览 优化 * update 去掉console.log * update 优化工作流代码 * fix 修复待办任务 重置查询条件失效问题 * add 增加待办任务 接口类型,优化页面 * add 增加vite 启动预编译css * fix 修复i18n无感刷新问题 * Merge branch 'dev' into future/flowable * update 调整选择请假事件 * Merge branch 'dev' into future/flowable * 同步dev代码 * Merge branch 'dev' into future/flowable * 合并dev * remove 设计器无用代码 调整请假查询 * update 调整请假申请 * update 移动请假表单包结构,调整设计器选择引用表单请求错误 * remove 移除动态表单 * update 调整流程办理 * Merge branch 'dev' into future/flowable * Merge branch 'dev' into future/flowable * Merge branch 'dev' into future/flowable * Merge branch 'dev' into future/flowable * 合并 * update 调整请假申请流程提交 * update 修改业务单据流程提交 * update 调整业务单据流程提交 * 优化代码 * 优化工作流代码缩进 * 项目格式化配置修改 * 调整代码缩进 * Merge branch 'ts' into future/flowable * add 添加动态表单提交流程 * add 添加动态表单单据 * add 新增流程定义与表单关联 * update 调整点击左侧部门查询人员,部门刷新问题 * update 调整按钮图标 * 调整错别字 * update 调整流程定义图片预览 * add 添加流程实例迁移版本 * fix 修复我的单据无法提交问题 * Merge branch 'ts' into future/flowable * remove 还原代码后端解决 * update 流程设计器中分配发起人变量错误,先移除 * fix 修复设计器无法编辑问题 * update 调整设计器请求头 * Merge branch 'ts' into future/flowable * Merge branch 'ts' into future/flowable * add 添加流程定义历史列表 * update 审批记录v2改为v3 * update 调整请假必填项 * add 添加请假申请示例,添加流程定义文件部署 * update 移除流程表单 formConfig 属性,表单配置信息都放一起便于使用。 * add 添加任务加签,减签 * update 调整流程作废 * update 优化流程状态 * add 添加任务驳回 * add 添加查询当前租户所有待办,已办任务 * add 增加审批意见 * add 添加流程办理弹窗确认组件 * add 添加任务作废理由 * update 调整流程实例,流程定义检索 * add 添加我的单据页面 * add 添加任务归还认领 * add 添加流程实例,流程定义分类查询 * add 添加模型分类查询 * add 添加流程分类 * Merge remote-tracking branch 'origin/future/flowable' into future/flowable * add 添加流程表单管理页面 * add 集成vForm动态表单组件 * update xml调整超出滚动 * add 添加已办列表 * add 添加单据状态 * update 优化流程实例删除 * fix 修复流程实例查询挂起状态错误 * update 调整流程实例挂起激活状态 * add 添加流程实例列表 * update 调整流程定义弹窗提示 * add 添加流程定义列表,添加流程图,xml预览,添加简单流程启动,办理 * 调整审批记录悬浮逻辑 * 删除无用代码 * add 添加节点悬浮信息 * 调整流程预览 * 调整流程追踪 * add 添加审批记录 * add 模型设计的types * update 修改排版 * add lang=ts * update 修改ele的废弃api * fix调整审批记录图片不显示问题 * add 添加任务待办,流程图 * 调整设计器关闭 * add 添加待办 * remove 删除无用代码 * update types * 添加模型token验证 * 隐藏设计器验证按钮,隐藏表单,案例,应用程序等 * 添加模型部署 * 添加画图接口token,优化画图接口 * 添加工作流模型新增,修改,查询,删除,画图工具
This commit is contained in:
		
							
								
								
									
										68
									
								
								src/components/BpmnDesign/panel/GatewayPanel.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								src/components/BpmnDesign/panel/GatewayPanel.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,68 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <el-collapse v-model="currentCollapseItem"> | ||||
|       <el-collapse-item name="1"> | ||||
|         <template #title> | ||||
|           <div class="collapse__title"> | ||||
|             <el-icon> | ||||
|               <InfoFilled /> | ||||
|             </el-icon> | ||||
|             常规 | ||||
|           </div> | ||||
|         </template> | ||||
|         <div> | ||||
|           <el-form ref="formRef" :model="formData" :rules="formRules" label-width="80px"> | ||||
|             <el-form-item prop="id" label="节点 ID"> | ||||
|               <el-input v-model="formData.id" @change="idChange"> </el-input> | ||||
|             </el-form-item> | ||||
|             <el-form-item prop="name" label="节点名称"> | ||||
|               <el-input v-model="formData.name" @change="nameChange"> </el-input> | ||||
|             </el-form-item> | ||||
|           </el-form> | ||||
|         </div> | ||||
|       </el-collapse-item> | ||||
|  | ||||
|       <el-collapse-item name="2"> | ||||
|         <template #title> | ||||
|           <div class="collapse__title"> | ||||
|             <el-icon> | ||||
|               <BellFilled /> | ||||
|             </el-icon> | ||||
|             执行监听器 | ||||
|           </div> | ||||
|         </template> | ||||
|         <div> | ||||
|           <ExecutionListener :element="element"></ExecutionListener> | ||||
|         </div> | ||||
|       </el-collapse-item> | ||||
|     </el-collapse> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import useParseElement from '@/components/BpmnDesign/hooks/useParseElement'; | ||||
| import usePanel from '@/components/BpmnDesign/hooks/usePanel'; | ||||
| import { Modeler, ModdleElement } from 'bpmn'; | ||||
| import { GatewayPanel } from 'bpmnDesign'; | ||||
| import ExecutionListener from '@/components/BpmnDesign/panel/property/ExecutionListener.vue'; | ||||
|  | ||||
| interface PropType { | ||||
|   element: ModdleElement; | ||||
| } | ||||
| const props = withDefaults(defineProps<PropType>(), {}); | ||||
| const { nameChange, idChange } = usePanel({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const { parseData } = useParseElement({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const currentCollapseItem = ref(['1', '2']); | ||||
| const formData = ref(parseData<GatewayPanel>()); | ||||
|  | ||||
| const formRules = ref<ElFormRules>({ | ||||
|   processCategory: [{ required: true, message: '请选择', trigger: 'blur' }], | ||||
|   id: [{ required: true, message: '请输入', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '请输入', trigger: 'blur' }] | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped></style> | ||||
							
								
								
									
										71
									
								
								src/components/BpmnDesign/panel/ProcessPanel.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								src/components/BpmnDesign/panel/ProcessPanel.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,71 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <el-collapse v-model="currentCollapseItem"> | ||||
|       <el-collapse-item name="1"> | ||||
|         <template #title> | ||||
|           <div class="collapse__title"> | ||||
|             <el-icon> | ||||
|               <InfoFilled /> | ||||
|             </el-icon> | ||||
|             常规 | ||||
|           </div> | ||||
|         </template> | ||||
|         <div> | ||||
|           <el-form ref="formRef" :model="formData" :rules="formRules" label-width="80px"> | ||||
|             <el-form-item label="流程标识" prop="id"> | ||||
|               <el-input v-model="formData.id" @change="idChange"></el-input> | ||||
|             </el-form-item> | ||||
|             <el-form-item label="流程名称" prop="name"> | ||||
|               <el-input v-model="formData.name" @change="nameChange"></el-input> | ||||
|             </el-form-item> | ||||
|           </el-form> | ||||
|         </div> | ||||
|       </el-collapse-item> | ||||
|  | ||||
|       <el-collapse-item name="2"> | ||||
|         <template #title> | ||||
|           <div class="collapse__title"> | ||||
|             <el-icon> | ||||
|               <BellFilled /> | ||||
|             </el-icon> | ||||
|             执行监听器 | ||||
|           </div> | ||||
|         </template> | ||||
|         <div> | ||||
|           <ExecutionListener :element="element"></ExecutionListener> | ||||
|         </div> | ||||
|       </el-collapse-item> | ||||
|     </el-collapse> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import ExecutionListener from './property/ExecutionListener.vue'; | ||||
| import useParseElement from '@/components/BpmnDesign/hooks/useParseElement'; | ||||
| import usePanel from '@/components/BpmnDesign/hooks/usePanel'; | ||||
| import { Modeler, ModdleElement } from 'bpmn'; | ||||
| import { ProcessPanel } from 'bpmnDesign'; | ||||
|  | ||||
| const { proxy } = getCurrentInstance() as ComponentInternalInstance; | ||||
|  | ||||
| interface PropType { | ||||
|   element: ModdleElement; | ||||
| } | ||||
| const props = withDefaults(defineProps<PropType>(), {}); | ||||
|  | ||||
| const { parseData } = useParseElement({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const { idChange, nameChange } = usePanel({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const currentCollapseItem = ref(['1', '2']); | ||||
| const formData = ref<ProcessPanel>(parseData<ProcessPanel>()); | ||||
|  | ||||
| const formRules = ref<ElFormRules>({ | ||||
|   id: [{ required: true, message: '请输入', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '请输入', trigger: 'blur' }] | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style scoped lang="scss"></style> | ||||
							
								
								
									
										94
									
								
								src/components/BpmnDesign/panel/SequenceFlowPanel.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								src/components/BpmnDesign/panel/SequenceFlowPanel.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,94 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <el-collapse v-model="currentCollapseItem"> | ||||
|       <el-collapse-item name="1"> | ||||
|         <template #title> | ||||
|           <div class="collapse__title"> | ||||
|             <el-icon> | ||||
|               <InfoFilled /> | ||||
|             </el-icon> | ||||
|             常规 | ||||
|           </div> | ||||
|         </template> | ||||
|         <div> | ||||
|           <el-form ref="formRef" :model="formData" :rules="formRules" label-width="90px"> | ||||
|             <el-form-item prop="id" label="节点 ID"> | ||||
|               <el-input v-model="formData.id" @change="idChange"> </el-input> | ||||
|             </el-form-item> | ||||
|             <el-form-item prop="name" label="节点名称"> | ||||
|               <el-input v-model="formData.name" @change="nameChange"> </el-input> | ||||
|             </el-form-item> | ||||
|             <el-form-item prop="conditionExpression" label="跳转条件"> | ||||
|               <el-input v-model="formData.conditionExpressionValue" @change="conditionExpressionChange"> </el-input> | ||||
|             </el-form-item> | ||||
|             <el-form-item prop="skipExpression" label="跳过表达式"> | ||||
|               <el-input v-model="formData.skipExpression" @change="skipExpressionChange"> </el-input> | ||||
|             </el-form-item> | ||||
|           </el-form> | ||||
|         </div> | ||||
|       </el-collapse-item> | ||||
|  | ||||
|       <el-collapse-item name="2"> | ||||
|         <template #title> | ||||
|           <div class="collapse__title"> | ||||
|             <el-icon> | ||||
|               <BellFilled /> | ||||
|             </el-icon> | ||||
|             执行监听器 | ||||
|           </div> | ||||
|         </template> | ||||
|         <div> | ||||
|           <ExecutionListener :element="element"></ExecutionListener> | ||||
|         </div> | ||||
|       </el-collapse-item> | ||||
|     </el-collapse> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import useParseElement from '@/components/BpmnDesign/hooks/useParseElement'; | ||||
| import usePanel from '@/components/BpmnDesign/hooks/usePanel'; | ||||
| import { Modeler, ModdleElement } from 'bpmn'; | ||||
| import { SequenceFlowPanel } from 'bpmnDesign'; | ||||
| import useModelerStore from '@/store/modules/modeler'; | ||||
|  | ||||
| interface PropType { | ||||
|   element: ModdleElement; | ||||
| } | ||||
| const props = withDefaults(defineProps<PropType>(), {}); | ||||
| const { nameChange, idChange, updateProperties } = usePanel({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const { parseData } = useParseElement({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const moddle = useModelerStore().getModdle(); | ||||
| const currentCollapseItem = ref(['1', '2']); | ||||
| const formData = ref(parseData<SequenceFlowPanel>()); | ||||
|  | ||||
| const formRules = ref<ElFormRules>({ | ||||
|   processCategory: [{ required: true, message: '请选择', trigger: 'blur' }], | ||||
|   id: [{ required: true, message: '请输入', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '请输入', trigger: 'blur' }] | ||||
| }); | ||||
|  | ||||
| const conditionExpressionChange = (val: string) => { | ||||
|   if (val) { | ||||
|     const newCondition = moddle.create('bpmn:FormalExpression', { body: val }); | ||||
|     updateProperties({ conditionExpression: newCondition }); | ||||
|   } else { | ||||
|     updateProperties({ conditionExpression: null }); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const skipExpressionChange = (val: string) => { | ||||
|   updateProperties({ 'flowable:skipExpression': val }); | ||||
| }; | ||||
|  | ||||
| onBeforeMount(() => { | ||||
|   if (formData.value.conditionExpression) { | ||||
|     formData.value.conditionExpressionValue = formData.value.conditionExpression.body; | ||||
|   } | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped></style> | ||||
							
								
								
									
										40
									
								
								src/components/BpmnDesign/panel/StartEndPanel.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/components/BpmnDesign/panel/StartEndPanel.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,40 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <el-form ref="formRef" :model="formData" :rules="formRules" label-width="80px"> | ||||
|       <el-form-item prop="id" label="节点 ID"> | ||||
|         <el-input v-model="formData.id" @change="idChange"> </el-input> | ||||
|       </el-form-item> | ||||
|       <el-form-item prop="name" label="节点名称"> | ||||
|         <el-input v-model="formData.name" @change="nameChange"> </el-input> | ||||
|       </el-form-item> | ||||
|       <el-form-item label="执行监听器" style="margin-bottom: 0"> </el-form-item> | ||||
|       <ExecutionListener :element="element"></ExecutionListener> | ||||
|     </el-form> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import useParseElement from '@/components/BpmnDesign/hooks/useParseElement'; | ||||
| import usePanel from '@/components/BpmnDesign/hooks/usePanel'; | ||||
| import { Modeler, ModdleElement } from 'bpmn'; | ||||
| import { StartEndPanel } from 'bpmnDesign'; | ||||
|  | ||||
| interface PropType { | ||||
|   element: ModdleElement; | ||||
| } | ||||
| const props = withDefaults(defineProps<PropType>(), {}); | ||||
| const { nameChange, idChange } = usePanel({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const { parseData } = useParseElement({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
|  | ||||
| const formData = ref(parseData<StartEndPanel>()); | ||||
|  | ||||
| const formRules = ref<ElFormRules>({ | ||||
|   id: [{ required: true, message: '请输入', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '请输入', trigger: 'blur' }] | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped></style> | ||||
							
								
								
									
										467
									
								
								src/components/BpmnDesign/panel/TaskPanel.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										467
									
								
								src/components/BpmnDesign/panel/TaskPanel.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,467 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <el-form ref="formRef" size="default" :model="formData" :rules="formRules" label-width="100px"> | ||||
|       <el-collapse v-model="currentCollapseItem"> | ||||
|         <el-collapse-item name="1"> | ||||
|           <template #title> | ||||
|             <div class="collapse__title"> | ||||
|               <el-icon> | ||||
|                 <InfoFilled /> | ||||
|               </el-icon> | ||||
|               常规 | ||||
|             </div> | ||||
|           </template> | ||||
|           <div> | ||||
|             <el-form-item prop="id" label="节点 ID"> | ||||
|               <el-input v-model="formData.id" @change="idChange"> </el-input> | ||||
|             </el-form-item> | ||||
|             <el-form-item prop="name" label="节点名称"> | ||||
|               <el-input v-model="formData.name" @change="nameChange"> </el-input> | ||||
|             </el-form-item> | ||||
|             <el-form-item v-if="showConfig.skipExpression" prop="skipExpression" label="跳过表达式"> | ||||
|               <el-input v-model="formData.skipExpression" @change="skipExpressionChange"> </el-input> | ||||
|             </el-form-item> | ||||
|           </div> | ||||
|         </el-collapse-item> | ||||
|         <el-collapse-item name="2"> | ||||
|           <template #title> | ||||
|             <div class="collapse__title"> | ||||
|               <el-icon> | ||||
|                 <Checked /> | ||||
|               </el-icon> | ||||
|               任务 | ||||
|             </div> | ||||
|           </template> | ||||
|           <div> | ||||
|             <el-form-item v-if="showConfig.async" prop="sync" label="是否异步"> | ||||
|               <el-switch v-model="formData.async" inline-prompt active-text="是" inactive-text="否" @change="syncChange" /> | ||||
|             </el-form-item> | ||||
|  | ||||
|             <el-tabs tab-position="left" class="demo-tabs" @tab-click="taskTabClick"> | ||||
|               <el-tab-pane label="身份存储"> | ||||
|                 <el-form-item label="分配人员"> | ||||
|                   <el-input v-model="assignee.userName" disabled> | ||||
|                     <template #append> | ||||
|                       <el-button icon="Search" type="primary" @click="openSingleUserSelect" /> | ||||
|                     </template> | ||||
|                   </el-input> | ||||
|                 </el-form-item> | ||||
|                 <el-form-item label="候选人员"> | ||||
|                   <el-badge :value="selectUserLength" :max="99"> | ||||
|                     <el-button size="small" type="primary" @click="openUserSelect">选择人员</el-button> | ||||
|                   </el-badge> | ||||
|                 </el-form-item> | ||||
|                 <el-form-item label="候选组"> | ||||
|                   <el-badge :value="selectRoleLength" :max="99"> | ||||
|                     <el-button size="small" type="primary" @click="openRoleSelect">选择组</el-button> | ||||
|                   </el-badge> | ||||
|                 </el-form-item> | ||||
|               </el-tab-pane> | ||||
|  | ||||
|               <el-tab-pane label="固定值"> | ||||
|                 <el-form-item prop="auditUserType" label="分配类型"> | ||||
|                   <el-select v-model="formData.allocationType"> | ||||
|                     <el-option v-for="item in AllocationTypeSelect" :key="item.id" :value="item.value" :label="item.label"> </el-option> | ||||
|                   </el-select> | ||||
|                 </el-form-item> | ||||
|                 <el-form-item v-if="formData.allocationType === AllocationTypeEnum.USER" label="分配人员"> | ||||
|                   <el-input v-model="formData.fixedAssignee" @change="fixedAssigneeChange"> | ||||
|                     <template #append> | ||||
|                       <el-button icon="Search" size="small" type="primary" @click="proxy.$modal.msgWarning('开发中。。。。。。')" /> | ||||
|                     </template> | ||||
|                   </el-input> | ||||
|                 </el-form-item> | ||||
|                 <div v-if="formData.allocationType === AllocationTypeEnum.CANDIDATE"> | ||||
|                   <el-form-item label="候选人员"> | ||||
|                     <el-badge :value="selectUserLength" :max="99"> | ||||
|                       <el-button size="small" type="primary" @click="openUserSelect">选择人员</el-button> | ||||
|                     </el-badge> | ||||
|                   </el-form-item> | ||||
|                   <el-form-item label="候选组"> | ||||
|                     <el-badge :value="selectRoleLength" :max="99"> | ||||
|                       <el-button size="small" type="primary" @click="openRoleSelect">选择组</el-button> | ||||
|                     </el-badge> | ||||
|                   </el-form-item> | ||||
|                 </div> | ||||
|                 <el-form-item v-if="formData.allocationType === AllocationTypeEnum.SPECIFY && showConfig.specifyDesc" style=""> | ||||
|                   <el-radio-group v-model="formData.specifyDesc" class="ml-4"> | ||||
|                     <el-radio v-for="item in SpecifyDesc" :key="item.id" :label="item.value" size="large">{{ item.label }}</el-radio> | ||||
|                   </el-radio-group> | ||||
|                 </el-form-item> | ||||
|               </el-tab-pane> | ||||
|             </el-tabs> | ||||
|  | ||||
|             <el-form-item v-if="showConfig.dueDate" prop="dueDate" label="到期时间"> | ||||
|               <el-input v-model="formData.dueDate" clearable @change="dueDateChange" @click="openDueDate"> | ||||
|                 <template #append> | ||||
|                   <el-button icon="Search" type="primary" @click="openDueDate" /> | ||||
|                 </template> | ||||
|               </el-input> | ||||
|             </el-form-item> | ||||
|             <el-form-item v-if="showConfig.priority" prop="priority" label="优先级"> | ||||
|               <el-input-number v-model="formData.priority" :min="0" @change="priorityChange"> </el-input-number> | ||||
|             </el-form-item> | ||||
|           </div> | ||||
|         </el-collapse-item> | ||||
|         <el-collapse-item name="3"> | ||||
|           <template #title> | ||||
|             <div class="collapse__title"> | ||||
|               <el-icon> | ||||
|                 <HelpFilled /> | ||||
|               </el-icon> | ||||
|               多实例 | ||||
|             </div> | ||||
|           </template> | ||||
|           <div> | ||||
|             <el-form-item label="多实例类型"> | ||||
|               <el-select v-model="formData.multiInstanceType" @change="multiInstanceTypeChange"> | ||||
|                 <el-option v-for="item in MultiInstanceType" :key="item.id" :value="item.value" :label="item.label"> </el-option> | ||||
|               </el-select> | ||||
|             </el-form-item> | ||||
|  | ||||
|             <div v-if="formData.multiInstanceType !== MultiInstanceTypeEnum.NONE"> | ||||
|               <el-form-item label="集合"> | ||||
|                 <template #label> | ||||
|                   <span> | ||||
|                     集合 | ||||
|                     <el-tooltip placement="top"> | ||||
|                       <el-icon><QuestionFilled /></el-icon> | ||||
|                       <template #content> | ||||
|                         属性会作为表达式进行解析。如果表达式解析为字符串而不是一个集合,<br /> | ||||
|                         不论是因为本身配置的就是静态字符串值,还是表达式计算结果为字符串,<br /> | ||||
|                         这个字符串都会被当做变量名,并从流程变量中用于获取实际的集合。 | ||||
|                       </template> | ||||
|                     </el-tooltip> | ||||
|                   </span> | ||||
|                 </template> | ||||
|                 <el-input v-model="formData.collection" @change="collectionChange"></el-input> | ||||
|               </el-form-item> | ||||
|               <el-form-item label="元素变量"> | ||||
|                 <template #label> | ||||
|                   <span> | ||||
|                     元素变量 | ||||
|                     <el-tooltip placement="top"> | ||||
|                       <el-icon><QuestionFilled /></el-icon> | ||||
|                       <template #content> | ||||
|                         每创建一个用户任务前,先以该元素变量为label,集合中的一项为value,<br /> | ||||
|                         创建(局部)流程变量,该局部流程变量被用于指派用户任务。<br /> | ||||
|                         一般来说,该字符串应与指定人员变量相同。 | ||||
|                       </template> | ||||
|                     </el-tooltip> | ||||
|                   </span> | ||||
|                 </template> | ||||
|                 <el-input v-model="formData.elementVariable" @change="elementVariableChange"> </el-input> | ||||
|               </el-form-item> | ||||
|               <el-form-item label="完成条件"> | ||||
|                 <template #label> | ||||
|                   <span> | ||||
|                     完成条件 | ||||
|                     <el-tooltip placement="top"> | ||||
|                       <el-icon><QuestionFilled /></el-icon> | ||||
|                       <template #content> | ||||
|                         多实例活动在所有实例都完成时结束,然而也可以指定一个表达式,在每个实例<br /> | ||||
|                         结束时进行计算。当表达式计算为true时,将销毁所有剩余的实例,并结束多实例<br /> | ||||
|                         活动,继续执行流程。例如 ${nrOfCompletedInstances/nrOfInstances >= 0.6 },<br /> | ||||
|                         表示当任务完成60%时,该节点就算完成 | ||||
|                       </template> | ||||
|                     </el-tooltip> | ||||
|                   </span> | ||||
|                 </template> | ||||
|                 <el-input v-model="formData.completionCondition" @change="completionConditionChange"> </el-input> | ||||
|               </el-form-item> | ||||
|             </div> | ||||
|           </div> | ||||
|         </el-collapse-item> | ||||
|         <el-collapse-item v-if="showConfig.taskListener" name="4"> | ||||
|           <template #title> | ||||
|             <div class="collapse__title"> | ||||
|               <el-icon> | ||||
|                 <BellFilled /> | ||||
|               </el-icon> | ||||
|               任务监听器 | ||||
|             </div> | ||||
|           </template> | ||||
|           <div> | ||||
|             <TaskListener v-if="showConfig.taskListener" :element="element"></TaskListener> | ||||
|           </div> | ||||
|         </el-collapse-item> | ||||
|         <el-collapse-item v-if="showConfig.executionListener" name="5"> | ||||
|           <template #title> | ||||
|             <div class="collapse__title"> | ||||
|               <el-icon> | ||||
|                 <BellFilled /> | ||||
|               </el-icon> | ||||
|               执行监听器 | ||||
|             </div> | ||||
|           </template> | ||||
|           <div> | ||||
|             <ExecutionListener v-if="showConfig.executionListener" :element="element"></ExecutionListener> | ||||
|           </div> | ||||
|         </el-collapse-item> | ||||
|  | ||||
|         <el-form-item v-if="showConfig.isForCompensation" prop="isForCompensation" label="是否为补偿"> | ||||
|           <el-switch v-model="formData.isForCompensation" inline-prompt active-text="是" inactive-text="否" /> | ||||
|         </el-form-item> | ||||
|         <el-form-item v-if="showConfig.triggerServiceTask" prop="triggerServiceTask" label="服务任务可触发"> | ||||
|           <el-switch v-model="formData.triggerServiceTask" inline-prompt active-text="是" inactive-text="否" /> | ||||
|         </el-form-item> | ||||
|         <el-form-item v-if="showConfig.autoStoreVariables" prop="autoStoreVariables" label="自动存储变量"> | ||||
|           <el-switch v-model="formData.autoStoreVariables" inline-prompt active-text="是" inactive-text="否" /> | ||||
|         </el-form-item> | ||||
|         <el-form-item v-if="showConfig.ruleVariablesInput" prop="skipExpression" label="输入变量"> | ||||
|           <el-input v-model="formData.ruleVariablesInput"> </el-input> | ||||
|         </el-form-item> | ||||
|         <el-form-item v-if="showConfig.exclude" prop="exclude" label="排除"> | ||||
|           <el-switch v-model="formData.exclude" inline-prompt active-text="是" inactive-text="否" /> | ||||
|         </el-form-item> | ||||
|         <el-form-item v-if="showConfig.class" prop="class" label="类"> | ||||
|           <el-input v-model="formData.class"> </el-input> | ||||
|         </el-form-item> | ||||
|       </el-collapse> | ||||
|     </el-form> | ||||
|     <UserSelect ref="userSelectRef" :data="formData.candidateUsers" @confirm-call-back="userSelectCallBack"></UserSelect> | ||||
|     <UserSelect ref="singleUserSelectRef" :data="formData.assignee" :multiple="false" @confirm-call-back="singleUserSelectCallBack"></UserSelect> | ||||
|     <RoleSelect ref="roleSelectRef" :data="formData.candidateGroups" @confirm-call-back="roleSelectCallBack"></RoleSelect> | ||||
|     <DueDate ref="dueDateRef" v-model="formData.dueDate" :data="formData.dueDate" @confirm-call-back="dueDateCallBack"></DueDate> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import useParseElement from '@/components/BpmnDesign/hooks/useParseElement'; | ||||
| import usePanel from '@/components/BpmnDesign/hooks/usePanel'; | ||||
| import UserSelect from '@/components/UserSelect'; | ||||
| import RoleSelect from '@/components/RoleSelect'; | ||||
| import DueDate from '@/components/BpmnDesign/panel/property/DueDate.vue'; | ||||
| import { ModdleElement } from 'bpmn'; | ||||
| import { TaskPanel } from 'bpmnDesign'; | ||||
| import { AllocationTypeEnum, MultiInstanceTypeEnum, SpecifyDescEnum } from '@/enums/bpmn/IndexEnums'; | ||||
| import { UserVO } from '@/api/system/user/types'; | ||||
| import { RoleVO } from '@/api/system/role/types'; | ||||
|  | ||||
| const { proxy } = getCurrentInstance() as ComponentInternalInstance; | ||||
|  | ||||
| interface PropType { | ||||
|   element: ModdleElement; | ||||
| } | ||||
| const props = withDefaults(defineProps<PropType>(), {}); | ||||
| const { showConfig, nameChange, idChange, updateProperties, getExtensionElements, createModdleElement } = usePanel({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const { parseData } = useParseElement({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
|  | ||||
| const initFormData = { | ||||
|   id: '', | ||||
|   name: '', | ||||
|   dueDate: '', | ||||
|   multiInstanceType: MultiInstanceTypeEnum.NONE, | ||||
|   allocationType: AllocationTypeEnum.USER, | ||||
|   specifyDesc: SpecifyDescEnum.SPECIFY_SINGLE | ||||
| }; | ||||
| const formData = ref({ ...initFormData, ...parseData<TaskPanel>() }); | ||||
| const assignee = ref<Partial<UserVO>>({ | ||||
|   userName: '' | ||||
| }); | ||||
| const currentCollapseItem = ref(['1', '2']); | ||||
| const userSelectRef = ref<InstanceType<typeof UserSelect>>(); | ||||
| const singleUserSelectRef = ref<InstanceType<typeof UserSelect>>(); | ||||
| const roleSelectRef = ref<InstanceType<typeof RoleSelect>>(); | ||||
| const dueDateRef = ref<InstanceType<typeof DueDate>>(); | ||||
|  | ||||
| const isMultiple = ref(true); | ||||
| const openUserSelect = () => { | ||||
|   userSelectRef.value.open(); | ||||
| }; | ||||
| const openSingleUserSelect = () => { | ||||
|   singleUserSelectRef.value.open(); | ||||
| }; | ||||
| const openRoleSelect = () => { | ||||
|   roleSelectRef.value.open(); | ||||
| }; | ||||
| const openDueDate = (e) => { | ||||
|   dueDateRef.value.openDialog(); | ||||
| }; | ||||
|  | ||||
| const singleUserSelectCallBack = (data: UserVO[]) => { | ||||
|   const user: UserVO = data.length !== 0 ? data[0] : undefined; | ||||
|   updateProperties({ 'flowable:assignee': user?.userId }); | ||||
|   assignee.value = user ? user : { userName: '' }; | ||||
|   formData.value.assignee = String(user?.userId); | ||||
|   let extensionElements = getExtensionElements(); | ||||
|   extensionElements.values = extensionElements.get('values').filter((item) => item.$type !== 'flowable:extAssignee'); | ||||
|   if (user) { | ||||
|     const extAssigneeElement = createModdleElement('flowable:extAssignee', { body: '' }, extensionElements); | ||||
|     extensionElements.get('values').push(extAssigneeElement); | ||||
|     extAssigneeElement.body = JSON.stringify({ userName: user.userName, userId: user.userId }); | ||||
|   } | ||||
|   if (extensionElements.values.length === 0) { | ||||
|     extensionElements = undefined; | ||||
|   } | ||||
|   updateProperties({ extensionElements: extensionElements }); | ||||
| }; | ||||
| const userSelectCallBack = (data: UserVO[]) => { | ||||
|   let extensionElements = getExtensionElements(); | ||||
|   extensionElements.values = extensionElements.values.filter((item) => item.$type !== 'flowable:extCandidateUsers'); | ||||
|   if (data.length === 0) { | ||||
|     formData.value.candidateUsers = undefined; | ||||
|     updateProperties({ 'flowable:candidateUsers': undefined }); | ||||
|   } else { | ||||
|     const userIds = data.map((item) => item.userId).join(','); | ||||
|     formData.value.candidateUsers = userIds; | ||||
|     updateProperties({ 'flowable:candidateUsers': userIds }); | ||||
|     const extCandidateUsersElement = createModdleElement('flowable:extCandidateUsers', { body: '' }, extensionElements); | ||||
|     extensionElements.values.push(extCandidateUsersElement); | ||||
|     const users = data.map((item) => { | ||||
|       return { | ||||
|         userId: item.userId, | ||||
|         userName: item.userName | ||||
|       }; | ||||
|     }); | ||||
|     extCandidateUsersElement.body = JSON.stringify(users); | ||||
|   } | ||||
|   if (extensionElements.values.length === 0) { | ||||
|     extensionElements = undefined; | ||||
|   } | ||||
|   updateProperties({ extensionElements: extensionElements }); | ||||
| }; | ||||
| const roleSelectCallBack = (data: RoleVO[]) => { | ||||
|   if (data.length === 0) { | ||||
|     formData.value.candidateGroups = ''; | ||||
|     updateProperties({ 'flowable:candidateGroups': undefined }); | ||||
|   } else { | ||||
|     const roleIds = data.map((item) => item.roleId).join(','); | ||||
|     formData.value.candidateGroups = roleIds; | ||||
|     updateProperties({ 'flowable:candidateGroups': roleIds }); | ||||
|   } | ||||
| }; | ||||
| const dueDateCallBack = (data: string) => { | ||||
|   updateProperties({ 'flowable:dueDate': data }); | ||||
| }; | ||||
|  | ||||
| const taskTabClick = (e) => { | ||||
|   formData.value.candidateGroups = ''; | ||||
|   formData.value.candidateUsers = ''; | ||||
|   formData.value.assignee = ''; | ||||
|   assignee.value = {}; | ||||
| }; | ||||
|  | ||||
| const syncChange = (newVal) => { | ||||
|   updateProperties({ 'flowable:async': newVal }); | ||||
| }; | ||||
| const skipExpressionChange = (newVal) => { | ||||
|   updateProperties({ 'flowable:skipExpression': newVal && newVal.length > 0 ? newVal : undefined }); | ||||
| }; | ||||
| const priorityChange = (newVal) => { | ||||
|   updateProperties({ 'flowable:priority': newVal }); | ||||
| }; | ||||
| const fixedAssigneeChange = (newVal) => { | ||||
|   updateProperties({ 'flowable:assignee': newVal && newVal.length > 0 ? newVal : undefined }); | ||||
| }; | ||||
| const multiInstanceTypeChange = (newVal) => { | ||||
|   if (newVal !== MultiInstanceTypeEnum.NONE) { | ||||
|     let loopCharacteristics = props.element.businessObject.get('loopCharacteristics'); | ||||
|     if (!loopCharacteristics) { | ||||
|       loopCharacteristics = createModdleElement('bpmn:MultiInstanceLoopCharacteristics', {}, props.element.businessObject); | ||||
|     } | ||||
|     loopCharacteristics.isSequential = newVal === MultiInstanceTypeEnum.SERIAL; | ||||
|     updateProperties({ loopCharacteristics: loopCharacteristics }); | ||||
|   } else { | ||||
|     updateProperties({ loopCharacteristics: undefined }); | ||||
|   } | ||||
| }; | ||||
| const collectionChange = (newVal) => { | ||||
|   let loopCharacteristics = props.element.businessObject.get('loopCharacteristics'); | ||||
|   if (!loopCharacteristics) { | ||||
|     loopCharacteristics = createModdleElement('bpmn:MultiInstanceLoopCharacteristics', {}, props.element.businessObject); | ||||
|   } | ||||
|   loopCharacteristics.collection = newVal && newVal.length > 0 ? newVal : undefined; | ||||
|   updateProperties({ loopCharacteristics: loopCharacteristics }); | ||||
| }; | ||||
| const elementVariableChange = (newVal) => { | ||||
|   let loopCharacteristics = props.element.businessObject.get('loopCharacteristics'); | ||||
|   if (!loopCharacteristics) { | ||||
|     loopCharacteristics = createModdleElement('bpmn:MultiInstanceLoopCharacteristics', {}, props.element.businessObject); | ||||
|   } | ||||
|   loopCharacteristics.elementVariable = newVal && newVal.length > 0 ? newVal : undefined; | ||||
|   updateProperties({ loopCharacteristics: loopCharacteristics }); | ||||
| }; | ||||
| const completionConditionChange = (newVal) => { | ||||
|   let loopCharacteristics = props.element.businessObject.get<ModdleElement>('loopCharacteristics'); | ||||
|   if (!loopCharacteristics) { | ||||
|     loopCharacteristics = createModdleElement('bpmn:MultiInstanceLoopCharacteristics', {}, props.element.businessObject); | ||||
|   } | ||||
|   if (newVal && newVal.length > 0) { | ||||
|     if (!loopCharacteristics.completionCondition) { | ||||
|       loopCharacteristics.completionCondition = createModdleElement('bpmn:Expression', { body: newVal }, loopCharacteristics); | ||||
|     } else { | ||||
|       loopCharacteristics.completionCondition.body = newVal; | ||||
|     } | ||||
|   } else { | ||||
|     loopCharacteristics.completionCondition = undefined; | ||||
|   } | ||||
|   updateProperties({ loopCharacteristics: loopCharacteristics }); | ||||
| }; | ||||
| const dueDateChange = (newVal) => { | ||||
|   updateProperties({ 'flowable:dueDate': newVal && newVal.length > 0 ? newVal : undefined }); | ||||
| }; | ||||
| const selectUserLength = computed(() => { | ||||
|   if (formData.value.candidateUsers) { | ||||
|     return formData.value.candidateUsers.split(',').length; | ||||
|   } else { | ||||
|     return 0; | ||||
|   } | ||||
| }); | ||||
| const selectRoleLength = computed(() => { | ||||
|   if (formData.value.candidateGroups) { | ||||
|     return formData.value.candidateGroups.split(',').length; | ||||
|   } else { | ||||
|     return 0; | ||||
|   } | ||||
| }); | ||||
|  | ||||
| onBeforeMount(() => { | ||||
|   const extensionElements = getExtensionElements(false); | ||||
|   if (extensionElements && extensionElements.get('values')) { | ||||
|     let extAssigneeElement = extensionElements.get('values').find((item) => item.$type === 'flowable:extAssignee'); | ||||
|     if (extAssigneeElement) { | ||||
|       assignee.value = JSON.parse(extAssigneeElement.body); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (formData.value.loopCharacteristics) { | ||||
|     const loopCharacteristics = formData.value.loopCharacteristics; | ||||
|     formData.value.collection = loopCharacteristics.collection || ''; | ||||
|     formData.value.elementVariable = loopCharacteristics.elementVariable || ''; | ||||
|     formData.value.completionCondition = loopCharacteristics.completionCondition?.body || ''; | ||||
|     formData.value.multiInstanceType = loopCharacteristics.isSequential ? MultiInstanceTypeEnum.SERIAL : MultiInstanceTypeEnum.PARALLEL; | ||||
|   } | ||||
|  | ||||
|   if (formData.value.assignee) { | ||||
|     formData.value.fixedAssignee = formData.value.assignee; | ||||
|   } | ||||
| }); | ||||
|  | ||||
| const formRules = ref<ElFormRules>({ | ||||
|   id: [{ required: true, message: '请输入', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '请输入', trigger: 'blur' }] | ||||
| }); | ||||
|  | ||||
| const AllocationTypeSelect = [ | ||||
|   { id: 'b9cdf970-dd91-47c0-819f-42a7010ca2a6', label: '指定人员', value: AllocationTypeEnum.USER }, | ||||
|   { id: '3f7ccbcd-c464-4602-bb9d-e96649d10585', label: '候选人员', value: AllocationTypeEnum.CANDIDATE }, | ||||
|   { id: 'c49065e0-7f2d-4c09-aedb-ab2d47d9a454', label: '发起人自己', value: AllocationTypeEnum.YOURSELF }, | ||||
|   { id: '6ef40a03-7e9a-4898-89b2-c88fe9064542', label: '发起人指定', value: AllocationTypeEnum.SPECIFY } | ||||
| ]; | ||||
| const SpecifyDesc = [ | ||||
|   { id: 'fa253b34-4335-458c-b1bc-b039e2a2b7a6', label: '指定一个人', value: 'specifySingle' }, | ||||
|   { id: '7365ff54-2e05-4312-9bfb-0b8edd779c5b', label: '指定多个人', value: 'specifyMultiple' } | ||||
| ]; | ||||
|  | ||||
| const MultiInstanceType = [ | ||||
|   { id: '373d4b81-a0d1-4eb8-8685-0d2fb1b468e2', label: '无', value: MultiInstanceTypeEnum.NONE }, | ||||
|   { id: 'b5acea7c-b7e5-46b0-8778-390db091bdab', label: '串行', value: MultiInstanceTypeEnum.SERIAL }, | ||||
|   { id: 'b4f0c683-1ccc-43c4-8380-e1b998986caf', label: '并行', value: MultiInstanceTypeEnum.PARALLEL } | ||||
| ]; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped></style> | ||||
							
								
								
									
										104
									
								
								src/components/BpmnDesign/panel/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								src/components/BpmnDesign/panel/index.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,104 @@ | ||||
| <template> | ||||
|   <div ref="propertyPanel"> | ||||
|     <div v-if="nodeName" class="node-name">{{ nodeName }}</div> | ||||
|     <component :is="component" v-if="element" :element="element" /> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts" name="PropertyPanel"> | ||||
| import { NodeName } from '../assets/lang/zh'; | ||||
| import TaskPanel from './TaskPanel.vue'; | ||||
| import ProcessPanel from './ProcessPanel.vue'; | ||||
| import StartEndPanel from './StartEndPanel.vue'; | ||||
| import GatewayPanel from './GatewayPanel.vue'; | ||||
| import SequenceFlowPanel from './SequenceFlowPanel.vue'; | ||||
| import { Modeler, ModdleElement } from 'bpmn'; | ||||
| const { proxy } = getCurrentInstance() as ComponentInternalInstance; | ||||
| interface propsType { | ||||
|   modeler: Modeler; | ||||
| } | ||||
| const props = withDefaults(defineProps<propsType>(), {}); | ||||
|  | ||||
| const element = ref<ModdleElement>(); | ||||
| const processElement = ref<ModdleElement>(); | ||||
|  | ||||
| const startEndType = ['bpmn:IntermediateThrowEvent', 'bpmn:StartEvent', 'bpmn:EndEvent']; | ||||
| const taskType = [ | ||||
|   'bpmn:UserTask', | ||||
|   'bpmn:Task', | ||||
|   'bpmn:SendTask', | ||||
|   'bpmn:ReceiveTask', | ||||
|   'bpmn:ManualTask', | ||||
|   'bpmn:BusinessRuleTask', | ||||
|   'bpmn:ServiceTask', | ||||
|   'bpmn:ScriptTask' | ||||
| ]; | ||||
| const sequenceType = ['bpmn:SequenceFlow']; | ||||
| const gatewayType = ['bpmn:InclusiveGateway', 'bpmn:ExclusiveGateway', 'bpmn:ParallelGateway', 'bpmn:EventBasedGateway', 'bpmn:ComplexGateway']; | ||||
| const processType = ['bpmn:Process']; | ||||
|  | ||||
| // 组件计算 | ||||
| const component = computed(() => { | ||||
|   if (!element.value) return null; | ||||
|   const type = element.value.type; | ||||
|   if (startEndType.includes(type)) return StartEndPanel; | ||||
|   if (taskType.includes(type)) return TaskPanel; | ||||
|   if (sequenceType.includes(type)) return SequenceFlowPanel; | ||||
|   if (gatewayType.includes(type)) return GatewayPanel; | ||||
|   if (processType.includes(type)) return ProcessPanel; | ||||
|   return proxy?.$modal.msgWarning('面板开发中....'); | ||||
| }); | ||||
|  | ||||
| const nodeName = computed(() => { | ||||
|   if (element.value) { | ||||
|     const bizObj = element.value.businessObject; | ||||
|     const type = bizObj?.eventDefinitions && bizObj?.eventDefinitions.length > 0 ? bizObj.eventDefinitions[0].$type : bizObj.$type; | ||||
|     return NodeName[type] || type; | ||||
|   } | ||||
| }); | ||||
|  | ||||
| const handleModeler = () => { | ||||
|   props.modeler.on('root.added', (e: any) => { | ||||
|     element.value = null; | ||||
|     if (e.element.type === 'bpmn:Process') { | ||||
|       nextTick(() => { | ||||
|         element.value = e.element; | ||||
|         processElement.value = e.element; | ||||
|       }); | ||||
|     } | ||||
|   }); | ||||
|   props.modeler.on('element.click', (e: any) => { | ||||
|     if (e.element.type === 'bpmn:Process') { | ||||
|       nextTick(() => { | ||||
|         element.value = e.element; | ||||
|         processElement.value = e.element; | ||||
|       }); | ||||
|     } | ||||
|   }); | ||||
|   props.modeler.on('selection.changed', (e: any) => { | ||||
|     // 先给null为了让vue刷新 | ||||
|     element.value = null; | ||||
|     const newElement = e.newSelection[0]; | ||||
|     if (newElement) { | ||||
|       nextTick(() => { | ||||
|         element.value = newElement; | ||||
|       }); | ||||
|     } else { | ||||
|       nextTick(() => { | ||||
|         element.value = processElement.value; | ||||
|       }); | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| onMounted(() => { | ||||
|   handleModeler(); | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style scoped lang="scss"> | ||||
| .node-name { | ||||
|   font-size: 16px; | ||||
|   font-weight: bold; | ||||
|   padding: 10px; | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										252
									
								
								src/components/BpmnDesign/panel/property/DueDate.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										252
									
								
								src/components/BpmnDesign/panel/property/DueDate.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,252 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <el-dialog v-model="visible" :title="title" width="600px" append-to-body> | ||||
|       <el-form label-width="100px"> | ||||
|         <el-form-item label="小时"> | ||||
|           <el-radio-group v-model="hourValue" @change="hourChange"> | ||||
|             <el-radio-button label="4" /> | ||||
|             <el-radio-button label="8" /> | ||||
|             <el-radio-button label="12" /> | ||||
|             <el-radio-button label="24" /> | ||||
|             <el-radio-button label="自定义" /> | ||||
|             <el-input-number v-show="hourValue === '自定义'" v-model="customHourValue" :min="1" @change="customHourValueChange"></el-input-number> | ||||
|           </el-radio-group> | ||||
|         </el-form-item> | ||||
|         <el-form-item label="天"> | ||||
|           <el-radio-group v-model="dayValue" @change="dayChange"> | ||||
|             <el-radio-button label="1" /> | ||||
|             <el-radio-button label="2" /> | ||||
|             <el-radio-button label="3" /> | ||||
|             <el-radio-button label="4" /> | ||||
|             <el-radio-button label="自定义" /> | ||||
|             <el-input-number v-show="dayValue === '自定义'" v-model="customDayValue" :min="1" @change="customDayValueChange"></el-input-number> | ||||
|           </el-radio-group> | ||||
|         </el-form-item> | ||||
|         <el-form-item label="周"> | ||||
|           <el-radio-group v-model="weekValue" @change="weekChange"> | ||||
|             <el-radio-button label="1" /> | ||||
|             <el-radio-button label="2" /> | ||||
|             <el-radio-button label="3" /> | ||||
|             <el-radio-button label="4" /> | ||||
|             <el-radio-button label="自定义" /> | ||||
|             <el-input-number v-show="weekValue === '自定义'" v-model="customWeekValue" :min="1" @change="customWeekValueChange"></el-input-number> | ||||
|           </el-radio-group> | ||||
|         </el-form-item> | ||||
|         <el-form-item label="月"> | ||||
|           <el-radio-group v-model="monthValue" @change="monthChange"> | ||||
|             <el-radio-button label="1" /> | ||||
|             <el-radio-button label="2" /> | ||||
|             <el-radio-button label="3" /> | ||||
|             <el-radio-button label="4" /> | ||||
|             <el-radio-button label="自定义" /> | ||||
|             <el-input-number v-show="monthValue === '自定义'" v-model="customMonthValue" :min="1" @change="customMonthValueChange"></el-input-number> | ||||
|           </el-radio-group> | ||||
|         </el-form-item> | ||||
|       </el-form> | ||||
|  | ||||
|       <template #footer> | ||||
|         <div> | ||||
|           <el-button @click="closeDialog">取消</el-button> | ||||
|           <el-button type="primary" @click="confirm">确定</el-button> | ||||
|         </div> | ||||
|       </template> | ||||
|     </el-dialog> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import useDialog from '@/hooks/useDialog'; | ||||
|  | ||||
| interface PropType { | ||||
|   modelValue?: string; | ||||
|   data?: string; | ||||
| } | ||||
| const prop = withDefaults(defineProps<PropType>(), { | ||||
|   modelValue: '', | ||||
|   data: '' | ||||
| }); | ||||
| const emit = defineEmits(['update:modelValue', 'confirmCallBack']); | ||||
|  | ||||
| const { title, visible, openDialog, closeDialog } = useDialog({ | ||||
|   title: '设置任务到期时间' | ||||
| }); | ||||
| const formValue = ref(); | ||||
| const valueType = ref(); | ||||
|  | ||||
| const hourValue = ref(''); | ||||
| const dayValue = ref(''); | ||||
| const weekValue = ref(''); | ||||
| const monthValue = ref(''); | ||||
|  | ||||
| const customHourValue = ref(1); | ||||
| const customDayValue = ref(1); | ||||
| const customWeekValue = ref(1); | ||||
| const customMonthValue = ref(1); | ||||
|  | ||||
| const hourValueConst = ['4', '8', '12', '24']; | ||||
| const dayAndWeekAndMonthValueConst = ['1', '2', '3', '4']; | ||||
|  | ||||
| const initValue = () => { | ||||
|   formValue.value = prop.data; | ||||
|   if (prop.data) { | ||||
|     const lastStr = prop.data.substring(prop.data.length - 1); | ||||
|     if (lastStr === 'H') { | ||||
|       const hourValueValue = prop.data.substring(2, prop.data.length - 1); | ||||
|       if (hourValueConst.includes(hourValueValue)) { | ||||
|         hourValue.value = hourValueValue; | ||||
|       } else { | ||||
|         hourValue.value = '自定义'; | ||||
|         customHourValue.value = Number(hourValueValue); | ||||
|       } | ||||
|     } | ||||
|     const dayAndWeekAndMonthValue = prop.data.substring(1, prop.data.length - 1); | ||||
|     if (lastStr === 'D') { | ||||
|       if (dayAndWeekAndMonthValueConst.includes(dayAndWeekAndMonthValue)) { | ||||
|         dayValue.value = dayAndWeekAndMonthValue; | ||||
|       } else { | ||||
|         dayValue.value = '自定义'; | ||||
|         customDayValue.value = Number(dayAndWeekAndMonthValue); | ||||
|       } | ||||
|     } | ||||
|     if (lastStr === 'W') { | ||||
|       if (dayAndWeekAndMonthValueConst.includes(dayAndWeekAndMonthValue)) { | ||||
|         weekValue.value = dayAndWeekAndMonthValue; | ||||
|       } else { | ||||
|         weekValue.value = '自定义'; | ||||
|         customWeekValue.value = Number(dayAndWeekAndMonthValue); | ||||
|       } | ||||
|     } | ||||
|     if (lastStr === 'M') { | ||||
|       if (dayAndWeekAndMonthValueConst.includes(dayAndWeekAndMonthValue)) { | ||||
|         monthValue.value = dayAndWeekAndMonthValue; | ||||
|       } else { | ||||
|         monthValue.value = '自定义'; | ||||
|         customMonthValue.value = Number(dayAndWeekAndMonthValue); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const confirm = () => { | ||||
|   emit('update:modelValue', formValue.value); | ||||
|   emit('confirmCallBack', formValue.value); | ||||
|   closeDialog(); | ||||
| }; | ||||
|  | ||||
| const customHourValueChange = (customHourValue) => { | ||||
|   formValue.value = `PT${customHourValue}H`; | ||||
|  | ||||
|   dayValue.value = ''; | ||||
|   weekValue.value = ''; | ||||
|   monthValue.value = ''; | ||||
|   customDayValue.value = 1; | ||||
|   customWeekValue.value = 1; | ||||
|   customMonthValue.value = 1; | ||||
| }; | ||||
| const customDayValueChange = (customDayValue) => { | ||||
|   formValue.value = `P${customDayValue}D`; | ||||
|   hourValue.value = ''; | ||||
|   weekValue.value = ''; | ||||
|   monthValue.value = ''; | ||||
|  | ||||
|   customHourValue.value = 1; | ||||
|   customWeekValue.value = 1; | ||||
|   customMonthValue.value = 1; | ||||
| }; | ||||
|  | ||||
| const customWeekValueChange = (customWeekValue) => { | ||||
|   formValue.value = `P${customWeekValue}W`; | ||||
|   hourValue.value = ''; | ||||
|   dayValue.value = ''; | ||||
|   monthValue.value = ''; | ||||
|  | ||||
|   customHourValue.value = 1; | ||||
|   customDayValue.value = 1; | ||||
|   customMonthValue.value = 1; | ||||
| }; | ||||
|  | ||||
| const customMonthValueChange = (customMonthValue) => { | ||||
|   formValue.value = `P${customMonthValue}M`; | ||||
|   hourValue.value = ''; | ||||
|   dayValue.value = ''; | ||||
|   weekValue.value = ''; | ||||
|  | ||||
|   customHourValue.value = 1; | ||||
|   customDayValue.value = 1; | ||||
|   customWeekValue.value = 1; | ||||
| }; | ||||
|  | ||||
| const hourChange = (hourValue) => { | ||||
|   if (hourValue === '自定义') { | ||||
|     formValue.value = `PT${customHourValue.value}H`; | ||||
|   } else { | ||||
|     formValue.value = `PT${hourValue}H`; | ||||
|   } | ||||
|  | ||||
|   dayValue.value = ''; | ||||
|   weekValue.value = ''; | ||||
|   monthValue.value = ''; | ||||
|   customDayValue.value = 1; | ||||
|   customWeekValue.value = 1; | ||||
|   customMonthValue.value = 1; | ||||
| }; | ||||
| const dayChange = (dayValue) => { | ||||
|   if (dayValue === '自定义') { | ||||
|     formValue.value = `P${customDayValue.value}D`; | ||||
|   } else { | ||||
|     formValue.value = `P${dayValue}D`; | ||||
|   } | ||||
|  | ||||
|   hourValue.value = ''; | ||||
|   weekValue.value = ''; | ||||
|   monthValue.value = ''; | ||||
|  | ||||
|   customHourValue.value = 1; | ||||
|   customWeekValue.value = 1; | ||||
|   customMonthValue.value = 1; | ||||
| }; | ||||
| const weekChange = (weekValue) => { | ||||
|   if (weekValue === '自定义') { | ||||
|     formValue.value = `P${customWeekValue.value}W`; | ||||
|   } else { | ||||
|     formValue.value = `P${weekValue}W`; | ||||
|   } | ||||
|  | ||||
|   hourValue.value = ''; | ||||
|   dayValue.value = ''; | ||||
|   monthValue.value = ''; | ||||
|  | ||||
|   customHourValue.value = 1; | ||||
|   customDayValue.value = 1; | ||||
|   customMonthValue.value = 1; | ||||
| }; | ||||
| const monthChange = (monthValue) => { | ||||
|   if (monthValue === '自定义') { | ||||
|     formValue.value = `P${customMonthValue.value}M`; | ||||
|   } else { | ||||
|     formValue.value = `P${monthValue}M`; | ||||
|   } | ||||
|  | ||||
|   hourValue.value = ''; | ||||
|   dayValue.value = ''; | ||||
|   weekValue.value = ''; | ||||
|  | ||||
|   customHourValue.value = 1; | ||||
|   customDayValue.value = 1; | ||||
|   customWeekValue.value = 1; | ||||
| }; | ||||
|  | ||||
| watch( | ||||
|   () => visible.value, | ||||
|   () => { | ||||
|     if (visible.value) { | ||||
|       initValue(); | ||||
|     } | ||||
|   } | ||||
| ); | ||||
|  | ||||
| defineExpose({ | ||||
|   openDialog, | ||||
|   closeDialog | ||||
| }); | ||||
| </script> | ||||
							
								
								
									
										305
									
								
								src/components/BpmnDesign/panel/property/ExecutionListener.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										305
									
								
								src/components/BpmnDesign/panel/property/ExecutionListener.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,305 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <vxe-toolbar> | ||||
|       <template #buttons> | ||||
|         <el-button type="primary" link size="small" @click="insertEvent">新增</el-button> | ||||
|         <el-button type="primary" link size="small" @click="removeSelectRowEvent">删除</el-button> | ||||
|       </template> | ||||
|     </vxe-toolbar> | ||||
|     <vxe-table | ||||
|       ref="tableRef" | ||||
|       size="mini" | ||||
|       height="100px" | ||||
|       border | ||||
|       show-overflow | ||||
|       keep-source | ||||
|       :data="tableData" | ||||
|       :menu-config="menuConfig" | ||||
|       @cell-dblclick="cellDBLClickEvent" | ||||
|       @menu-click="contextMenuClickEvent" | ||||
|     > | ||||
|       <vxe-column type="checkbox" width="40"></vxe-column> | ||||
|       <vxe-column type="seq" width="40"></vxe-column> | ||||
|       <vxe-column field="event" title="事件" min-width="100px"> | ||||
|         <template #default="slotParams"> | ||||
|           <span>{{ eventSelect.find((e) => e.value === slotParams.row.event)?.label }}</span> | ||||
|         </template> | ||||
|       </vxe-column> | ||||
|       <vxe-column field="type" title="类型" min-width="100px"> | ||||
|         <template #default="slotParams"> | ||||
|           <span>{{ typeSelect.find((e) => e.value === slotParams.row.type)?.label }}</span> | ||||
|         </template> | ||||
|       </vxe-column> | ||||
|       <vxe-column field="className" title="Java 类名" min-width="100px"> </vxe-column> | ||||
|     </vxe-table> | ||||
|  | ||||
|     <el-dialog | ||||
|       v-model="formDialog.visible.value" | ||||
|       :title="formDialog.title.value" | ||||
|       width="600px" | ||||
|       :close-on-click-modal="false" | ||||
|       :close-on-press-escape="false" | ||||
|       :show-close="false" | ||||
|       append-to-body | ||||
|     > | ||||
|       <el-form ref="formRef" :model="formData" :rules="tableRules" label-width="90px"> | ||||
|         <el-form-item label="事件" prop="event"> | ||||
|           <el-select v-model="formData.event"> | ||||
|             <el-option v-for="item in eventSelect" :key="item.id" :value="item.value" :label="item.label"></el-option> | ||||
|           </el-select> | ||||
|         </el-form-item> | ||||
|         <el-form-item label="类型" prop="type"> | ||||
|           <template #label> | ||||
|             <span> | ||||
|               类型 | ||||
|               <el-tooltip placement="top"> | ||||
|                 <el-icon><QuestionFilled /></el-icon> | ||||
|                 <template #content> | ||||
|                   类:示例 com.company.MyCustomListener,自定义类必须实现 org.flowable.engine.delegate.TaskListener 接口<br /> | ||||
|                   表达式:示例 ${myObject.callMethod(task, task.eventName)}<br /> | ||||
|                   委托表达式:示例 ${myListenerSpringBean} ,该 springBean 需要实现 org.flowable.engine.delegate.TaskListener 接口 | ||||
|                 </template> | ||||
|               </el-tooltip> | ||||
|             </span> | ||||
|           </template> | ||||
|           <el-select v-model="formData.type"> | ||||
|             <el-option v-for="item in typeSelect" :key="item.id" :value="item.value" :label="item.label"></el-option> | ||||
|           </el-select> | ||||
|         </el-form-item> | ||||
|         <el-form-item label="Java 类名" prop="className"> | ||||
|           <el-input v-model="formData.className" type="text"></el-input> | ||||
|         </el-form-item> | ||||
|       </el-form> | ||||
|       <el-tabs type="border-card"> | ||||
|         <el-tab-pane label="参数"> | ||||
|           <ListenerParam ref="listenerParamRef" :table-data="formData.params" /> | ||||
|         </el-tab-pane> | ||||
|       </el-tabs> | ||||
|       <template #footer> | ||||
|         <div class="dialog-footer"> | ||||
|           <el-button @click="formDialog.closeDialog">取 消</el-button> | ||||
|           <el-button type="primary" @click="submitEvent">确 定</el-button> | ||||
|         </div> | ||||
|       </template> | ||||
|     </el-dialog> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import ListenerParam from './ListenerParam.vue'; | ||||
| import { VxeTableEvents, VxeTableInstance, VxeTablePropTypes } from 'vxe-table'; | ||||
| import { ExecutionListenerVO } from 'bpmnDesign'; | ||||
| import { Moddle, Modeler, ModdleElement } from 'bpmn'; | ||||
|  | ||||
| import usePanel from '@/components/BpmnDesign/hooks/usePanel'; | ||||
| import useDialog from '@/hooks/useDialog'; | ||||
| import useModelerStore from '@/store/modules/modeler'; | ||||
|  | ||||
| const emit = defineEmits(['close']); | ||||
| interface PropType { | ||||
|   element: ModdleElement; | ||||
| } | ||||
| const props = withDefaults(defineProps<PropType>(), {}); | ||||
|  | ||||
| const { proxy } = getCurrentInstance() as ComponentInternalInstance; | ||||
|  | ||||
| const selectRow = ref<ExecutionListenerVO | null>(); | ||||
| const formDialog = useDialog({ | ||||
|   title: selectRow.value ? '编辑&保存' : '新增&保存' | ||||
| }); | ||||
|  | ||||
| const { showConfig, elementType, updateProperties } = usePanel({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const { getModdle } = useModelerStore(); | ||||
| const moddle = getModdle(); | ||||
|  | ||||
| const listenerParamRef = ref<InstanceType<typeof ListenerParam>>(); | ||||
| const tableRef = ref<VxeTableInstance<ExecutionListenerVO>>(); | ||||
| const formRef = ref<ElFormInstance>(); | ||||
|  | ||||
| const initData: ExecutionListenerVO = { | ||||
|   event: '', | ||||
|   type: '', | ||||
|   className: '', | ||||
|   params: [] | ||||
| }; | ||||
| const formData = ref<ExecutionListenerVO>({ ...initData }); | ||||
| const tableData = ref<ExecutionListenerVO[]>([]); | ||||
| const tableRules = ref<ElFormRules>({ | ||||
|   event: [{ required: true, message: '请选择', trigger: 'blur' }], | ||||
|   type: [{ required: true, message: '请选择', trigger: 'blur' }], | ||||
|   className: [{ required: true, message: '请输入', trigger: 'blur' }] | ||||
| }); | ||||
|  | ||||
| const submitEvent = async () => { | ||||
|   const error = await listenerParamRef.value.validate(); | ||||
|   await formRef.value.validate((validate) => { | ||||
|     if (validate && !error) { | ||||
|       const $table = tableRef.value; | ||||
|       if ($table) { | ||||
|         formData.value.params = listenerParamRef.value.getTableData(); | ||||
|         if (selectRow.value) { | ||||
|           Object.assign(selectRow.value, formData.value); | ||||
|         } else { | ||||
|           $table.insertAt({ ...formData.value }, -1); | ||||
|         } | ||||
|         updateElement(); | ||||
|         formDialog.closeDialog(); | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| const removeSelectRowEvent = async () => { | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     const selectCount = $table.getCheckboxRecords().length; | ||||
|     if (selectCount === 0) { | ||||
|       proxy?.$modal.msgWarning('请选择行'); | ||||
|     } else { | ||||
|       await $table.removeCheckboxRow(); | ||||
|       updateElement(); | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| const insertEvent = async () => { | ||||
|   Object.assign(formData.value, initData); | ||||
|   selectRow.value = null; | ||||
|   formDialog.openDialog(); | ||||
| }; | ||||
|  | ||||
| const editEvent = (row: ExecutionListenerVO) => { | ||||
|   Object.assign(formData.value, row); | ||||
|   selectRow.value = row; | ||||
|   formDialog.openDialog(); | ||||
| }; | ||||
|  | ||||
| const removeEvent = async (row: ExecutionListenerVO) => { | ||||
|   await proxy?.$modal.confirm('您确定要删除该数据?'); | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     await $table.remove(row); | ||||
|     updateElement(); | ||||
|   } | ||||
| }; | ||||
| const updateElement = () => { | ||||
|   const $table = tableRef.value; | ||||
|   const data = $table.getTableData().fullData; | ||||
|   if (data.length) { | ||||
|     let extensionElements = props.element.businessObject.get('extensionElements'); | ||||
|     if (!extensionElements) { | ||||
|       extensionElements = moddle.create('bpmn:ExtensionElements'); | ||||
|     } | ||||
|     // 清除旧值 | ||||
|     extensionElements.values = extensionElements.values?.filter((item) => item.$type !== 'flowable:ExecutionListener') ?? []; | ||||
|     data.forEach((item) => { | ||||
|       const executionListener = moddle.create('flowable:ExecutionListener'); | ||||
|       executionListener['event'] = item.event; | ||||
|       executionListener[item.type] = item.className; | ||||
|       if (item.params && item.params.length) { | ||||
|         item.params.forEach((field) => { | ||||
|           const fieldElement = moddle.create('flowable:Field'); | ||||
|           fieldElement['name'] = field.name; | ||||
|           fieldElement[field.type] = field.value; | ||||
|           executionListener.get('fields').push(fieldElement); | ||||
|         }); | ||||
|       } | ||||
|       extensionElements.get('values').push(executionListener); | ||||
|     }); | ||||
|     updateProperties({ extensionElements: extensionElements }); | ||||
|   } else { | ||||
|     const extensionElements = props.element.businessObject[`extensionElements`]; | ||||
|     if (extensionElements) { | ||||
|       extensionElements.values = extensionElements.values?.filter((item) => item.$type !== 'flowable:ExecutionListener') ?? []; | ||||
|     } | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const cellDBLClickEvent: VxeTableEvents.CellDblclick<ExecutionListenerVO> = ({ row }) => { | ||||
|   editEvent(row); | ||||
| }; | ||||
|  | ||||
| const menuConfig = reactive<VxeTablePropTypes.MenuConfig<ExecutionListenerVO>>({ | ||||
|   body: { | ||||
|     options: [ | ||||
|       [ | ||||
|         { code: 'edit', name: '编辑', prefixIcon: 'vxe-icon-edit', disabled: false }, | ||||
|         { code: 'remove', name: '删除', prefixIcon: 'vxe-icon-delete', disabled: false } | ||||
|       ] | ||||
|     ] | ||||
|   }, | ||||
|   visibleMethod({ options, column }) { | ||||
|     const isDisabled = !column; | ||||
|     options.forEach((list) => { | ||||
|       list.forEach((item) => { | ||||
|         item.disabled = isDisabled; | ||||
|       }); | ||||
|     }); | ||||
|     return true; | ||||
|   } | ||||
| }); | ||||
| const contextMenuClickEvent: VxeTableEvents.MenuClick<ExecutionListenerVO> = ({ menu, row, column }) => { | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     switch (menu.code) { | ||||
|       case 'edit': | ||||
|         editEvent(row); | ||||
|         break; | ||||
|       case 'remove': | ||||
|         removeEvent(row); | ||||
|         break; | ||||
|     } | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const initTableData = () => { | ||||
|   tableData.value = | ||||
|     props.element.businessObject.extensionElements?.values | ||||
|       .filter((item) => item.$type === 'flowable:ExecutionListener') | ||||
|       .map((item) => { | ||||
|         let type; | ||||
|         if ('class' in item) type = 'class'; | ||||
|         if ('expression' in item) type = 'expression'; | ||||
|         if ('delegateExpression' in item) type = 'delegateExpression'; | ||||
|         return { | ||||
|           event: item.event, | ||||
|           type: type, | ||||
|           className: item[type], | ||||
|           params: | ||||
|             item.fields?.map((field) => { | ||||
|               let fieldType; | ||||
|               if ('stringValue' in field) fieldType = 'stringValue'; | ||||
|               if ('expression' in field) fieldType = 'expression'; | ||||
|               return { | ||||
|                 name: field.name, | ||||
|                 type: fieldType, | ||||
|                 value: field[fieldType] | ||||
|               }; | ||||
|             }) ?? [] | ||||
|         }; | ||||
|       }) ?? []; | ||||
| }; | ||||
|  | ||||
| onMounted(() => { | ||||
|   initTableData(); | ||||
| }); | ||||
|  | ||||
| const typeSelect = [ | ||||
|   { id: '742fdeb7-23b4-416b-ac66-cd4ec8b901b7', label: '类', value: 'class' }, | ||||
|   { id: '660c9c46-8fae-4bae-91a0-0335420019dc', label: '表达式', value: 'expression' }, | ||||
|   { id: '4b8135ab-6bc3-4a0f-80be-22f58bc6c5fd', label: '委托表达式', value: 'delegateExpression' } | ||||
| ]; | ||||
| const eventSelect = [ | ||||
|   { id: 'e6e0a51a-2d5d-4dc4-b847-b5c14f43a6ab', label: 'start', value: 'start' }, | ||||
|   { id: '6da97c1e-15fc-4445-8943-75d09f49778e', label: 'end', value: 'end' }, | ||||
|   { id: '6a2cbcec-e026-4f11-bef7-fff0b5c871e2', label: 'take', value: 'take' } | ||||
| ]; | ||||
| </script> | ||||
|  | ||||
| <style scoped lang="scss"> | ||||
| .el-badge { | ||||
|   :deep(.el-badge__content) { | ||||
|     top: 10px; | ||||
|   } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										121
									
								
								src/components/BpmnDesign/panel/property/ListenerParam.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								src/components/BpmnDesign/panel/property/ListenerParam.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,121 @@ | ||||
| <template> | ||||
|   <vxe-toolbar> | ||||
|     <template #buttons> | ||||
|       <el-button icon="Plus" @click="insertRow">新增</el-button> | ||||
|     </template> | ||||
|   </vxe-toolbar> | ||||
|   <vxe-table | ||||
|     ref="tableRef" | ||||
|     :height="height" | ||||
|     border | ||||
|     show-overflow | ||||
|     keep-source | ||||
|     :data="tableData" | ||||
|     :edit-rules="tableRules" | ||||
|     :edit-config="{ trigger: 'click', mode: 'row', showStatus: true }" | ||||
|   > | ||||
|     <vxe-column type="seq" width="40"></vxe-column> | ||||
|     <vxe-column field="type" title="类型" :edit-render="{}"> | ||||
|       <template #default="slotParams"> | ||||
|         <span>{{ typeSelect.find((e) => e.value === slotParams.row.type)?.label }}</span> | ||||
|       </template> | ||||
|       <template #edit="slotParams"> | ||||
|         <vxe-select v-model="slotParams.row.type"> | ||||
|           <vxe-option v-for="item in typeSelect" :key="item.id" :value="item.value" :label="item.label"></vxe-option> | ||||
|         </vxe-select> | ||||
|       </template> | ||||
|     </vxe-column> | ||||
|     <vxe-column field="name" title="名称" :edit-render="{}"> | ||||
|       <template #edit="slotParams"> | ||||
|         <vxe-input v-model="slotParams.row.name" type="text"></vxe-input> | ||||
|       </template> | ||||
|     </vxe-column> | ||||
|     <vxe-column field="value" title="值" :edit-render="{}"> | ||||
|       <template #edit="slotParams"> | ||||
|         <vxe-input v-model="slotParams.row.value" type="text"></vxe-input> | ||||
|       </template> | ||||
|     </vxe-column> | ||||
|     <vxe-column title="操作" width="100" show-overflow align="center"> | ||||
|       <template #default="slotParams"> | ||||
|         <el-tooltip content="删除" placement="top"> | ||||
|           <el-button link type="danger" icon="Delete" @click="removeRow(slotParams.row)"></el-button> | ||||
|         </el-tooltip> | ||||
|       </template> | ||||
|     </vxe-column> | ||||
|   </vxe-table> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { VXETable, VxeTableInstance, VxeTablePropTypes } from 'vxe-table'; | ||||
| import { ParamVO } from 'bpmnDesign'; | ||||
| import useDialog from '@/hooks/useDialog'; | ||||
|  | ||||
| interface PropType { | ||||
|   height?: string; | ||||
|   tableData?: ParamVO[]; | ||||
| } | ||||
|  | ||||
| const { proxy } = getCurrentInstance() as ComponentInternalInstance; | ||||
|  | ||||
| const props = withDefaults(defineProps<PropType>(), { | ||||
|   height: '200px', | ||||
|   tableData: () => [] | ||||
| }); | ||||
|  | ||||
| const tableRules = ref<VxeTablePropTypes.EditRules>({ | ||||
|   type: [{ required: true, message: '请选择', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '请输入', trigger: 'blur' }], | ||||
|   value: [{ required: true, message: '请输入', trigger: 'blur' }] | ||||
| }); | ||||
|  | ||||
| const { title, visible, openDialog, closeDialog } = useDialog({ | ||||
|   title: '监听器参数' | ||||
| }); | ||||
| const typeSelect = [ | ||||
|   { id: '742fdeb7-23b4-416b-ac66-cd4ec8b901b7', label: '字符串', value: 'stringValue' }, | ||||
|   { id: '660c9c46-8fae-4bae-91a0-0335420019dc', label: '表达式', value: 'expression' } | ||||
| ]; | ||||
|  | ||||
| const tableRef = ref<VxeTableInstance<ParamVO>>(); | ||||
|  | ||||
| const getTableData = () => { | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     return $table.getTableData().fullData; | ||||
|   } | ||||
|   return []; | ||||
| }; | ||||
|  | ||||
| const insertRow = async () => { | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     const { row: newRow } = await $table.insertAt({}, -1); | ||||
|     // 插入一条数据并触发校验 | ||||
|     await $table.validate(newRow); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const removeRow = async (row: ParamVO) => { | ||||
|   await proxy?.$modal.confirm('您确定要删除该数据?'); | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     await $table.remove(row); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const validate = async () => { | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     return await $table.validate(true); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| defineExpose({ | ||||
|   closeDialog, | ||||
|   openDialog, | ||||
|   validate, | ||||
|   getTableData | ||||
| }); | ||||
| </script> | ||||
|  | ||||
| <style scoped lang="scss"></style> | ||||
							
								
								
									
										307
									
								
								src/components/BpmnDesign/panel/property/TaskListener.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										307
									
								
								src/components/BpmnDesign/panel/property/TaskListener.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,307 @@ | ||||
| <template> | ||||
|   <div> | ||||
|     <vxe-toolbar> | ||||
|       <template #buttons> | ||||
|         <el-button type="primary" link size="small" @click="insertEvent">新增</el-button> | ||||
|         <el-button type="primary" link size="small" @click="removeSelectRowEvent">删除</el-button> | ||||
|       </template> | ||||
|     </vxe-toolbar> | ||||
|     <vxe-table | ||||
|       ref="tableRef" | ||||
|       size="mini" | ||||
|       height="100px" | ||||
|       border | ||||
|       show-overflow | ||||
|       keep-source | ||||
|       :data="tableData" | ||||
|       :menu-config="menuConfig" | ||||
|       @cell-dblclick="cellDBLClickEvent" | ||||
|       @menu-click="contextMenuClickEvent" | ||||
|     > | ||||
|       <vxe-column type="checkbox" width="40"></vxe-column> | ||||
|       <vxe-column type="seq" width="40"></vxe-column> | ||||
|       <vxe-column field="event" title="事件" min-width="100px"> | ||||
|         <template #default="slotParams"> | ||||
|           <span>{{ eventSelect.find((e) => e.value === slotParams.row.event)?.label }}</span> | ||||
|         </template> | ||||
|       </vxe-column> | ||||
|       <vxe-column field="type" title="类型" min-width="100px"> | ||||
|         <template #default="slotParams"> | ||||
|           <span>{{ typeSelect.find((e) => e.value === slotParams.row.type)?.label }}</span> | ||||
|         </template> | ||||
|       </vxe-column> | ||||
|       <vxe-column field="className" title="Java 类名" min-width="100px"> </vxe-column> | ||||
|     </vxe-table> | ||||
|  | ||||
|     <el-dialog | ||||
|       v-model="formDialog.visible.value" | ||||
|       :title="formDialog.title.value" | ||||
|       width="600px" | ||||
|       :close-on-click-modal="false" | ||||
|       :close-on-press-escape="false" | ||||
|       :show-close="false" | ||||
|       append-to-body | ||||
|     > | ||||
|       <el-form ref="formRef" :model="formData" :rules="tableRules" label-width="90px"> | ||||
|         <el-form-item label="事件" prop="event"> | ||||
|           <template #label> | ||||
|             <span> | ||||
|               事件 | ||||
|               <el-tooltip placement="top"> | ||||
|                 <el-icon><QuestionFilled /></el-icon> | ||||
|                 <template #content> | ||||
|                   create(创建):当任务已经创建,并且所有任务参数都已经设置时触发。<br /> | ||||
|                   assignment(指派):当任务已经指派给某人时触发。请注意:当流程执行到达用户任务时,在触发create事件之前,会首先触发assignment事件。<br /> | ||||
|                   complete(完成):当任务已经完成,从运行时数据中删除前触发。<br /> | ||||
|                   delete(删除):在任务即将被删除前触发。请注意任务由completeTask正常完成时也会触发。 | ||||
|                 </template> | ||||
|               </el-tooltip> | ||||
|             </span> | ||||
|           </template> | ||||
|           <el-select v-model="formData.event"> | ||||
|             <el-option v-for="item in eventSelect" :key="item.id" :value="item.value" :label="item.label"></el-option> | ||||
|           </el-select> | ||||
|         </el-form-item> | ||||
|         <el-form-item label="类型" prop="type"> | ||||
|           <el-select v-model="formData.type"> | ||||
|             <el-option v-for="item in typeSelect" :key="item.id" :value="item.value" :label="item.label"></el-option> | ||||
|           </el-select> | ||||
|         </el-form-item> | ||||
|         <el-form-item label="Java 类名" prop="className"> | ||||
|           <el-input v-model="formData.className" type="text"></el-input> | ||||
|         </el-form-item> | ||||
|       </el-form> | ||||
|       <el-tabs type="border-card"> | ||||
|         <el-tab-pane label="参数"> | ||||
|           <ListenerParam ref="listenerParamRef" :table-data="formData.params" /> | ||||
|         </el-tab-pane> | ||||
|       </el-tabs> | ||||
|       <template #footer> | ||||
|         <div class="dialog-footer"> | ||||
|           <el-button @click="formDialog.closeDialog">取 消</el-button> | ||||
|           <el-button type="primary" @click="submitEvent">确 定</el-button> | ||||
|         </div> | ||||
|       </template> | ||||
|     </el-dialog> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import ListenerParam from './ListenerParam.vue'; | ||||
| import { VxeTableEvents, VxeTableInstance, VxeTablePropTypes } from 'vxe-table'; | ||||
| import { TaskListenerVO } from 'bpmnDesign'; | ||||
| import { ModdleElement } from 'bpmn'; | ||||
|  | ||||
| import usePanel from '@/components/BpmnDesign/hooks/usePanel'; | ||||
| import useDialog from '@/hooks/useDialog'; | ||||
| import useModelerStore from '@/store/modules/modeler'; | ||||
|  | ||||
| const { proxy } = getCurrentInstance() as ComponentInternalInstance; | ||||
|  | ||||
| interface PropType { | ||||
|   element: ModdleElement; | ||||
| } | ||||
| const props = withDefaults(defineProps<PropType>(), {}); | ||||
|  | ||||
| const selectRow = ref<TaskListenerVO | null>(); | ||||
| const formDialog = useDialog({ | ||||
|   title: selectRow.value ? '编辑&保存' : '新增&保存' | ||||
| }); | ||||
| const { showConfig, elementType, updateProperties } = usePanel({ | ||||
|   element: toRaw(props.element) | ||||
| }); | ||||
| const { getModdle } = useModelerStore(); | ||||
| const moddle = getModdle(); | ||||
|  | ||||
| const listenerParamRef = ref<InstanceType<typeof ListenerParam>>(); | ||||
| const tableRef = ref<VxeTableInstance<TaskListenerVO>>(); | ||||
| const formRef = ref<ElFormInstance>(); | ||||
|  | ||||
| const initData: TaskListenerVO = { | ||||
|   event: '', | ||||
|   type: '', | ||||
|   className: '', | ||||
|   name: '', | ||||
|   params: [] | ||||
| }; | ||||
| const formData = ref<TaskListenerVO>({ ...initData }); | ||||
| const currentIndex = ref(0); | ||||
| const tableData = ref<TaskListenerVO[]>([]); | ||||
| const tableRules = ref<VxeTablePropTypes.EditRules>({ | ||||
|   event: [{ required: true, message: '请选择', trigger: 'blur' }], | ||||
|   type: [{ required: true, message: '请选择', trigger: 'blur' }], | ||||
|   name: [{ required: true, message: '请输入', trigger: 'blur' }], | ||||
|   className: [{ required: true, message: '请输入', trigger: 'blur' }] | ||||
| }); | ||||
|  | ||||
| const submitEvent = async () => { | ||||
|   const error = await listenerParamRef.value.validate(); | ||||
|   await formRef.value.validate((validate) => { | ||||
|     if (validate && !error) { | ||||
|       const $table = tableRef.value; | ||||
|       if ($table) { | ||||
|         formData.value.params = listenerParamRef.value.getTableData(); | ||||
|         if (selectRow.value) { | ||||
|           Object.assign(selectRow.value, formData.value); | ||||
|         } else { | ||||
|           $table.insertAt({ ...formData.value }, -1); | ||||
|         } | ||||
|         updateElement(); | ||||
|         formDialog.closeDialog(); | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| const insertEvent = async () => { | ||||
|   Object.assign(formData.value, initData); | ||||
|   selectRow.value = null; | ||||
|   formDialog.openDialog(); | ||||
| }; | ||||
|  | ||||
| const editEvent = (row: TaskListenerVO) => { | ||||
|   Object.assign(formData.value, row); | ||||
|   selectRow.value = row; | ||||
|   formDialog.openDialog(); | ||||
| }; | ||||
| const removeEvent = async (row: TaskListenerVO) => { | ||||
|   await proxy?.$modal.confirm('您确定要删除该数据?'); | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     await $table.remove(row); | ||||
|     updateElement(); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const removeSelectRowEvent = async () => { | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     const selectCount = $table.getCheckboxRecords().length; | ||||
|     if (selectCount === 0) { | ||||
|       proxy?.$modal.msgWarning('请选择行'); | ||||
|     } else { | ||||
|       await $table.removeCheckboxRow(); | ||||
|       updateElement(); | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| const updateElement = () => { | ||||
|   const $table = tableRef.value; | ||||
|   const data = $table.getTableData().fullData; | ||||
|   if (data.length) { | ||||
|     let extensionElements = props.element.businessObject.get('extensionElements'); | ||||
|     if (!extensionElements) { | ||||
|       extensionElements = moddle.create('bpmn:ExtensionElements'); | ||||
|     } | ||||
|     // 清除旧值 | ||||
|     extensionElements.values = extensionElements.values?.filter((item) => item.$type !== 'flowable:TaskListener') ?? []; | ||||
|     data.forEach((item) => { | ||||
|       const taskListener = moddle.create('flowable:TaskListener'); | ||||
|       taskListener['event'] = item.event; | ||||
|       taskListener[item.type] = item.className; | ||||
|       if (item.params && item.params.length) { | ||||
|         item.params.forEach((field) => { | ||||
|           const fieldElement = moddle.create('flowable:Field'); | ||||
|           fieldElement['name'] = field.name; | ||||
|           fieldElement[field.type] = field.value; | ||||
|           taskListener.get('fields').push(fieldElement); | ||||
|         }); | ||||
|       } | ||||
|       extensionElements.get('values').push(taskListener); | ||||
|     }); | ||||
|     updateProperties({ extensionElements: extensionElements }); | ||||
|   } else { | ||||
|     const extensionElements = props.element.businessObject[`extensionElements`]; | ||||
|     if (extensionElements) { | ||||
|       extensionElements.values = extensionElements.values?.filter((item) => item.$type !== 'flowable:TaskListener') ?? []; | ||||
|     } | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const cellDBLClickEvent: VxeTableEvents.CellDblclick<TaskListenerVO> = ({ row }) => { | ||||
|   editEvent(row); | ||||
| }; | ||||
|  | ||||
| const menuConfig = reactive<VxeTablePropTypes.MenuConfig<TaskListenerVO>>({ | ||||
|   body: { | ||||
|     options: [ | ||||
|       [ | ||||
|         { code: 'edit', name: '编辑', prefixIcon: 'vxe-icon-edit', disabled: false }, | ||||
|         { code: 'remove', name: '删除', prefixIcon: 'vxe-icon-delete', disabled: false } | ||||
|       ] | ||||
|     ] | ||||
|   }, | ||||
|   visibleMethod({ options, column }) { | ||||
|     const isDisabled = !column; | ||||
|     options.forEach((list) => { | ||||
|       list.forEach((item) => { | ||||
|         item.disabled = isDisabled; | ||||
|       }); | ||||
|     }); | ||||
|     return true; | ||||
|   } | ||||
| }); | ||||
| const contextMenuClickEvent: VxeTableEvents.MenuClick<TaskListenerVO> = ({ menu, row, column }) => { | ||||
|   const $table = tableRef.value; | ||||
|   if ($table) { | ||||
|     switch (menu.code) { | ||||
|       case 'edit': | ||||
|         editEvent(row); | ||||
|         break; | ||||
|       case 'remove': | ||||
|         removeEvent(row); | ||||
|         break; | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| const initTableData = () => { | ||||
|   tableData.value = | ||||
|     props.element.businessObject.extensionElements?.values | ||||
|       .filter((item) => item.$type === 'flowable:TaskListener') | ||||
|       .map((item) => { | ||||
|         let type; | ||||
|         if ('class' in item) type = 'class'; | ||||
|         if ('expression' in item) type = 'expression'; | ||||
|         if ('delegateExpression' in item) type = 'delegateExpression'; | ||||
|         return { | ||||
|           event: item.event, | ||||
|           type: type, | ||||
|           className: item[type], | ||||
|           params: | ||||
|             item.fields?.map((field) => { | ||||
|               let fieldType; | ||||
|               if ('stringValue' in field) fieldType = 'stringValue'; | ||||
|               if ('expression' in field) fieldType = 'expression'; | ||||
|               return { | ||||
|                 name: field.name, | ||||
|                 type: fieldType, | ||||
|                 value: field[fieldType] | ||||
|               }; | ||||
|             }) ?? [] | ||||
|         }; | ||||
|       }) ?? []; | ||||
| }; | ||||
|  | ||||
| onMounted(() => { | ||||
|   initTableData(); | ||||
| }); | ||||
|  | ||||
| const typeSelect = [ | ||||
|   { id: '742fdeb7-23b4-416b-ac66-cd4ec8b901b7', label: '类', value: 'class' }, | ||||
|   { id: '660c9c46-8fae-4bae-91a0-0335420019dc', label: '表达式', value: 'expression' }, | ||||
|   { id: '4b8135ab-6bc3-4a0f-80be-22f58bc6c5fd', label: '委托表达式', value: 'delegateExpression' } | ||||
| ]; | ||||
| const eventSelect = [ | ||||
|   { id: 'e6e0a51a-2d5d-4dc4-b847-b5c14f43a6ab', label: '创建', value: 'create' }, | ||||
|   { id: '6da97c1e-15fc-4445-8943-75d09f49778e', label: '指派', value: 'assignment' }, | ||||
|   { id: '6a2cbcec-e026-4f11-bef7-fff0b5c871e2', label: '完成', value: 'complete' }, | ||||
|   { id: '68801972-85f1-482f-bd86-1fad015c26ed', label: '删除', value: 'delete' } | ||||
| ]; | ||||
| </script> | ||||
|  | ||||
| <style scoped lang="scss"> | ||||
| .el-badge { | ||||
|   :deep(.el-badge__content) { | ||||
|     top: 10px; | ||||
|   } | ||||
| } | ||||
| </style> | ||||
		Reference in New Issue
	
	Block a user
	 疯狂的狮子Li
					疯狂的狮子Li