ElementForm.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463
  1. <template>
  2. <div class="panel-tab__content">
  3. <el-form label-width="80px">
  4. <el-form-item label="表单标识">
  5. <el-input v-model="formKey" clearable @change="updateElementFormKey" />
  6. </el-form-item>
  7. <el-form-item label="业务标识">
  8. <el-select v-model="businessKey" @change="updateElementBusinessKey">
  9. <el-option v-for="i in fieldList" :key="i.id" :value="i.id" :label="i.label" />
  10. <el-option label="无" value="" />
  11. </el-select>
  12. </el-form-item>
  13. </el-form>
  14. <!--字段列表-->
  15. <div class="element-property list-property">
  16. <el-divider><Icon icon="ep:coin" /> 表单字段</el-divider>
  17. <el-table :data="fieldList" max-height="240" border fit>
  18. <el-table-column label="序号" type="index" width="50px" />
  19. <el-table-column label="字段名称" prop="label" min-width="80px" show-overflow-tooltip />
  20. <el-table-column
  21. label="字段类型"
  22. prop="type"
  23. min-width="80px"
  24. :formatter="(row) => fieldType[row.type] || row.type"
  25. show-overflow-tooltip
  26. />
  27. <el-table-column
  28. label="默认值"
  29. prop="defaultValue"
  30. min-width="80px"
  31. show-overflow-tooltip
  32. />
  33. <el-table-column label="操作" width="90px">
  34. <template #default="scope">
  35. <el-button type="primary" link @click="openFieldForm(scope, scope.$index)"
  36. >编辑</el-button
  37. >
  38. <el-divider direction="vertical" />
  39. <el-button
  40. type="primary"
  41. link
  42. style="color: #ff4d4f"
  43. @click="removeField(scope, scope.$index)"
  44. >移除</el-button
  45. >
  46. </template>
  47. </el-table-column>
  48. </el-table>
  49. </div>
  50. <div class="element-drawer__button">
  51. <XButton type="primary" proIcon="ep:plus" title="添加字段" @click="openFieldForm(null, -1)" />
  52. </div>
  53. <!--字段配置侧边栏-->
  54. <el-drawer
  55. v-model="fieldModelVisible"
  56. title="字段配置"
  57. :size="`${width}px`"
  58. append-to-body
  59. destroy-on-close
  60. >
  61. <el-form :model="formFieldForm" label-width="90px">
  62. <el-form-item label="字段ID">
  63. <el-input v-model="formFieldForm.id" clearable />
  64. </el-form-item>
  65. <el-form-item label="类型">
  66. <el-select
  67. v-model="formFieldForm.typeType"
  68. placeholder="请选择字段类型"
  69. clearable
  70. @change="changeFieldTypeType"
  71. >
  72. <el-option v-for="(value, key) of fieldType" :label="value" :value="key" :key="key" />
  73. </el-select>
  74. </el-form-item>
  75. <el-form-item label="类型名称" v-if="formFieldForm.typeType === 'custom'">
  76. <el-input v-model="formFieldForm.type" clearable />
  77. </el-form-item>
  78. <el-form-item label="名称">
  79. <el-input v-model="formFieldForm.label" clearable />
  80. </el-form-item>
  81. <el-form-item label="时间格式" v-if="formFieldForm.typeType === 'date'">
  82. <el-input v-model="formFieldForm.datePattern" clearable />
  83. </el-form-item>
  84. <el-form-item label="默认值">
  85. <el-input v-model="formFieldForm.defaultValue" clearable />
  86. </el-form-item>
  87. </el-form>
  88. <!-- 枚举值设置 -->
  89. <template v-if="formFieldForm.type === 'enum'">
  90. <el-divider key="enum-divider" />
  91. <p class="listener-filed__title" key="enum-title">
  92. <span><Icon icon="ep:menu" />枚举值列表:</span>
  93. <el-button type="primary" @click="openFieldOptionForm(null, -1, 'enum')"
  94. >添加枚举值</el-button
  95. >
  96. </p>
  97. <el-table :data="fieldEnumList" key="enum-table" max-height="240" border fit>
  98. <el-table-column label="序号" width="50px" type="index" />
  99. <el-table-column label="枚举值编号" prop="id" min-width="100px" show-overflow-tooltip />
  100. <el-table-column label="枚举值名称" prop="name" min-width="100px" show-overflow-tooltip />
  101. <el-table-column label="操作" width="90px">
  102. <template #default="scope">
  103. <el-button
  104. type="primary"
  105. link
  106. @click="openFieldOptionForm(scope, scope.$index, 'enum')"
  107. >编辑</el-button
  108. >
  109. <el-divider direction="vertical" />
  110. <el-button
  111. type="primary"
  112. link
  113. style="color: #ff4d4f"
  114. @click="removeFieldOptionItem(scope, scope.$index, 'enum')"
  115. >移除</el-button
  116. >
  117. </template>
  118. </el-table-column>
  119. </el-table>
  120. </template>
  121. <!-- 校验规则 -->
  122. <el-divider key="validation-divider" />
  123. <p class="listener-filed__title" key="validation-title">
  124. <span><Icon icon="ep:menu" />约束条件列表:</span>
  125. <el-button type="primary" @click="openFieldOptionForm(null, -1, 'constraint')"
  126. >添加约束</el-button
  127. >
  128. </p>
  129. <el-table :data="fieldConstraintsList" key="validation-table" max-height="240" border fit>
  130. <el-table-column label="序号" width="50px" type="index" />
  131. <el-table-column label="约束名称" prop="name" min-width="100px" show-overflow-tooltip />
  132. <el-table-column label="约束配置" prop="config" min-width="100px" show-overflow-tooltip />
  133. <el-table-column label="操作" width="90px">
  134. <template #default="scope">
  135. <el-button
  136. type="primary"
  137. link
  138. @click="openFieldOptionForm(scope, scope.$index, 'constraint')"
  139. >编辑</el-button
  140. >
  141. <el-divider direction="vertical" />
  142. <el-button
  143. type="primary"
  144. link
  145. style="color: #ff4d4f"
  146. @click="removeFieldOptionItem(scope, scope.$index, 'constraint')"
  147. >移除</el-button
  148. >
  149. </template>
  150. </el-table-column>
  151. </el-table>
  152. <!-- 表单属性 -->
  153. <el-divider key="property-divider" />
  154. <p class="listener-filed__title" key="property-title">
  155. <span><Icon icon="ep:menu" />字段属性列表:</span>
  156. <el-button type="primary" @click="openFieldOptionForm(null, -1, 'property')"
  157. >添加属性</el-button
  158. >
  159. </p>
  160. <el-table :data="fieldPropertiesList" key="property-table" max-height="240" border fit>
  161. <el-table-column label="序号" width="50px" type="index" />
  162. <el-table-column label="属性编号" prop="id" min-width="100px" show-overflow-tooltip />
  163. <el-table-column label="属性值" prop="value" min-width="100px" show-overflow-tooltip />
  164. <el-table-column label="操作" width="90px">
  165. <template #default="scope">
  166. <el-button
  167. type="primary"
  168. link
  169. @click="openFieldOptionForm(scope, scope.$index, 'property')"
  170. >编辑</el-button
  171. >
  172. <el-divider direction="vertical" />
  173. <el-button
  174. type="primary"
  175. link
  176. style="color: #ff4d4f"
  177. @click="removeFieldOptionItem(scope, scope.$index, 'property')"
  178. >移除</el-button
  179. >
  180. </template>
  181. </el-table-column>
  182. </el-table>
  183. <!-- 底部按钮 -->
  184. <div class="element-drawer__button">
  185. <el-button>取 消</el-button>
  186. <el-button type="primary" @click="saveField">保 存</el-button>
  187. </div>
  188. </el-drawer>
  189. <el-dialog
  190. v-model="fieldOptionModelVisible"
  191. :title="optionModelTitle"
  192. width="600px"
  193. append-to-body
  194. destroy-on-close
  195. >
  196. <el-form :model="fieldOptionForm" label-width="96px">
  197. <el-form-item label="编号/ID" v-if="fieldOptionType !== 'constraint'" key="option-id">
  198. <el-input v-model="fieldOptionForm.id" clearable />
  199. </el-form-item>
  200. <el-form-item label="名称" v-if="fieldOptionType !== 'property'" key="option-name">
  201. <el-input v-model="fieldOptionForm.name" clearable />
  202. </el-form-item>
  203. <el-form-item label="配置" v-if="fieldOptionType === 'constraint'" key="option-config">
  204. <el-input v-model="fieldOptionForm.config" clearable />
  205. </el-form-item>
  206. <el-form-item label="值" v-if="fieldOptionType === 'property'" key="option-value">
  207. <el-input v-model="fieldOptionForm.value" clearable />
  208. </el-form-item>
  209. </el-form>
  210. <template #footer>
  211. <el-button @click="fieldOptionModelVisible = false">取 消</el-button>
  212. <el-button type="primary" @click="saveFieldOption">确 定</el-button>
  213. </template>
  214. </el-dialog>
  215. </div>
  216. </template>
  217. <script setup lang="ts" name="ElementForm">
  218. const props = defineProps({
  219. id: String,
  220. type: String
  221. })
  222. const prefix = inject('prefix')
  223. const width = inject('width')
  224. const formKey = ref('')
  225. const businessKey = ref('')
  226. const optionModelTitle = ref('')
  227. const fieldList = ref<any[]>([])
  228. const formFieldForm = ref<any>({})
  229. const fieldType = ref({
  230. long: '长整型',
  231. string: '字符串',
  232. boolean: '布尔类',
  233. date: '日期类',
  234. enum: '枚举类',
  235. custom: '自定义类型'
  236. })
  237. const formFieldIndex = ref(-1) // 编辑中的字段, -1 为新增
  238. const formFieldOptionIndex = ref(-1) // 编辑中的字段配置项, -1 为新增
  239. const fieldModelVisible = ref(false)
  240. const fieldOptionModelVisible = ref(false)
  241. const fieldOptionForm = ref<any>({}) // 当前激活的字段配置项数据
  242. const fieldOptionType = ref('') // 当前激活的字段配置项弹窗 类型
  243. const fieldEnumList = ref<any[]>([]) // 枚举值列表
  244. const fieldConstraintsList = ref<any[]>([]) // 约束条件列表
  245. const fieldPropertiesList = ref<any[]>([]) // 绑定属性列表
  246. const bpmnELement = ref()
  247. const elExtensionElements = ref()
  248. const formData = ref()
  249. const otherExtensions = ref()
  250. const bpmnInstances = () => (window as any)?.bpmnInstances
  251. const resetFormList = () => {
  252. bpmnELement.value = bpmnInstances().bpmnElement
  253. formKey.value = bpmnELement.value.businessObject.formKey
  254. // 获取元素扩展属性 或者 创建扩展属性
  255. elExtensionElements.value =
  256. bpmnELement.value.businessObject.get('extensionElements') ||
  257. bpmnInstances().moddle.create('bpmn:ExtensionElements', { values: [] })
  258. // 获取元素表单配置 或者 创建新的表单配置
  259. formData.value =
  260. elExtensionElements.value.values.filter((ex) => ex.$type === `${prefix}:FormData`)?.[0] ||
  261. bpmnInstances().moddle.create(`${prefix}:FormData`, { fields: [] })
  262. // 业务标识 businessKey, 绑定在 formData 中
  263. businessKey.value = formData.value.businessKey
  264. // 保留剩余扩展元素,便于后面更新该元素对应属性
  265. otherExtensions.value = elExtensionElements.value.values.filter(
  266. (ex) => ex.$type !== `${prefix}:FormData`
  267. )
  268. // 复制原始值,填充表格
  269. fieldList.value = JSON.parse(JSON.stringify(formData.value.fields || []))
  270. // 更新元素扩展属性,避免后续报错
  271. updateElementExtensions()
  272. }
  273. const updateElementFormKey = () => {
  274. bpmnInstances().modeling.updateProperties(toRaw(bpmnELement.value), {
  275. formKey: formKey.value
  276. })
  277. }
  278. const updateElementBusinessKey = () => {
  279. bpmnInstances().modeling.updateModdleProperties(toRaw(bpmnELement.value), formData.value, {
  280. businessKey: businessKey.value
  281. })
  282. }
  283. // 根据类型调整字段type
  284. const changeFieldTypeType = (type) => {
  285. // this.$set(this.formFieldForm, "type", type === "custom" ? "" : type);
  286. formFieldForm.value['type'] = type === 'custom' ? '' : type
  287. }
  288. // 打开字段详情侧边栏
  289. const openFieldForm = (field, index) => {
  290. formFieldIndex.value = index
  291. if (index !== -1) {
  292. const FieldObject = formData.value.fields[index]
  293. formFieldForm.value = JSON.parse(JSON.stringify(field))
  294. // 设置自定义类型
  295. // this.$set(this.formFieldForm, "typeType", !this.fieldType[field.type] ? "custom" : field.type);
  296. formFieldForm.value['typeType'] = !fieldType.value[field.type] ? 'custom' : field.type
  297. // 初始化枚举值列表
  298. field.type === 'enum' &&
  299. (fieldEnumList.value = JSON.parse(JSON.stringify(FieldObject?.values || [])))
  300. // 初始化约束条件列表
  301. fieldConstraintsList.value = JSON.parse(
  302. JSON.stringify(FieldObject?.validation?.constraints || [])
  303. )
  304. // 初始化自定义属性列表
  305. fieldPropertiesList.value = JSON.parse(JSON.stringify(FieldObject?.properties?.values || []))
  306. } else {
  307. formFieldForm.value = {}
  308. // 初始化枚举值列表
  309. fieldEnumList.value = []
  310. // 初始化约束条件列表
  311. fieldConstraintsList.value = []
  312. // 初始化自定义属性列表
  313. fieldPropertiesList.value = []
  314. }
  315. fieldModelVisible.value = true
  316. }
  317. // 打开字段 某个 配置项 弹窗
  318. const openFieldOptionForm = (option, index, type) => {
  319. fieldOptionModelVisible.value = true
  320. fieldOptionType.value = type
  321. formFieldOptionIndex.value = index
  322. if (type === 'property') {
  323. fieldOptionForm.value = option ? JSON.parse(JSON.stringify(option)) : {}
  324. return (optionModelTitle.value = '属性配置')
  325. }
  326. if (type === 'enum') {
  327. fieldOptionForm.value = option ? JSON.parse(JSON.stringify(option)) : {}
  328. return (optionModelTitle.value = '枚举值配置')
  329. }
  330. fieldOptionForm.value = option ? JSON.parse(JSON.stringify(option)) : {}
  331. return (optionModelTitle.value = '约束条件配置')
  332. }
  333. // 保存字段 某个 配置项
  334. const saveFieldOption = () => {
  335. if (formFieldOptionIndex.value === -1) {
  336. if (fieldOptionType.value === 'property') {
  337. fieldPropertiesList.value.push(fieldOptionForm.value)
  338. }
  339. if (fieldOptionType.value === 'constraint') {
  340. fieldConstraintsList.value.push(fieldOptionForm.value)
  341. }
  342. if (fieldOptionType.value === 'enum') {
  343. fieldEnumList.value.push(fieldOptionForm.value)
  344. }
  345. } else {
  346. fieldOptionType.value === 'property' &&
  347. fieldPropertiesList.value.splice(formFieldOptionIndex.value, 1, fieldOptionForm.value)
  348. fieldOptionType.value === 'constraint' &&
  349. fieldConstraintsList.value.splice(formFieldOptionIndex.value, 1, fieldOptionForm.value)
  350. fieldOptionType.value === 'enum' &&
  351. fieldEnumList.value.splice(formFieldOptionIndex.value, 1, fieldOptionForm.value)
  352. }
  353. fieldOptionModelVisible.value = false
  354. fieldOptionForm.value = {}
  355. }
  356. // 保存字段配置
  357. const saveField = () => {
  358. const { id, type, label, defaultValue, datePattern } = formFieldForm.value
  359. const Field = bpmnInstances().moddle.create(`${prefix}:FormField`, { id, type, label })
  360. defaultValue && (Field.defaultValue = defaultValue)
  361. datePattern && (Field.datePattern = datePattern)
  362. // 构建属性
  363. if (fieldPropertiesList.value && fieldPropertiesList.value.length) {
  364. const fieldPropertyList = fieldPropertiesList.value.map((fp) => {
  365. return bpmnInstances().moddle.create(`${prefix}:Property`, {
  366. id: fp.id,
  367. value: fp.value
  368. })
  369. })
  370. Field.properties = bpmnInstances().moddle.create(`${prefix}:Properties`, {
  371. values: fieldPropertyList
  372. })
  373. }
  374. // 构建校验规则
  375. if (fieldConstraintsList.value && fieldConstraintsList.value.length) {
  376. const fieldConstraintList = fieldConstraintsList.value.map((fc) => {
  377. return bpmnInstances().moddle.create(`${prefix}:Constraint`, {
  378. name: fc.name,
  379. config: fc.config
  380. })
  381. })
  382. Field.validation = bpmnInstances().moddle.create(`${prefix}:Validation`, {
  383. constraints: fieldConstraintList
  384. })
  385. }
  386. // 构建枚举值
  387. if (fieldEnumList.value && fieldEnumList.value.length) {
  388. Field.values = fieldEnumList.value.map((fe) => {
  389. return bpmnInstances().moddle.create(`${prefix}:Value`, { name: fe.name, id: fe.id })
  390. })
  391. }
  392. // 更新数组 与 表单配置实例
  393. if (formFieldIndex.value === -1) {
  394. fieldList.value.push(formFieldForm.value)
  395. formData.value.fields.push(Field)
  396. } else {
  397. fieldList.value.splice(formFieldIndex.value, 1, formFieldForm.value)
  398. formData.value.fields.splice(formFieldIndex.value, 1, Field)
  399. }
  400. updateElementExtensions()
  401. fieldModelVisible.value = false
  402. }
  403. // 移除某个 字段的 配置项
  404. const removeFieldOptionItem = (option, index, type) => {
  405. console.log(option, 'option')
  406. if (type === 'property') {
  407. fieldPropertiesList.value.splice(index, 1)
  408. return
  409. }
  410. if (type === 'enum') {
  411. fieldEnumList.value.splice(index, 1)
  412. return
  413. }
  414. fieldConstraintsList.value.splice(index, 1)
  415. }
  416. // 移除 字段
  417. const removeField = (field, index) => {
  418. console.log(field, 'field')
  419. fieldList.value.splice(index, 1)
  420. formData.value.fields.splice(index, 1)
  421. updateElementExtensions()
  422. }
  423. const updateElementExtensions = () => {
  424. // 更新回扩展元素
  425. const newElExtensionElements = bpmnInstances().moddle.create(`bpmn:ExtensionElements`, {
  426. values: otherExtensions.value.concat(formData.value)
  427. })
  428. // 更新到元素上
  429. bpmnInstances().modeling.updateProperties(toRaw(bpmnELement.value), {
  430. extensionElements: newElExtensionElements
  431. })
  432. }
  433. watch(
  434. () => props.id,
  435. (val) => {
  436. val &&
  437. val.length &&
  438. nextTick(() => {
  439. resetFormList()
  440. })
  441. },
  442. { immediate: true }
  443. )
  444. </script>