relation.vue 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. <template>
  2. <CusDialog
  3. :show="show"
  4. title="数据构成"
  5. @onClose="$emit('update:show', false)"
  6. width="90%"
  7. height="90%"
  8. :show-submit="false"
  9. :loading="state.loading"
  10. >
  11. <div class="__cus-manage_content">
  12. <div class="__cus-manage_content-filters" v-loading="state.loadingIndex">
  13. <el-row>
  14. <CusFormColumn
  15. :span="2"
  16. v-model:param="state.type"
  17. link="select"
  18. :options="DictionaryStore.serviceTypeList"
  19. @change="initIndex"
  20. :clearable="false"
  21. />
  22. <el-col :span="22">
  23. <el-autocomplete
  24. ref="ref_search"
  25. v-model="state.indexSelect"
  26. :fetch-suggestions="fetchSuggestions"
  27. clearable
  28. placeholder="搜索添加数据"
  29. @select="handleSelect"
  30. :select-when-unmatched="true"
  31. >
  32. <template #default="{ item }">
  33. <div class="index-item" v-html="item.html"/>
  34. </template>
  35. </el-autocomplete>
  36. </el-col>
  37. </el-row>
  38. </div>
  39. <div class="__cus-manage_content-main" v-loading="state.query.loading">
  40. <CusTable
  41. :data="state.query.result.data"
  42. :table-head="state.query.tableHead"
  43. :no-page="true"
  44. >
  45. <template #type-column-value="{scope}">
  46. {{DictionaryStore.serviceTypeMap.get(scope.row.type)}}
  47. </template>
  48. <template #isMain-column-value="{scope}">
  49. {{DictionaryStore.isMainIndexMap.get(scope.row.isMain)}}
  50. </template>
  51. <template #do-column-value="{scope}">
  52. <CusButton v-if="scope.row.isMain == 1" type="table-add" icon="text" title="主配置" @click="onMain(scope.row)"/>
  53. <CusButton type="table-add" icon="text" title="列配置" @click="onText(scope.row)"/>
  54. <CusButton type="table-edit" @click="onEdit(scope.row)"/>
  55. <CusButton type="table-del" @click="onDel(scope.row)"/>
  56. </template>
  57. </CusTable>
  58. </div>
  59. </div>
  60. <RelationDetailCom v-model:show="state.relationDetail.show" :transfer="state.relationDetail.transfer" @refresh="initRelation"/>
  61. <RelationColumnCom v-model:show="state.relationColumn.show" :transfer="state.relationColumn.transfer" @refresh="initRelation"/>
  62. <RelationMainCom v-model:show="state.relationMain.show" :transfer="state.relationMain.transfer" @refresh="initRelation"/>
  63. </CusDialog>
  64. </template>
  65. <script setup lang="ts">
  66. import {computed, getCurrentInstance, nextTick, reactive, ref, watch} from "vue";
  67. import {useDictionaryStore} from "@/stores";
  68. import {ElMessage, ElMessageBox} from "element-plus";
  69. import RelationDetailCom from "./relation-detail.vue";
  70. import RelationColumnCom from "./relation-column.vue";
  71. import RelationMainCom from "./relation-main.vue";
  72. import {
  73. sysLabelDeleteLink,
  74. sysLabelGetAllIndexsByKey
  75. } from "@/api/modules/manage/type";
  76. import {sysThemeFind, sysThemeIndexDelete, sysThemeIndexFindAll} from "@/api/modules/manage/theme";
  77. const emit = defineEmits(['update:show', 'refresh'])
  78. const {proxy} = getCurrentInstance()
  79. const DictionaryStore = useDictionaryStore()
  80. const props = defineProps({
  81. show: {default: false},
  82. transfer: {}
  83. })
  84. const state: any = reactive({
  85. loading: false,
  86. type: 'inner_index', // outside_service
  87. query: {
  88. loading: false,
  89. tableHead: [
  90. {value: "type", label: "数据类型", width: 120},
  91. {value: "indexTableName", label: "数据英文", popover: true},
  92. {value: "indexName", label: "数据名称"},
  93. {value: "indexNameShort", label: "数据简称"},
  94. {value: "isMain", label: "主数据"},
  95. {value: "weight", label: "权重"},
  96. {value: "fieldName", label: "关联数据"},
  97. {value: "description", label: "关系描述"},
  98. {value: "do", label: "操作", width: 360, fixed: 'right'},
  99. ],
  100. result: {
  101. data: []
  102. }
  103. },
  104. form: {},
  105. loadingIndex: false,
  106. indexList: [],
  107. indexSelect: '',
  108. relationDetail: {
  109. transfer: {},
  110. show: false
  111. },
  112. relationColumn: {
  113. transfer: {},
  114. show: false
  115. },
  116. relationMain: {
  117. transfer: {},
  118. show: false
  119. }
  120. })
  121. const ref_search = ref()
  122. const fetchSuggestions = (queryString: string, cb: any) => {
  123. const arr = state.indexList.filter(v => state.query.result.data.every(s => s.indexId != v.id))
  124. const results = queryString
  125. ? arr.filter(v => [v.indexName, v.indexTableName].some(s => s?.includes(queryString)))
  126. : arr
  127. results.map(v => {
  128. const str1 = queryString ? v.indexName.replace(new RegExp(queryString, 'g'), `<em>${queryString}</em>`) : v.indexName
  129. const str2 = queryString ? v.indexTableName.replace(new RegExp(queryString, 'g'), `<em>${queryString}</em>`) : v.indexTableName
  130. v.html = `${str1}<span style="margin-right: 20px;"></span>${str2}`
  131. return v
  132. })
  133. cb(results)
  134. }
  135. const handleSelect = (item) => {
  136. state.relationDetail.transfer = {
  137. type: state.type,
  138. mode: 'add',
  139. indexId: item.id,
  140. indexTableName: item.indexTableName,
  141. indexName: item.indexName,
  142. themeId: props.transfer.id,
  143. conditionOptions: state.form.themeParam,
  144. relationOptions: state.query.result.data.map(v => ({dictLabel: v.indexName, dictValue: v.id})),
  145. hasMain: state.query.result.data.some(v => v.isMain == 1) // || state.type !== 'inner_index'
  146. }
  147. state.relationDetail.show = true
  148. }
  149. const initIndex = () => {
  150. state.loadingIndex = true
  151. setTimeout(() => {
  152. sysLabelGetAllIndexsByKey(proxy.$util.formatGetParam({type: state.type})).then(res => {
  153. state.indexList = res.data
  154. state.loadingIndex = false
  155. })
  156. }, 100)
  157. }
  158. const initRelation = () => {
  159. state.query.loading = true
  160. sysThemeIndexFindAll(props.transfer.id).then(res => {
  161. state.query.result.data = res.data
  162. state.query.loading = false
  163. })
  164. }
  165. const onMain = (row) => {
  166. state.relationMain.transfer = {
  167. indexId: row.indexId,
  168. id: row.id,
  169. indexStyle: row.indexStyle,
  170. type: row.type,
  171. }
  172. state.relationMain.show = true
  173. }
  174. const onText = (row) => {
  175. state.relationColumn.transfer = {
  176. indexId: row.indexId,
  177. themeId: props.transfer.id,
  178. type: row.type
  179. }
  180. state.relationColumn.show = true
  181. }
  182. const onEdit = (row) => {
  183. let hasMain = false
  184. state.query.result.data.forEach(v => {
  185. if (v.isMain == 1) {
  186. if (v.id != row.id) {
  187. hasMain = true
  188. }
  189. }
  190. })
  191. state.relationDetail.transfer = {
  192. type: row.type,
  193. mode: 'edit',
  194. id: row.id,
  195. indexTableName: row.indexTableName,
  196. indexName: row.indexName,
  197. conditionOptions: state.form.themeParam,
  198. relationOptions: state.query.result.data.filter(v => v.id !== row.id).map(v => ({dictLabel: v.indexName, dictValue: v.id})),
  199. hasMain: hasMain // || row.type !== 'inner_index'
  200. }
  201. state.relationDetail.show = true
  202. }
  203. const onDel = (row) => {
  204. ElMessageBox.confirm(`请确认是否删除${row.indexName}?`, "提示", {
  205. confirmButtonText: "确定",
  206. cancelButtonText: "取消",
  207. type: "warning",
  208. } as any).then(() => {
  209. state.loading = true
  210. sysThemeIndexDelete(row.id).then(res => {
  211. ElMessage.success('删除成功!')
  212. emit('refresh')
  213. state.loading = false
  214. initRelation()
  215. })
  216. }).catch(() => {})
  217. }
  218. const initDetail = () => {
  219. state.loading = true
  220. sysThemeFind(props.transfer.id).then(res => {
  221. state.form = res.data
  222. if (state.form.themeType == '2') {
  223. state.form.themeParam = state.form.themeParam.split(',')
  224. }
  225. state.loading = false
  226. })
  227. }
  228. watch(() => props.show, (n) => {
  229. initDictionary()
  230. if (n) {
  231. initIndex()
  232. initRelation()
  233. initDetail()
  234. }
  235. })
  236. const initDictionary = () => {
  237. DictionaryStore.initDict('is_main_index')
  238. DictionaryStore.initDict('service_type')
  239. }
  240. </script>
  241. <style lang="scss" scoped>
  242. .__cus-manage_content {
  243. margin-bottom: 0;
  244. height: calc(100% - 24px);
  245. }
  246. </style>