edit.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464
  1. <template>
  2. <el-dialog
  3. :title="title"
  4. :visible.sync="dialogVisible"
  5. :close-on-click-modal="false"
  6. :close-on-press-escape="false"
  7. append-to-body
  8. top="15vh"
  9. class="dataset-dialog"
  10. width="70%"
  11. @open="getFormData"
  12. @close="closeDialog"
  13. >
  14. <el-form
  15. ref="form"
  16. v-loading="dialogLoading"
  17. :element-loading-text="$t('common.loading')"
  18. :model="form"
  19. :rules="rules"
  20. :label-width="formLabelWidth"
  21. @submit.native.prevent
  22. >
  23. <el-row>
  24. <el-col :span="span">
  25. <el-form-item label="名称:" prop="name">
  26. <el-input v-if="!readonly" v-model="form.name" v-pinyin="{vm:form}" />
  27. <span v-else>{{ form.name }}</span>
  28. </el-form-item>
  29. </el-col>
  30. <el-col :span="span">
  31. <el-form-item label="数据集key:" prop="key">
  32. <el-input v-if="!readonly" v-model="form.key" :disabled="disabled" />
  33. <span v-else>{{ form.key }}</span>
  34. </el-form-item>
  35. </el-col>
  36. <el-col v-if="!readonly" :span="span">
  37. <el-form-item key="typeId" label="分类:" prop="typeId">
  38. <ibps-type
  39. ref="ibpsType"
  40. v-model="form.typeId"
  41. :readonly="readonly"
  42. category-key="BO_TYPE"
  43. class="type"
  44. filter-label="name"
  45. :filterable="true"
  46. />
  47. </el-form-item>
  48. </el-col>
  49. <!-- <el-col :span="12">
  50. <el-form-item label="外部数据:" prop="external">
  51. <el-switch
  52. v-if="!readonly"
  53. v-model="form.external"
  54. :active-value="'Y'"
  55. :inactive-value="'N'"
  56. active-text="是"
  57. inactive-text="否"
  58. />
  59. <el-tag v-else :type="form.external|optionsFilter(externalOptions,'type')">
  60. {{ form.external|optionsFilter(externalOptions,'label') }}
  61. </el-tag>
  62. </el-form-item>
  63. </el-col> -->
  64. <el-col v-if="form.external==='Y'" :span="12" style="height:50px;">
  65. <el-form-item label="数据源:" prop="dsAlias">
  66. <el-select
  67. v-if="!readonly"
  68. v-model="form.dsAlias"
  69. clearable
  70. placeholder="请选择"
  71. style="width:100%;"
  72. >
  73. <el-option
  74. v-for="item in dsAliasOptions"
  75. :key="item.value"
  76. :label="item.label"
  77. :value="item.value"
  78. />
  79. </el-select>
  80. <el-tag v-else :type="form.dsAlias|optionsFilter(externalOptions,'type')">
  81. {{ form.dsAlias|optionsFilter(externalOptions,'label') }}
  82. </el-tag>
  83. </el-form-item>
  84. </el-col>
  85. <el-col :span="span">
  86. <el-form-item label="类型:" prop="type">
  87. <el-row>
  88. <el-col :span="5">
  89. <el-select
  90. v-if="!readonly"
  91. v-model="form.type"
  92. placeholder="请选择"
  93. style="width:100%;"
  94. @change="changeType"
  95. >
  96. <el-option
  97. v-for="item in datasetTypeOptions"
  98. :key="item.value"
  99. :label="item.label"
  100. :value="item.value"
  101. />
  102. </el-select>
  103. <el-tag v-else :type="form.type|optionsFilter(datasetTypeOptions,'type')">{{ form.type|optionsFilter(datasetTypeOptions,'label') }}</el-tag>
  104. </el-col>
  105. <el-col :span="19">
  106. <el-form-item v-if="!readonly" prop="from" class="from">
  107. <el-select
  108. v-if="form.type!=='thirdparty'"
  109. v-model="form.from"
  110. filterable
  111. placeholder="请选择或者搜索关键字后选择"
  112. style="width:100%"
  113. @change="changeFrom"
  114. >
  115. <el-option
  116. v-for="item in fromOptions"
  117. :key="item.value"
  118. :label="item.label"
  119. :value="item.value"
  120. />
  121. </el-select>
  122. <ibps-tree-select
  123. v-else
  124. v-model="form.from"
  125. :data="treeData"
  126. :props="props"
  127. :icon="handleIcon"
  128. :multiple="false"
  129. :allow-selection="handleAllowSelection"
  130. node-key="key"
  131. empty-text="暂无第三方服务"
  132. clearable
  133. filterable
  134. />
  135. </el-form-item>
  136. </el-col>
  137. </el-row>
  138. <p v-if="!readonly&&form.type==='thirdparty'" class="ibps-m-0"><i class="el-icon-warning" style="font-size:12px;color:#E6A23C" /> <span style="font-size:12px;color:#E6A23C">带图标</span> <i class="ibps-icon ibps-icon-list" /> <span style="font-size:12px;color:#E6A23C">数据为目录不能用作第三方服务类型参数</span> </p>
  139. </el-form-item>
  140. </el-col>
  141. <template v-if="readonly">
  142. <el-col :span="span">
  143. <el-form-item key="from" label="来源:">
  144. <span>{{ form.from }}</span>
  145. </el-form-item>
  146. </el-col>
  147. <!-- <el-col :span="span">
  148. <el-form-item key="isTree" label="是否树型:">
  149. <span>{{ form.isTree|optionsFilter(externalOptions,'label') }}</span>
  150. </el-form-item>
  151. </el-col> -->
  152. <el-col :span="span">
  153. <el-form-item key="createBy" label="创建人:">
  154. <ibps-employee-selector
  155. :value="form.createBy"
  156. :disabled="true"
  157. class="dataset-readonly-selector"
  158. />
  159. </el-form-item>
  160. </el-col>
  161. <el-col :span="span">
  162. <el-form-item key="createTime" label="创建时间:">
  163. <span>{{ form.createTime }}</span>
  164. </el-form-item>
  165. </el-col>
  166. <el-col :span="span">
  167. <el-form-item key="updateBy" label="更新人:">
  168. <ibps-employee-selector
  169. :value="form.updateBy"
  170. :disabled="true"
  171. class="dataset-readonly-selector"
  172. />
  173. </el-form-item>
  174. </el-col>
  175. <el-col :span="span">
  176. <el-form-item key="updateTime" label="更新时间:">
  177. <span>{{ form.updateTime }}</span>
  178. </el-form-item>
  179. </el-col>
  180. </template>
  181. </el-row>
  182. </el-form>
  183. <div slot="footer" class="el-dialog--center">
  184. <ibps-toolbar
  185. :actions="toolbars"
  186. @action-event="handleActionEvent"
  187. />
  188. </div>
  189. </el-dialog>
  190. </template>
  191. <script>
  192. import IbpsEmployeeSelector from '@/business/platform/org/employee/selector'
  193. import { save, get, getTableOrViewList } from '@/api/platform/data/dataset'
  194. import { datasetTypeOptions } from '@/business/platform/data/constants'
  195. import { externalOptions, dsAliasOptions } from './constants'
  196. import { validateKey } from '@/utils/validate'
  197. import IbpsType from '@/business/platform/cat/type/select'
  198. import ActionUtils from '@/utils/action'
  199. import { findTreeData } from '@/api/platform/serv/service'
  200. import TreeUtils from '@/utils/tree'
  201. import IbpsTreeSelect from '@/components/ibps-tree-select'
  202. export default {
  203. components: {
  204. IbpsType,
  205. IbpsTreeSelect,
  206. IbpsEmployeeSelector
  207. },
  208. props: {
  209. visible: {
  210. type: Boolean,
  211. default: false
  212. },
  213. readonly: {
  214. type: Boolean,
  215. default: false
  216. },
  217. id: String,
  218. title: String,
  219. typeId: String
  220. },
  221. data () {
  222. return {
  223. formName: 'form',
  224. formLabelWidth: '120px',
  225. span: 24,
  226. dialogVisible: this.visible,
  227. dialogLoading: false,
  228. dsAlias: false,
  229. disabled: false,
  230. datasetTypeOptions,
  231. externalOptions: externalOptions,
  232. dsAliasOptions: dsAliasOptions,
  233. fromOptions: [],
  234. defaultForm: {},
  235. props: {
  236. children: 'children',
  237. label: 'name'
  238. },
  239. treeData: [],
  240. form: {
  241. key: '',
  242. name: '',
  243. typeId: '',
  244. external: 'N',
  245. type: 'table',
  246. isTree: 'N',
  247. from: '',
  248. dsAlias: 'dataSource_default'
  249. },
  250. rules: {
  251. key: [{ required: true, validator: validateKey },
  252. { min: 1, max: 64, message: '长度不能超过64个字符', trigger: 'blur' }],
  253. name: [{ required: true, message: this.$t('validate.required') },
  254. { min: 1, max: 64, message: '长度不能超过64个字符', trigger: 'blur' }],
  255. from: [{ required: true, message: this.$t('validate.required') }],
  256. external: [{ required: true, message: this.$t('validate.required') }],
  257. type: [{ required: true, message: this.$t('validate.required') }]
  258. },
  259. toolbars: [
  260. { key: 'save', hidden: () => { return this.readonly } },
  261. { key: 'cancel' }
  262. ]
  263. }
  264. },
  265. computed: {
  266. formId () {
  267. return this.id
  268. }
  269. },
  270. watch: {
  271. visible: {
  272. handler: function (val, oldVal) {
  273. this.dialogVisible = this.visible
  274. },
  275. immediate: true
  276. }
  277. },
  278. created () {
  279. this.defaultForm = JSON.parse(JSON.stringify(this.form))
  280. },
  281. methods: {
  282. handleActionEvent ({ key }) {
  283. switch (key) {
  284. case 'save':
  285. this.handleSave()
  286. break
  287. case 'cancel':
  288. this.closeDialog()
  289. break
  290. default:
  291. break
  292. }
  293. },
  294. // 保存数据
  295. handleSave () {
  296. this.$refs[this.formName].validate(valid => {
  297. if (valid) {
  298. this.saveData()
  299. } else {
  300. ActionUtils.saveErrorMessage()
  301. }
  302. })
  303. },
  304. // 提交保存数据
  305. saveData () {
  306. const data = JSON.parse(JSON.stringify(this.form))
  307. save(data).then(response => {
  308. this.$emit('callback', this)
  309. ActionUtils.saveSuccessMessage(response.message, (rtn) => {
  310. if (this.$utils.isEmpty(this.formId)) {
  311. this.$refs[this.formName].resetFields()
  312. }
  313. if (rtn) {
  314. this.closeDialog()
  315. } else {
  316. this.formValidate()
  317. }
  318. })
  319. }).catch((err) => {
  320. console.error(err)
  321. })
  322. },
  323. // 关闭当前窗口
  324. closeDialog () {
  325. this.$emit('close', false)
  326. this.$refs[this.formName].resetFields()
  327. },
  328. /**
  329. * 表单验证
  330. */
  331. formValidate () {
  332. if (this.readonly) return
  333. this.$nextTick(() => {
  334. this.$refs[this.formName].validate(() => {})
  335. })
  336. },
  337. changeType (type) {
  338. this.form.from = ''
  339. this.fromOptions = []
  340. this.loadDataTableOrView(type)
  341. },
  342. async changeFrom (value) {
  343. // 新建数据集时,不可与已有数据集共用数据表,编辑数据集时,不可自身之外的数据集共用数据表
  344. const params = this.$utils.isNotEmpty(this.form.id) ? ` and id_ != '${this.form.id}'` : ''
  345. const sql = `select * from ibps_data_dataset where from_ = '${value}'${params}`
  346. const res = await this.$common.request('sql', sql)
  347. const { data = [] } = res.variables || {}
  348. if (data.length) {
  349. this.form.from = ''
  350. return this.$message.warning('已存在相同来源数据集,请勿重复创建!')
  351. }
  352. },
  353. // 子集Key数据获取
  354. loadDataTableOrView (type = 'table') {
  355. if (this.$utils.isEmpty(this.formId)) {
  356. this.form.from = ''
  357. this.fromOptions = []
  358. }
  359. if (type !== 'thirdparty') {
  360. getTableOrViewList({
  361. from: '',
  362. external: this.form.external,
  363. dsAlias: this.form.dsAlias || 'dataSource_default',
  364. type: type
  365. }).then(response => {
  366. const data = response.data
  367. this.fromOptions = data.map(item => {
  368. return {
  369. value: item.id,
  370. label: item.text + '【' + item.comment + '】'
  371. }
  372. })
  373. }).catch(() => {})
  374. } else {
  375. findTreeData().then(response => {
  376. const data = response.data
  377. const treeData = []
  378. data.forEach(d => {
  379. if (d.parentId !== '-1') {
  380. if (d.isDir === 'Y') {
  381. d.disabled = true
  382. }
  383. treeData.push(d)
  384. }
  385. })
  386. this.treeData = TreeUtils.transformToTreeFormat(treeData)
  387. })
  388. }
  389. },
  390. handleIcon (data) {
  391. let icon = ''
  392. if (data.isDir === 'Y') {
  393. icon = 'ibps-icon ibps-icon-list'
  394. }
  395. return icon
  396. },
  397. handleAllowSelection (data) {
  398. if (data.isDir === 'Y') {
  399. this.$message.warning('数据为目录不能用作第三方服务参数,请重新选择!')
  400. return false
  401. }
  402. return true
  403. },
  404. /**
  405. * 获取表单数据
  406. */
  407. getFormData () {
  408. this.$nextTick(() => {
  409. if (this.$refs['ibpsType']) {
  410. this.$refs['ibpsType'].loadTreeData()
  411. }
  412. })
  413. if (this.$utils.isEmpty(this.formId)) {
  414. // 重置表单
  415. this.disabled = false
  416. this.form = JSON.parse(JSON.stringify(this.defaultForm))
  417. this.form.typeId = this.typeId
  418. this.span = 24
  419. this.loadDataTableOrView()
  420. this.formValidate()
  421. return
  422. }
  423. this.dialogLoading = true
  424. get({
  425. datasetId: this.formId
  426. }).then(response => {
  427. this.span = 24
  428. if (this.readonly) {
  429. this.span = 12
  430. this.dsAlias = false
  431. }
  432. this.loadDataTableOrView(response.data.type)
  433. this.disabled = true
  434. this.form = response.data
  435. this.formValidate()
  436. this.dialogLoading = false
  437. }).catch(() => {
  438. this.dialogLoading = false
  439. })
  440. }
  441. }
  442. }
  443. </script>
  444. <style lang="scss">
  445. .dataset-dialog{
  446. .el-tree-select{
  447. line-height: 28px;
  448. }
  449. .dataset-readonly-selector{
  450. .is-disabled{
  451. input{
  452. display:none;
  453. }
  454. }
  455. }
  456. }
  457. </style>