index.vue 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. <template>
  2. <!-- 第一步,通过流程定义的列表,选择对应的流程 -->
  3. <template v-if="!selectProcessDefinition">
  4. <el-input
  5. v-model="currentSearchKey"
  6. class="!w-50% mb-15px"
  7. placeholder="请输入流程名称"
  8. clearable
  9. @keyup.enter="handleQuery"
  10. @clear="handleClear"
  11. >
  12. <template #prefix>
  13. <Icon icon="ep:search" />
  14. </template>
  15. </el-input>
  16. <ContentWrap
  17. :class="{ 'process-definition-container': filteredProcessDefinitionList?.length }"
  18. class="position-relative pb-20px h-700px"
  19. v-loading="loading"
  20. >
  21. <el-row v-if="filteredProcessDefinitionList?.length" :gutter="20" class="!flex-nowrap">
  22. <el-col :span="5">
  23. <div class="flex flex-col">
  24. <div
  25. v-for="category in categoryList"
  26. :key="category.code"
  27. class="flex items-center p-10px cursor-pointer text-14px rounded-md"
  28. :class="categoryActive.code === category.code ? 'text-#3e7bff bg-#e8eeff' : ''"
  29. @click="handleCategoryClick(category)"
  30. >
  31. {{ category.name }}
  32. </div>
  33. </div>
  34. </el-col>
  35. <el-col :span="19">
  36. <el-scrollbar ref="scrollWrapper" height="700">
  37. <div
  38. class="mb-20px pl-10px"
  39. v-for="(definitions, title) in processDefinitionGroup"
  40. :key="title"
  41. :ref="`category-${title}`"
  42. >
  43. <h3 class="text-18px font-bold mb-10px mt-5px">{{ title }}</h3>
  44. <div class="grid grid-cols-3 gap3">
  45. <el-tooltip
  46. v-for="definition in definitions"
  47. :key="definition.id"
  48. :content="definition.description"
  49. placement="top"
  50. >
  51. <el-card
  52. shadow="hover"
  53. class="cursor-pointer definition-item-card"
  54. @click="handleSelect(definition)"
  55. >
  56. <template #default>
  57. <div class="flex">
  58. <el-image :src="definition.icon" class="w-32px h-32px" />
  59. <el-text class="!ml-10px" size="large">{{ definition.name }}</el-text>
  60. </div>
  61. </template>
  62. </el-card>
  63. </el-tooltip>
  64. </div>
  65. </div>
  66. </el-scrollbar>
  67. </el-col>
  68. </el-row>
  69. <el-empty class="!py-200px" :image-size="200" description="没有找到搜索结果" v-else />
  70. </ContentWrap>
  71. </template>
  72. <!-- 第二步,填写表单,进行流程的提交 -->
  73. <ProcessDefinitionDetail
  74. v-else
  75. ref="processDefinitionDetailRef"
  76. :selectProcessDefinition="selectProcessDefinition"
  77. @cancel="selectProcessDefinition = undefined"
  78. />
  79. </template>
  80. <script lang="ts" setup>
  81. import * as DefinitionApi from '@/api/bpm/definition'
  82. import * as ProcessInstanceApi from '@/api/bpm/processInstance'
  83. import { CategoryApi } from '@/api/bpm/category'
  84. import ProcessDefinitionDetail from './ProcessDefinitionDetail.vue'
  85. import { groupBy } from 'lodash-es'
  86. defineOptions({ name: 'BpmProcessInstanceCreate' })
  87. const { proxy } = getCurrentInstance() as any
  88. const route = useRoute() // 路由
  89. const message = useMessage() // 消息
  90. const currentSearchKey = ref('') // 当前搜索关键字
  91. const processInstanceId: any = route.query.processInstanceId
  92. const loading = ref(true) // 加载中
  93. const categoryList: any = ref([]) // 分类的列表
  94. const categoryActive: any = ref({}) // 选中的分类
  95. const processDefinitionList = ref([]) // 流程定义的列表
  96. /** 查询列表 */
  97. const getList = async () => {
  98. loading.value = true
  99. try {
  100. // 所有流程分类数据
  101. await getCategoryList()
  102. // 所有流程定义数据
  103. await getDefinitionList()
  104. // 如果 processInstanceId 非空,说明是重新发起
  105. if (processInstanceId?.length > 0) {
  106. const processInstance = await ProcessInstanceApi.getProcessInstance(processInstanceId)
  107. if (!processInstance) {
  108. message.error('重新发起流程失败,原因:流程实例不存在')
  109. return
  110. }
  111. const processDefinition = processDefinitionList.value.find(
  112. (item: any) => item.key == processInstance.processDefinition?.key
  113. )
  114. if (!processDefinition) {
  115. message.error('重新发起流程失败,原因:流程定义不存在')
  116. return
  117. }
  118. await handleSelect(processDefinition, processInstance.formVariables)
  119. }
  120. } finally {
  121. loading.value = false
  122. }
  123. }
  124. // 获取所有流程分类数据
  125. const getCategoryList = async () => {
  126. try {
  127. // 流程分类
  128. categoryList.value = await CategoryApi.getCategorySimpleList()
  129. if (categoryList.value.length > 0) {
  130. categoryActive.value = categoryList.value[0]
  131. }
  132. } finally {
  133. }
  134. }
  135. // 获取所有流程定义数据
  136. const getDefinitionList = async () => {
  137. try {
  138. // 流程定义
  139. processDefinitionList.value = await DefinitionApi.getProcessDefinitionList({
  140. suspensionState: 1
  141. })
  142. // 初始化过滤列表为全部流程定义
  143. filteredProcessDefinitionList.value = processDefinitionList.value
  144. } finally {
  145. }
  146. }
  147. const filteredProcessDefinitionList = ref([]) // 用于存储搜索过滤后的流程定义
  148. // 直接进行前端搜索
  149. const handleQuery = () => {
  150. if (currentSearchKey.value.trim()) {
  151. // 如果有搜索关键字,进行过滤
  152. filteredProcessDefinitionList.value = processDefinitionList.value.filter(
  153. (definition: any) =>
  154. definition.name.toLowerCase().includes(currentSearchKey.value.toLowerCase()) // 假设搜索依据是流程定义的名称
  155. )
  156. } else {
  157. // 如果没有搜索关键字,恢复所有数据
  158. filteredProcessDefinitionList.value = processDefinitionList.value
  159. }
  160. }
  161. // 监听input `clearable` 事件
  162. const handleClear = () => {
  163. filteredProcessDefinitionList.value = processDefinitionList.value
  164. }
  165. // 流程定义的分组
  166. const processDefinitionGroup: any = computed(() => {
  167. if (!processDefinitionList.value?.length) return {}
  168. return groupBy(filteredProcessDefinitionList.value, 'categoryName')
  169. })
  170. // ========== 表单相关 ==========
  171. const selectProcessDefinition = ref() // 选择的流程定义
  172. const processDefinitionDetailRef = ref()
  173. /** 处理选择流程的按钮操作 **/
  174. const handleSelect = async (row, formVariables?) => {
  175. // 设置选择的流程
  176. selectProcessDefinition.value = row
  177. // 初始化流程定义详情
  178. await nextTick()
  179. processDefinitionDetailRef.value?.initProcessInfo(row, formVariables)
  180. }
  181. // 左侧分类切换
  182. const handleCategoryClick = (category) => {
  183. categoryActive.value = category
  184. const categoryRef = proxy.$refs[`category-${category.name}`] // 获取点击分类对应的 DOM 元素
  185. if (categoryRef?.length) {
  186. const scrollWrapper = proxy.$refs.scrollWrapper // 获取右侧滚动容器
  187. const categoryOffsetTop = categoryRef[0].offsetTop
  188. // 滚动到对应位置
  189. scrollWrapper.scrollTo({ top: categoryOffsetTop, behavior: 'smooth' })
  190. }
  191. }
  192. /** 初始化 */
  193. onMounted(() => {
  194. getList()
  195. })
  196. </script>
  197. <style lang="scss" scoped>
  198. .process-definition-container::before {
  199. content: '';
  200. border-left: 1px solid #e6e6e6;
  201. position: absolute;
  202. left: 20.8%;
  203. height: 100%;
  204. }
  205. :deep() {
  206. .definition-item-card {
  207. .el-card__body {
  208. padding: 14px;
  209. }
  210. }
  211. }
  212. </style>