relation.vue 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  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 v-if="scope.row.type === 'inner_index'" 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: "indexName", label: "索引名称"},
  99. // {value: "indexNameShort", label: "索引简称"},
  100. // {value: "isMain", label: "主索引"},
  101. // {value: "fieldName", label: "关联索引"},
  102. // {value: "weight", label: "权重"},
  103. // {value: "description", label: "关系描述"},
  104. {value: "do", label: "操作", width: 360, fixed: 'right'},
  105. ],
  106. result: {
  107. data: []
  108. }
  109. },
  110. form: {},
  111. loadingIndex: false,
  112. indexList: [],
  113. indexSelect: '',
  114. relationDetail: {
  115. transfer: {},
  116. show: false
  117. },
  118. relationColumn: {
  119. transfer: {},
  120. show: false
  121. },
  122. relationMain: {
  123. transfer: {},
  124. show: false
  125. }
  126. })
  127. const ref_search = ref()
  128. const fetchSuggestions = (queryString: string, cb: any) => {
  129. const arr = state.indexList.filter(v => state.query.result.data.every(s => s.indexId != v.id))
  130. const results = queryString
  131. ? arr.filter(v => [v.indexName, v.indexTableName].some(s => s.includes(queryString)))
  132. : arr
  133. results.map(v => {
  134. const str1 = queryString ? v.indexName.replace(new RegExp(queryString, 'g'), `<em>${queryString}</em>`) : v.indexName
  135. const str2 = queryString ? v.indexTableName.replace(new RegExp(queryString, 'g'), `<em>${queryString}</em>`) : v.indexTableName
  136. v.html = `${str1}<span style="margin-right: 20px;"></span>${str2}`
  137. return v
  138. })
  139. cb(results)
  140. }
  141. const handleSelect = (item) => {
  142. state.relationDetail.transfer = {
  143. type: state.type,
  144. mode: 'add',
  145. indexId: item.id,
  146. indexTableName: item.indexTableName,
  147. indexName: item.indexName,
  148. themeId: props.transfer.id,
  149. conditionOptions: state.form.themeParam,
  150. relationOptions: state.query.result.data.map(v => ({dictLabel: v.indexName, dictValue: v.id})),
  151. hasMain: state.query.result.data.some(v => v.isMain == 1) || state.type !== 'inner_index'
  152. }
  153. state.relationDetail.show = true
  154. }
  155. const initIndex = () => {
  156. state.loadingIndex = true
  157. setTimeout(() => {
  158. sysLabelGetAllIndexsByKey(proxy.$util.formatGetParam({type: state.type})).then(res => {
  159. state.indexList = res.data
  160. state.loadingIndex = false
  161. })
  162. }, 100)
  163. }
  164. const initRelation = () => {
  165. state.query.loading = true
  166. sysThemeIndexFindAll(props.transfer.id).then(res => {
  167. state.query.result.data = res.data
  168. state.query.loading = false
  169. })
  170. }
  171. const onMain = (row) => {
  172. state.relationMain.transfer = {
  173. indexId: row.indexId,
  174. id: row.id,
  175. indexStyle: row.indexStyle
  176. }
  177. state.relationMain.show = true
  178. }
  179. const onText = (row) => {
  180. state.relationColumn.transfer = {
  181. indexId: row.indexId,
  182. themeId: props.transfer.id
  183. }
  184. state.relationColumn.show = true
  185. }
  186. const onEdit = (row) => {
  187. let hasMain = false
  188. state.query.result.data.forEach(v => {
  189. if (v.isMain == 1) {
  190. if (v.id != row.id) {
  191. hasMain = true
  192. }
  193. }
  194. })
  195. state.relationDetail.transfer = {
  196. type: row.type,
  197. mode: 'edit',
  198. id: row.id,
  199. indexTableName: row.indexTableName,
  200. indexName: row.indexName,
  201. conditionOptions: state.form.themeParam,
  202. relationOptions: state.query.result.data.filter(v => v.id !== row.id).map(v => ({dictLabel: v.indexName, dictValue: v.id})),
  203. hasMain: hasMain || row.type !== 'inner_index'
  204. }
  205. state.relationDetail.show = true
  206. }
  207. const onDel = (row) => {
  208. ElMessageBox.confirm(`请确认是否删除${row.indexName}?`, "提示", {
  209. confirmButtonText: "确定",
  210. cancelButtonText: "取消",
  211. type: "warning",
  212. } as any).then(() => {
  213. state.loading = true
  214. sysThemeIndexDelete(row.id).then(res => {
  215. ElMessage.success('删除成功!')
  216. emit('refresh')
  217. state.loading = false
  218. initRelation()
  219. })
  220. }).catch(() => {})
  221. }
  222. const initDetail = () => {
  223. state.loading = true
  224. sysThemeFind(props.transfer.id).then(res => {
  225. state.form = res.data
  226. if (state.form.themeType == '2') {
  227. state.form.themeParam = state.form.themeParam.split(',')
  228. }
  229. state.loading = false
  230. })
  231. }
  232. watch(() => props.show, (n) => {
  233. initDictionary()
  234. if (n) {
  235. initIndex()
  236. initRelation()
  237. initDetail()
  238. }
  239. })
  240. const initDictionary = () => {
  241. DictionaryStore.initDict('is_main_index')
  242. DictionaryStore.initDict('service_type')
  243. }
  244. </script>
  245. <style lang="scss" scoped>
  246. .__cus-manage_content {
  247. margin-bottom: 0;
  248. height: calc(100% - 24px);
  249. }
  250. </style>