PropertiesPanel.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  1. <template>
  2. <div class="process-panel__container" :style="{ width: `${width}px`, maxHeight: '600px' }">
  3. <el-collapse v-model="activeTab" v-if="isReady">
  4. <el-collapse-item name="base">
  5. <!-- class="panel-tab__title" -->
  6. <template #title>
  7. <Icon icon="ep:info-filled" />
  8. 常规</template
  9. >
  10. <ElementBaseInfo
  11. :id-edit-disabled="idEditDisabled"
  12. :business-object="elementBusinessObject"
  13. :type="elementType"
  14. :model="model"
  15. />
  16. </el-collapse-item>
  17. <el-collapse-item name="condition" v-if="elementType === 'Process'" key="message">
  18. <template #title><Icon icon="ep:comment" />消息与信号</template>
  19. <signal-and-massage />
  20. </el-collapse-item>
  21. <el-collapse-item name="condition" v-if="conditionFormVisible" key="condition">
  22. <template #title><Icon icon="ep:promotion" />流转条件</template>
  23. <flow-condition :business-object="elementBusinessObject" :type="elementType" />
  24. </el-collapse-item>
  25. <el-collapse-item name="condition" v-if="formVisible" key="form">
  26. <template #title><Icon icon="ep:list" />表单</template>
  27. <element-form :id="elementId" :type="elementType" />
  28. </el-collapse-item>
  29. <el-collapse-item name="task" v-if="isTaskCollapseItemShow(elementType)" key="task">
  30. <template #title
  31. ><Icon icon="ep:checked" />{{ getTaskCollapseItemName(elementType) }}</template
  32. >
  33. <element-task :id="elementId" :type="elementType" />
  34. </el-collapse-item>
  35. <el-collapse-item
  36. name="multiInstance"
  37. v-if="elementType.indexOf('Task') !== -1"
  38. key="multiInstance"
  39. >
  40. <template #title><Icon icon="ep:help-filled" />多人审批方式</template>
  41. <element-multi-instance
  42. :id="elementId"
  43. :business-object="elementBusinessObject"
  44. :type="elementType"
  45. />
  46. </el-collapse-item>
  47. <el-collapse-item name="listeners" key="listeners">
  48. <template #title><Icon icon="ep:bell-filled" />执行监听器</template>
  49. <element-listeners :id="elementId" :type="elementType" />
  50. </el-collapse-item>
  51. <el-collapse-item name="taskListeners" v-if="elementType === 'UserTask'" key="taskListeners">
  52. <template #title><Icon icon="ep:bell-filled" />任务监听器</template>
  53. <user-task-listeners :id="elementId" :type="elementType" />
  54. </el-collapse-item>
  55. <el-collapse-item name="extensions" key="extensions">
  56. <template #title><Icon icon="ep:circle-plus-filled" />扩展属性</template>
  57. <element-properties :id="elementId" :type="elementType" />
  58. </el-collapse-item>
  59. <el-collapse-item name="other" key="other">
  60. <template #title><Icon icon="ep:promotion" />其他</template>
  61. <element-other-config :id="elementId" />
  62. </el-collapse-item>
  63. <el-collapse-item name="customConfig" key="customConfig">
  64. <template #title><Icon icon="ep:tools" />自定义配置</template>
  65. <element-custom-config
  66. :id="elementId"
  67. :type="elementType"
  68. :business-object="elementBusinessObject"
  69. />
  70. </el-collapse-item>
  71. <!-- 新增的时间事件配置项 -->
  72. <el-collapse-item v-if="elementType === 'IntermediateCatchEvent'" name="timeEvent">
  73. <template #title><Icon icon="ep:timer" />时间事件</template>
  74. <!-- 相关 issue:https://gitee.com/xindazhou/xdz-admin/issues/ICNRW2 -->
  75. <TimeEventConfig :businessObject="elementBusinessObject" :key="elementId" />
  76. </el-collapse-item>
  77. </el-collapse>
  78. </div>
  79. </template>
  80. <script lang="ts" setup>
  81. import ElementBaseInfo from './base/ElementBaseInfo.vue'
  82. import ElementOtherConfig from './other/ElementOtherConfig.vue'
  83. import ElementTask from './task/ElementTask.vue'
  84. import ElementMultiInstance from './multi-instance/ElementMultiInstance.vue'
  85. import FlowCondition from './flow-condition/FlowCondition.vue'
  86. import SignalAndMassage from './signal-message/SignalAndMessage.vue'
  87. import ElementListeners from './listeners/ElementListeners.vue'
  88. import ElementProperties from './properties/ElementProperties.vue'
  89. // import ElementForm from './form/ElementForm.vue'
  90. import UserTaskListeners from './listeners/UserTaskListeners.vue'
  91. import { getTaskCollapseItemName, isTaskCollapseItemShow } from './task/data'
  92. import TimeEventConfig from './time-event-config/TimeEventConfig.vue'
  93. import { ref, watch, onMounted } from 'vue'
  94. defineOptions({ name: 'MyPropertiesPanel' })
  95. /**
  96. * 侧边栏
  97. * @Author MiyueFE
  98. * @Home https://github.com/miyuesc
  99. * @Date 2021年3月31日18:57:51
  100. */
  101. const props = defineProps({
  102. bpmnModeler: {
  103. type: Object,
  104. default: () => {}
  105. },
  106. prefix: {
  107. type: String,
  108. default: 'camunda'
  109. },
  110. width: {
  111. type: Number,
  112. default: 480
  113. },
  114. idEditDisabled: {
  115. type: Boolean,
  116. default: false
  117. },
  118. model: Object // 流程模型的数据
  119. })
  120. const activeTab = ref('base')
  121. const elementId = ref('')
  122. const elementType = ref('')
  123. const elementBusinessObject = ref<any>({}) // 元素 businessObject 镜像,提供给需要做判断的组件使用
  124. const conditionFormVisible = ref(false) // 流转条件设置
  125. const formVisible = ref(false) // 表单配置
  126. const bpmnElement = ref()
  127. const isReady = ref(false)
  128. const type = ref('time')
  129. const condition = ref('')
  130. provide('prefix', props.prefix)
  131. provide('width', props.width)
  132. // 初始化 bpmnInstances
  133. const initBpmnInstances = () => {
  134. if (!props.bpmnModeler) return false
  135. try {
  136. const instances = {
  137. modeler: props.bpmnModeler,
  138. modeling: props.bpmnModeler.get('modeling'),
  139. moddle: props.bpmnModeler.get('moddle'),
  140. eventBus: props.bpmnModeler.get('eventBus'),
  141. bpmnFactory: props.bpmnModeler.get('bpmnFactory'),
  142. elementFactory: props.bpmnModeler.get('elementFactory'),
  143. elementRegistry: props.bpmnModeler.get('elementRegistry'),
  144. replace: props.bpmnModeler.get('replace'),
  145. selection: props.bpmnModeler.get('selection')
  146. }
  147. // 检查所有实例是否都存在
  148. const allInstancesExist = Object.values(instances).every((instance) => instance)
  149. if (allInstancesExist) {
  150. const w = window as any
  151. w.bpmnInstances = instances
  152. return true
  153. }
  154. return false
  155. } catch (error) {
  156. console.error('初始化 bpmnInstances 失败:', error)
  157. return false
  158. }
  159. }
  160. const bpmnInstances = () => (window as any)?.bpmnInstances
  161. // 监听 props.bpmnModeler 然后 initModels
  162. const unwatchBpmn = watch(
  163. () => props.bpmnModeler,
  164. async () => {
  165. // 避免加载时 流程图 并未加载完成
  166. if (!props.bpmnModeler) {
  167. console.log('缺少props.bpmnModeler')
  168. return
  169. }
  170. try {
  171. // 等待 modeler 初始化完成
  172. await nextTick()
  173. if (initBpmnInstances()) {
  174. isReady.value = true
  175. await nextTick()
  176. getActiveElement()
  177. } else {
  178. console.error('modeler 实例未完全初始化')
  179. }
  180. } catch (error) {
  181. console.error('初始化失败:', error)
  182. }
  183. },
  184. {
  185. immediate: true
  186. }
  187. )
  188. const getActiveElement = () => {
  189. if (!isReady.value || !props.bpmnModeler) return
  190. // 初始第一个选中元素 bpmn:Process
  191. initFormOnChanged(null)
  192. props.bpmnModeler.on('import.done', (e) => {
  193. console.log(e, 'eeeee')
  194. initFormOnChanged(null)
  195. })
  196. // 监听选择事件,修改当前激活的元素以及表单
  197. props.bpmnModeler.on('selection.changed', ({ newSelection }) => {
  198. initFormOnChanged(newSelection[0] || null)
  199. })
  200. props.bpmnModeler.on('element.changed', ({ element }) => {
  201. // 保证 修改 "默认流转路径" 类似需要修改多个元素的事件发生的时候,更新表单的元素与原选中元素不一致。
  202. if (element && element.id === elementId.value) {
  203. initFormOnChanged(element)
  204. }
  205. })
  206. }
  207. // 初始化数据
  208. const initFormOnChanged = (element) => {
  209. if (!isReady.value || !bpmnInstances()) return
  210. let activatedElement = element
  211. if (!activatedElement) {
  212. activatedElement =
  213. bpmnInstances().elementRegistry.find((el) => el.type === 'bpmn:Process') ??
  214. bpmnInstances().elementRegistry.find((el) => el.type === 'bpmn:Collaboration')
  215. }
  216. if (!activatedElement) return
  217. try {
  218. console.log(`
  219. ----------
  220. select element changed:
  221. id: ${activatedElement.id}
  222. type: ${activatedElement.businessObject.$type}
  223. ----------
  224. `)
  225. console.log('businessObject: ', activatedElement.businessObject)
  226. bpmnInstances().bpmnElement = activatedElement
  227. bpmnElement.value = activatedElement
  228. elementId.value = activatedElement.id
  229. elementType.value = activatedElement.type.split(':')[1] || ''
  230. elementBusinessObject.value = JSON.parse(JSON.stringify(activatedElement.businessObject))
  231. conditionFormVisible.value = !!(
  232. elementType.value === 'SequenceFlow' &&
  233. activatedElement.source &&
  234. activatedElement.source.type.indexOf('StartEvent') === -1
  235. )
  236. formVisible.value = elementType.value === 'UserTask' || elementType.value === 'StartEvent'
  237. } catch (error) {
  238. console.error('初始化表单数据失败:', error)
  239. }
  240. }
  241. onBeforeUnmount(() => {
  242. const w = window as any
  243. w.bpmnInstances = null
  244. isReady.value = false
  245. })
  246. watch(
  247. () => elementId.value,
  248. () => {
  249. activeTab.value = 'base'
  250. }
  251. )
  252. function updateNode() {
  253. const moddle = window.bpmnInstances?.moddle
  254. const modeling = window.bpmnInstances?.modeling
  255. const elementRegistry = window.bpmnInstances?.elementRegistry
  256. if (!moddle || !modeling || !elementRegistry) return
  257. const element = elementRegistry.get(props.businessObject.id)
  258. if (!element) return
  259. let timerDef = moddle.create('bpmn:TimerEventDefinition', {})
  260. if (type.value === 'time') {
  261. timerDef.timeDate = moddle.create('bpmn:FormalExpression', { body: condition.value })
  262. } else if (type.value === 'duration') {
  263. timerDef.timeDuration = moddle.create('bpmn:FormalExpression', { body: condition.value })
  264. } else if (type.value === 'cycle') {
  265. timerDef.timeCycle = moddle.create('bpmn:FormalExpression', { body: condition.value })
  266. }
  267. modeling.updateModdleProperties(element, element.businessObject, {
  268. eventDefinitions: [timerDef]
  269. })
  270. }
  271. // 初始化和监听
  272. function syncFromBusinessObject() {
  273. if (props.businessObject) {
  274. const timerDef = (props.businessObject.eventDefinitions || [])[0]
  275. if (timerDef) {
  276. if (timerDef.timeDate) {
  277. type.value = 'time'
  278. condition.value = timerDef.timeDate.body
  279. } else if (timerDef.timeDuration) {
  280. type.value = 'duration'
  281. condition.value = timerDef.timeDuration.body
  282. } else if (timerDef.timeCycle) {
  283. type.value = 'cycle'
  284. condition.value = timerDef.timeCycle.body
  285. }
  286. }
  287. }
  288. }
  289. onMounted(syncFromBusinessObject)
  290. watch(() => props.businessObject, syncFromBusinessObject, { deep: true })
  291. </script>