index.vue 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. <template>
  2. <!-- 第一步,通过流程定义的列表,选择对应的流程 -->
  3. <ContentWrap v-if="!selectProcessDefinition" v-loading="loading">
  4. <el-tabs tab-position="left" v-model="categoryActive">
  5. <el-tab-pane
  6. :label="category.name"
  7. :name="category.code"
  8. :key="category.code"
  9. v-for="category in categoryList"
  10. >
  11. <el-row :gutter="20">
  12. <el-col
  13. :lg="6"
  14. :sm="12"
  15. :xs="24"
  16. v-for="definition in categoryProcessDefinitionList"
  17. :key="definition.id"
  18. >
  19. <el-card
  20. shadow="hover"
  21. class="mb-20px cursor-pointer"
  22. @click="handleSelect(definition)"
  23. >
  24. <template #default>
  25. <div class="flex">
  26. <el-image :src="definition.icon" class="w-32px h-32px" />
  27. <el-text class="!ml-10px" size="large">{{ definition.name }}</el-text>
  28. </div>
  29. </template>
  30. </el-card>
  31. </el-col>
  32. </el-row>
  33. </el-tab-pane>
  34. </el-tabs>
  35. </ContentWrap>
  36. <!-- 第二步,填写表单,进行流程的提交 -->
  37. <ContentWrap v-else>
  38. <el-card class="box-card">
  39. <div class="clearfix">
  40. <span class="el-icon-document">申请信息【{{ selectProcessDefinition.name }}】</span>
  41. <el-button style="float: right" type="primary" @click="selectProcessDefinition = undefined">
  42. <Icon icon="ep:delete" /> 选择其它流程
  43. </el-button>
  44. </div>
  45. <el-col :span="16" :offset="6" style="margin-top: 20px">
  46. <form-create
  47. :rule="detailForm.rule"
  48. v-model:api="fApi"
  49. v-model="detailForm.value"
  50. :option="detailForm.option"
  51. @submit="submitForm"
  52. >
  53. <template #type-startUserSelect>
  54. <el-col :span="24">
  55. <el-card class="mb-10px">
  56. <template #header>指定审批人</template>
  57. <el-form
  58. :model="startUserSelectAssignees"
  59. :rules="startUserSelectAssigneesFormRules"
  60. ref="startUserSelectAssigneesFormRef"
  61. >
  62. <el-form-item
  63. v-for="userTask in startUserSelectTasks"
  64. :key="userTask.id"
  65. :label="`任务【${userTask.name}】`"
  66. :prop="userTask.id"
  67. >
  68. <el-select
  69. v-model="startUserSelectAssignees[userTask.id]"
  70. multiple
  71. placeholder="请选择审批人"
  72. >
  73. <el-option
  74. v-for="user in userList"
  75. :key="user.id"
  76. :label="user.nickname"
  77. :value="user.id"
  78. />
  79. </el-select>
  80. </el-form-item>
  81. </el-form>
  82. </el-card>
  83. </el-col>
  84. </template>
  85. </form-create>
  86. </el-col>
  87. </el-card>
  88. <!-- 流程图预览 -->
  89. <ProcessInstanceBpmnViewer :bpmn-xml="bpmnXML as any" />
  90. </ContentWrap>
  91. </template>
  92. <script lang="ts" setup>
  93. import * as DefinitionApi from '@/api/bpm/definition'
  94. import * as ProcessInstanceApi from '@/api/bpm/processInstance'
  95. import { setConfAndFields2 } from '@/utils/formCreate'
  96. import type { ApiAttrs } from '@form-create/element-ui/types/config'
  97. import ProcessInstanceBpmnViewer from '../detail/ProcessInstanceBpmnViewer.vue'
  98. import { CategoryApi } from '@/api/bpm/category'
  99. import { useTagsViewStore } from '@/store/modules/tagsView'
  100. import * as UserApi from '@/api/system/user'
  101. defineOptions({ name: 'BpmProcessInstanceCreate' })
  102. const route = useRoute() // 路由
  103. const { push, currentRoute } = useRouter() // 路由
  104. const message = useMessage() // 消息
  105. const { delView } = useTagsViewStore() // 视图操作
  106. const processInstanceId = route.query.processInstanceId
  107. const loading = ref(true) // 加载中
  108. const categoryList = ref([]) // 分类的列表
  109. const categoryActive = ref('') // 选中的分类
  110. const processDefinitionList = ref([]) // 流程定义的列表
  111. /** 查询列表 */
  112. const getList = async () => {
  113. loading.value = true
  114. try {
  115. // 流程分类
  116. categoryList.value = await CategoryApi.getCategorySimpleList()
  117. if (categoryList.value.length > 0) {
  118. categoryActive.value = categoryList.value[0].code
  119. }
  120. // 流程定义
  121. processDefinitionList.value = await DefinitionApi.getProcessDefinitionList({
  122. suspensionState: 1
  123. })
  124. // 如果 processInstanceId 非空,说明是重新发起
  125. if (processInstanceId?.length > 0) {
  126. const processInstance = await ProcessInstanceApi.getProcessInstance(processInstanceId)
  127. if (!processInstance) {
  128. message.error('重新发起流程失败,原因:流程实例不存在')
  129. return
  130. }
  131. const processDefinition = processDefinitionList.value.find(
  132. (item) => item.key == processInstance.processDefinition?.key
  133. )
  134. if (!processDefinition) {
  135. message.error('重新发起流程失败,原因:流程定义不存在')
  136. return
  137. }
  138. await handleSelect(processDefinition, processInstance.formVariables)
  139. }
  140. } finally {
  141. loading.value = false
  142. }
  143. }
  144. /** 选中分类对应的流程定义列表 */
  145. const categoryProcessDefinitionList = computed(() => {
  146. return processDefinitionList.value.filter((item) => item.category == categoryActive.value)
  147. })
  148. // ========== 表单相关 ==========
  149. const fApi = ref<ApiAttrs>()
  150. const detailForm = ref({
  151. rule: [],
  152. option: {},
  153. value: {}
  154. }) // 流程表单详情
  155. const selectProcessDefinition = ref() // 选择的流程定义
  156. // 指定审批人
  157. const bpmnXML = ref(null) // BPMN 数据
  158. const startUserSelectTasks = ref([]) // 发起人需要选择审批人的用户任务列表
  159. const startUserSelectAssignees = ref({}) // 发起人选择审批人的数据
  160. const startUserSelectAssigneesFormRef = ref() // 发起人选择审批人的表单 Ref
  161. const startUserSelectAssigneesFormRules = ref({}) // 发起人选择审批人的表单 Rules
  162. const userList = ref<any[]>([]) // 用户列表
  163. /** 处理选择流程的按钮操作 **/
  164. const handleSelect = async (row, formVariables) => {
  165. // 设置选择的流程
  166. selectProcessDefinition.value = row
  167. // 重置指定审批人
  168. startUserSelectTasks.value = []
  169. startUserSelectAssignees.value = {}
  170. startUserSelectAssigneesFormRules.value = {}
  171. // 情况一:流程表单
  172. if (row.formType == 10) {
  173. // 设置表单
  174. setConfAndFields2(detailForm, row.formConf, row.formFields, formVariables)
  175. // 加载流程图
  176. const processDefinitionDetail = await DefinitionApi.getProcessDefinition(row.id)
  177. if (processDefinitionDetail) {
  178. bpmnXML.value = processDefinitionDetail.bpmnXml
  179. startUserSelectTasks.value = processDefinitionDetail.startUserSelectTasks
  180. // 设置指定审批人
  181. if (startUserSelectTasks.value?.length > 0) {
  182. detailForm.value.rule.push({
  183. type: 'startUserSelect',
  184. props: {
  185. title: '指定审批人'
  186. }
  187. })
  188. // 设置校验规则
  189. for (const userTask of startUserSelectTasks.value) {
  190. startUserSelectAssignees.value[userTask.id] = []
  191. startUserSelectAssigneesFormRules.value[userTask.id] = [
  192. { required: true, message: '请选择审批人', trigger: 'blur' }
  193. ]
  194. }
  195. // 加载用户列表
  196. userList.value = await UserApi.getSimpleUserList()
  197. }
  198. }
  199. // 情况二:业务表单
  200. } else if (row.formCustomCreatePath) {
  201. await push({
  202. path: row.formCustomCreatePath
  203. })
  204. // 这里暂时无需加载流程图,因为跳出到另外个 Tab;
  205. }
  206. }
  207. /** 提交按钮 */
  208. const submitForm = async (formData) => {
  209. if (!fApi.value || !selectProcessDefinition.value) {
  210. return
  211. }
  212. // 如果有指定审批人,需要校验
  213. if (startUserSelectTasks.value?.length > 0) {
  214. await startUserSelectAssigneesFormRef.value.validate()
  215. }
  216. // 提交请求
  217. fApi.value.btn.loading(true)
  218. try {
  219. await ProcessInstanceApi.createProcessInstance({
  220. processDefinitionId: selectProcessDefinition.value.id,
  221. variables: formData,
  222. startUserSelectAssignees: startUserSelectAssignees.value
  223. })
  224. // 提示
  225. message.success('发起流程成功')
  226. // 跳转回去
  227. delView(unref(currentRoute))
  228. await push({
  229. name: 'BpmProcessInstanceMy'
  230. })
  231. } finally {
  232. fApi.value.btn.loading(false)
  233. }
  234. }
  235. /** 初始化 */
  236. onMounted(() => {
  237. getList()
  238. })
  239. </script>