index.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695
  1. <template>
  2. <div class="archive-main">
  3. <HeadMenu/>
  4. <div class="archive-content">
  5. <div class="archive-basic" v-if="themeConfigCpt.main.indexStyle" v-loading="state.mainObj.loading">
  6. <div class="archive-basic-title">
  7. <SvgIcon class="flag" name="flag_1" color="var(--cus-main-color)"/>
  8. <span class="main">{{state.mainObj.data[themeConfigCpt.main.indexStyle.mainParam]}}</span>
  9. <template v-for="k in themeConfigCpt.main.indexStyle.tagParams">
  10. <span class="tag" v-if="state.mainObj.data[k]">{{ state.mainObj.data[k] }}</span>
  11. </template>
  12. </div>
  13. <div class="archive-basic-sub" v-if="themeConfigCpt.main.indexStyle.subParams?.length > 0">
  14. <template v-for="(k, i) in themeConfigCpt.main.indexStyle.subParams">
  15. <span>{{ state.mainObj.data[k] }}</span>
  16. <span class="line" v-if="i < themeConfigCpt.main.indexStyle.subParams.length - 1"/>
  17. </template>
  18. </div>
  19. <div class="archive-basic-avatar" v-if="themeConfigCpt.main.indexStyle.imgParam">
  20. <el-image
  21. :src="state.mainObj.data[themeConfigCpt.main.indexStyle.imgParam]"
  22. :zoom-rate="1.2"
  23. :max-scale="7"
  24. :min-scale="0.2"
  25. :preview-src-list="[state.mainObj.data[themeConfigCpt.main.indexStyle.imgParam]]"
  26. :preview-teleported="true"
  27. >
  28. <template #error>
  29. <div class="image-slot">
  30. <img src="../../../assets/images/web/archive-main-img.png" alt="暂无图片"/>
  31. </div>
  32. </template>
  33. </el-image>
  34. </div>
  35. <div class="archive-basic-cols" v-if="themeConfigCpt.main.indexStyle.otherParams?.length > 0">
  36. <template v-for="item in themeConfigCpt.main.indexStyle.otherParams">
  37. <div class="cols-item">
  38. <div class="label">{{ item.label }}</div>
  39. <div class="value">{{ state.mainObj.data[item.value] }}</div>
  40. </div>
  41. </template>
  42. </div>
  43. </div>
  44. <div class="archive-list">
  45. <el-tabs v-model="state.activeTab" class="archive-list-tabs">
  46. <el-tab-pane label="首页" name="首页"/>
  47. <template v-for="([key, value], index) in themeConfigCpt.tabs">
  48. <el-tab-pane :label="value.indexName" :name="key"/>
  49. </template>
  50. </el-tabs>
  51. <template v-if="state.activeTab === '首页'">
  52. <div class="archive-list-block">
  53. <div class="archive-list-block-title">
  54. <SvgIcon class="flag" name="flag_1" color="var(--cus-main-color)"/>关系图谱
  55. </div>
  56. <div class="archive-list-block-page" v-loading="state.chartData.loading">
  57. <RelationChart :data="state.chartData.data" @chartPage="onChartPage"/>
  58. </div>
  59. </div>
  60. <div class="archive-list-block" v-if="state.chartTable.chartId">
  61. <div class="archive-list-block-title">
  62. <SvgIcon class="flag" name="flag_1" color="var(--cus-main-color)"/>
  63. {{ themeConfigCpt.chart.nodes.filter(v => v.id == state.chartTable.chartId)?.[0]?.name }}
  64. </div>
  65. <div class="archive-list-block-page" v-loading="state.chartTable.loading">
  66. <CusTable
  67. :page-num="state.chartTable.page.pageNum"
  68. :page-size="state.chartTable.page.pageSize"
  69. :total="state.chartTable.result.total"
  70. :data="state.chartTable.result.data"
  71. :table-head="state.chartTable.tableHead"
  72. @handlePage="onPageChart"
  73. >
  74. </CusTable>
  75. </div>
  76. </div>
  77. </template>
  78. <template v-else>
  79. <div class="archive-list-block">
  80. <div class="archive-list-block-title">
  81. <SvgIcon class="flag" name="flag_1" color="var(--cus-main-color)"/>
  82. {{ themeConfigCpt.tabs.get(state.activeTab)?.indexName }}
  83. </div>
  84. <div class="archive-list-block-page" v-loading="state.tabTable.loading">
  85. <CusTable
  86. :page-num="state.tabTable.page.pageNum"
  87. :page-size="state.tabTable.page.pageSize"
  88. :total="state.tabTable.result.total"
  89. :data="state.tabTable.result.data"
  90. :table-head="state.tabTable.tableHead"
  91. @handlePage="onPage"
  92. :page-sizes="[20, 40, 60, 80, 100]"
  93. >
  94. </CusTable>
  95. </div>
  96. </div>
  97. </template>
  98. </div>
  99. </div>
  100. </div>
  101. </template>
  102. <script setup lang="ts">
  103. import {computed, getCurrentInstance, onMounted, reactive, watch} from "vue";
  104. import RelationChart from "./relation-chart.vue";
  105. import {ElLoading, ElMessage} from "element-plus";
  106. import {useAppStore, useThemeStore} from "@/stores";
  107. import {frontGetThemeByThemeId} from "@/api/modules/web/list";
  108. import {useRoute} from "vue-router";
  109. const ThemeStore = useThemeStore()
  110. const AppStore = useAppStore()
  111. const route = useRoute()
  112. const {proxy} = getCurrentInstance()
  113. const state: any = reactive({
  114. activeTab: '',
  115. tabTable: {
  116. loading: false,
  117. page: {
  118. pageNum: 1,
  119. pageSize: 20
  120. },
  121. tableHead: [],
  122. result: {
  123. total: 0,
  124. data: []
  125. },
  126. },
  127. chartTable: {
  128. chartId: '',
  129. loading: false,
  130. page: {
  131. pageNum: 1,
  132. pageSize: 10
  133. },
  134. tableHead: [],
  135. result: {
  136. total: 0,
  137. data: []
  138. },
  139. },
  140. chartData: {
  141. loading: false,
  142. data: {
  143. nodes: [],
  144. edges: []
  145. }
  146. },
  147. ws: {
  148. instance: null,
  149. getMain: () => {},
  150. tabPage: () => {},
  151. getChart: () => {},
  152. chartPage: () => {},
  153. loading: null,
  154. loadingTimer: null
  155. },
  156. themeDetail: {},
  157. mainObj: {
  158. loading: false,
  159. data: {}
  160. }
  161. })
  162. const themeConfigCpt = computed(() => {
  163. const getSearchParam = (item) => {
  164. // es查询参数
  165. const param = {}
  166. param.indexCode = item.indexTableName
  167. if (item.indexOrder) {
  168. const obj = {}
  169. JSON.parse(item.indexOrder).forEach(s => {
  170. obj[s.sortParam] = s.sortType
  171. })
  172. param.orderBy = obj
  173. }
  174. if (item.indexCondition) {
  175. const arr = []
  176. JSON.parse(item.indexCondition).forEach(s => {
  177. arr.push({
  178. fieldCode: s.indexParam,
  179. fieldQueryType: s.condition,
  180. fieldQueryValue: route.query[s.themeParam]
  181. })
  182. })
  183. param.datas = arr
  184. }
  185. return param
  186. }
  187. const getTableHead = (item) => {
  188. const arr = []
  189. item.indexFields.forEach(v => {
  190. arr.push({value: v.fieldNameEn, label: v.fieldNameCn})
  191. })
  192. return arr
  193. }
  194. const res = {
  195. main: {
  196. indexStyle: {}
  197. },
  198. tabs: new Map(),
  199. chart: {
  200. nodes: <any>[],
  201. edges: <any>[],
  202. }
  203. }
  204. state.themeDetail.indexDtos?.forEach(v => {
  205. if (v.isMain == 1) {
  206. res.main = {
  207. indexTableName: v.indexTableName,
  208. indexStyle: v.indexStyle ? JSON.parse(v.indexStyle) : '',
  209. searchParam: getSearchParam(v),
  210. tableHead: getTableHead(v)
  211. }
  212. res.chart.nodes.unshift({
  213. id: String(v.id),
  214. iconSrc: v.url,
  215. weight: v.weight || 100,
  216. num: 0,
  217. name: v.indexNameShort,
  218. indexTableName: v.indexTableName,
  219. searchParam: getSearchParam(v),
  220. tableHead: getTableHead(v)
  221. })
  222. } else if (v.relateIndexId) {
  223. res.chart.nodes.push({
  224. id: String(v.id),
  225. iconSrc: v.url,
  226. weight: v.weight,
  227. num: 0,
  228. name: v.indexNameShort,
  229. indexTableName: v.indexTableName,
  230. searchParam: getSearchParam(v),
  231. tableHead: getTableHead(v)
  232. })
  233. res.chart.edges.push({source: String(v.relateIndexId), target: String(v.id), labelText: v.description || '关系描述'})
  234. } else {
  235. res.tabs.set(v.indexTableName, {
  236. indexName: v.indexName,
  237. indexTableName: v.indexTableName,
  238. searchParam: getSearchParam(v),
  239. tableHead: getTableHead(v)
  240. })
  241. }
  242. })
  243. console.log(res)
  244. return res
  245. })
  246. const onPage = (pageNum, pageSize) => {
  247. state.tabTable.page = {
  248. pageNum: pageNum,
  249. pageSize: pageSize
  250. }
  251. state.ws.tabPage()
  252. }
  253. const onPageChart = (pageNum, pageSize) => {
  254. state.chartTable.page = {
  255. pageNum: pageNum,
  256. pageSize: pageSize
  257. }
  258. state.ws.chartPage()
  259. }
  260. const onChartPage = (id) => {
  261. state.chartTable = {
  262. chartId: id,
  263. loading: false,
  264. page: {
  265. pageNum: 1,
  266. pageSize: 10
  267. },
  268. tableHead: themeConfigCpt.value.chart.nodes.filter(v => v.id == id)?.[0]?.tableHead,
  269. result: {
  270. total: 0,
  271. data: []
  272. },
  273. }
  274. state.ws.chartPage()
  275. }
  276. const initTheme = () => {
  277. const loading = ElLoading.service({
  278. lock: true,
  279. text: '获取主题中……',
  280. background: ThemeStore.loadingBg
  281. })
  282. frontGetThemeByThemeId(route.query.themeId).then(res => {
  283. state.themeDetail = res.data
  284. loading.close()
  285. initWS()
  286. })
  287. }
  288. watch(() => state.activeTab, (n) => {
  289. if (n === '首页') {
  290. state.chartTable.chartId = ''
  291. state.ws.getChart()
  292. } else {
  293. state.tabTable = {
  294. loading: false,
  295. page: {
  296. pageNum: 1,
  297. pageSize: 20
  298. },
  299. tableHead: themeConfigCpt.value.tabs.get(n).tableHead,
  300. result: {
  301. total: 0,
  302. data: []
  303. },
  304. }
  305. state.ws.tabPage()
  306. }
  307. })
  308. const initWS = () => {
  309. const ws: any = new WebSocket(`ws://${location.host}/ws-api/smart-ws?userId=${AppStore.userInfo.id}`)
  310. let sessionId = ''
  311. let lastTabParams = ''
  312. let lastMainParams = ''
  313. let lastChartParams = ''
  314. let lastChartPageParams = ''
  315. const loading = ElLoading.service({
  316. lock: true,
  317. text: '搜索服务连接中……',
  318. background: ThemeStore.loadingBg
  319. })
  320. ws.onopen = (e) => {
  321. loading.close()
  322. state.ws.loading?.close()
  323. state.ws.loading = null
  324. state.ws.getMain = () => {
  325. // 如果有的话,终止上一次请求
  326. if (lastMainParams) {
  327. const p = JSON.parse(lastMainParams)
  328. p.flag = 'stop'
  329. ws.send(JSON.stringify(p))
  330. lastMainParams = ''
  331. }
  332. const params = {
  333. pageNumber: 1,
  334. pageSize: 1,
  335. type: 'list-fix',
  336. sessionId: sessionId,
  337. timestamp: new Date().getTime(),
  338. flag: 'run',
  339. orderBy: {},
  340. builder:[
  341. {
  342. tagCode: '',
  343. datas:[
  344. {
  345. typeCode: '',
  346. datas: [themeConfigCpt.value.main.searchParam]
  347. }
  348. ]
  349. }
  350. ],
  351. }
  352. lastMainParams = JSON.stringify(params)
  353. state.mainObj.loading = true
  354. ws.send(lastMainParams)
  355. }
  356. state.ws.tabPage = () => {
  357. // 如果有的话,终止上一次请求
  358. if (lastTabParams) {
  359. const p = JSON.parse(lastTabParams)
  360. p.flag = 'stop'
  361. ws.send(JSON.stringify(p))
  362. lastTabParams = ''
  363. }
  364. const params = {
  365. pageNumber: state.tabTable.page.pageNum,
  366. pageSize: state.tabTable.page.pageSize,
  367. type: 'list-fix',
  368. sessionId: sessionId,
  369. timestamp: new Date().getTime(),
  370. flag: 'run',
  371. builder:[
  372. {
  373. tagCode: '',
  374. datas:[
  375. {
  376. typeCode: '',
  377. datas: [themeConfigCpt.value.tabs.get(state.activeTab).searchParam]
  378. }
  379. ]
  380. }
  381. ],
  382. }
  383. lastTabParams = JSON.stringify(params)
  384. state.tabTable.loading = true
  385. ws.send(lastTabParams)
  386. }
  387. state.ws.getChart = () => {
  388. // 如果有的话,终止上一次请求
  389. if (lastChartParams) {
  390. const p = JSON.parse(lastChartParams)
  391. p.flag = 'stop'
  392. ws.send(JSON.stringify(p))
  393. lastChartParams = ''
  394. }
  395. const params = {
  396. pageNumber: state.tabTable.page.pageNum,
  397. pageSize: state.tabTable.page.pageSize,
  398. type: 'count-fix',
  399. sessionId: sessionId,
  400. timestamp: new Date().getTime(),
  401. flag: 'run',
  402. builder:[
  403. {
  404. tagCode: '',
  405. datas:[
  406. {
  407. typeCode: '',
  408. datas: themeConfigCpt.value.chart.nodes.map(v => v.searchParam)
  409. }
  410. ]
  411. }
  412. ]
  413. }
  414. lastChartParams = JSON.stringify(params)
  415. state.chartData.loading = true
  416. ws.send(lastChartParams)
  417. }
  418. state.ws.chartPage = () => {
  419. // 如果有的话,终止上一次请求
  420. if (lastChartPageParams) {
  421. const p = JSON.parse(lastChartPageParams)
  422. p.flag = 'stop'
  423. ws.send(JSON.stringify(p))
  424. lastChartPageParams = ''
  425. }
  426. const params = {
  427. pageNumber: state.chartTable.page.pageNum,
  428. pageSize: state.chartTable.page.pageSize,
  429. type: 'list-fix',
  430. sessionId: sessionId,
  431. timestamp: new Date().getTime(),
  432. flag: 'run',
  433. builder:[
  434. {
  435. tagCode: '',
  436. datas:[
  437. {
  438. typeCode: '',
  439. datas: [themeConfigCpt.value.chart.nodes.filter(v => v.id == state.chartTable.chartId)[0].searchParam]
  440. }
  441. ]
  442. }
  443. ],
  444. }
  445. lastChartPageParams = JSON.stringify(params)
  446. state.chartTable.loading = true
  447. ws.send(lastChartPageParams)
  448. }
  449. }
  450. ws.onmessage = (e) => {
  451. try {
  452. const json = JSON.parse(e.data)
  453. if (json.type === 'session') {
  454. sessionId = json.sessionId
  455. // 左侧
  456. state.ws.getMain()
  457. state.activeTab = '首页'
  458. } else {
  459. switch (json.type) {
  460. case 'list-fix': {
  461. if (lastMainParams) {
  462. const pMain = JSON.parse(lastMainParams)
  463. // 返回为最新批次的
  464. if (json.timestamp == pMain.timestamp) {
  465. state.mainObj.loading = false
  466. state.mainObj.data = json.datas?.[0] || {}
  467. lastMainParams = ''
  468. }
  469. }
  470. if (lastTabParams) {
  471. const pTab = JSON.parse(lastTabParams)
  472. // 返回为最新批次的
  473. if (json.timestamp == pTab.timestamp) {
  474. state.tabTable.result.data = json.datas
  475. state.tabTable.result.total = json.records
  476. state.tabTable.loading = false
  477. lastTabParams = ''
  478. }
  479. }
  480. if (lastChartPageParams) {
  481. const pChartPage = JSON.parse(lastChartPageParams)
  482. // 返回为最新批次的
  483. if (json.timestamp == pChartPage.timestamp) {
  484. state.chartTable.result.data = json.datas
  485. state.chartTable.result.total = json.records
  486. state.chartTable.loading = false
  487. lastChartPageParams = ''
  488. }
  489. }
  490. } break
  491. case 'count-fix': {
  492. if (lastChartParams) {
  493. const pChart = JSON.parse(lastChartParams)
  494. // 返回为最新批次的
  495. if (json.timestamp == pChart.timestamp) {
  496. const obj = {
  497. edges: [...themeConfigCpt.value.chart.edges],
  498. nodes: themeConfigCpt.value.chart.nodes.map(v => {
  499. v.num = json.datas.filter(s => s.indexCode === v.indexTableName)?.[0].data || 0
  500. return v
  501. })
  502. }
  503. state.chartData.data = obj
  504. state.chartData.loading = false
  505. lastChartParams = ''
  506. }
  507. }
  508. } break
  509. }
  510. }
  511. } catch (e) {
  512. console.log(e)
  513. }
  514. }
  515. ws.onclose = function (){
  516. state.ws.loading = ElLoading.service({
  517. lock: true,
  518. text: '智搜服务重新连接中……',
  519. background: loadingBg,
  520. })
  521. setTimeout(initWS, 5000);
  522. }
  523. state.ws.instance = ws
  524. }
  525. onMounted(() => {
  526. initTheme()
  527. })
  528. </script>
  529. <style lang="scss" scoped>
  530. .archive-main {
  531. width: 100%;
  532. height: 100%;
  533. background-color: #F6F7FB;
  534. display: flex;
  535. flex-direction: column;
  536. .archive-content {
  537. flex: 1;
  538. padding: 24px;
  539. display: flex;
  540. gap: 16px;
  541. overflow: hidden;
  542. .archive-basic {
  543. width: 362px;
  544. height: 100%;
  545. background: linear-gradient( 180deg, rgba(var(--cus-main-color-rgb), 0.2) 0%, rgba(var(--cus-main-color-rgb),0) 100%), #FFFFFF;
  546. box-shadow: 0px 2px 4px 0px rgba(46,129,255,0.2);
  547. border-radius: 4px;
  548. padding: 16px;
  549. display: flex;
  550. flex-direction: column;
  551. gap: 10px;
  552. .archive-basic-title {
  553. display: flex;
  554. align-items: center;
  555. gap: 8px;
  556. .main {
  557. font-weight: bold;
  558. font-size: 21px;
  559. color: var(--cus-main-color);
  560. }
  561. .tag {
  562. display: flex;
  563. align-items: center;
  564. justify-content: center;
  565. padding: 0 8px;
  566. background: #f8e5e5;
  567. border-radius: 2px;
  568. border: 1px solid #FF5454;
  569. font-size: 12px;
  570. color: #FF5454;
  571. }
  572. }
  573. .archive-basic-sub {
  574. display: flex;
  575. align-items: center;
  576. font-size: 14px;
  577. color: var(--cus-text-color-2);
  578. line-height: 16px;
  579. gap: 10px;
  580. .line {
  581. width: 1px;
  582. height: 10px;
  583. background-color: var(--cus-text-color-4);
  584. }
  585. }
  586. .archive-basic-avatar {
  587. width: 100%;
  588. .el-image {
  589. width: 100%;
  590. .image-slot {
  591. width: 100%;
  592. >img {
  593. width: 100%;
  594. }
  595. }
  596. }
  597. }
  598. .archive-basic-cols {
  599. display: flex;
  600. flex-wrap: wrap;
  601. row-gap: 12px;
  602. .cols-item {
  603. font-size: 14px;
  604. min-width: 50%;
  605. .label {
  606. font-weight: bold;
  607. color: var(--cus-text-color-1);
  608. }
  609. .value {
  610. margin-top: 2px;
  611. color: var(--cus-text-color-2);
  612. text-align: justify;
  613. }
  614. }
  615. }
  616. }
  617. .archive-list {
  618. flex: 1;
  619. overflow: hidden;
  620. display: flex;
  621. flex-direction: column;
  622. :deep(.archive-list-tabs) {
  623. width: 100%;
  624. .el-tabs__header {
  625. margin: 0;
  626. height: 45px;
  627. .el-tabs__nav-wrap {
  628. height: 100%;
  629. &:after {
  630. display: none;
  631. }
  632. .el-tabs__nav-prev, .el-tabs__nav-next {
  633. height: 100%;
  634. display: flex;
  635. justify-content: center;
  636. align-items: center;
  637. }
  638. .el-tabs__nav-scroll {
  639. height: 100%;
  640. .el-tabs__nav {
  641. height: 100%;
  642. .el-tabs__item {
  643. height: 100%;
  644. font-size: 16px;
  645. color: var(--cus-text-color-2);
  646. padding: 0 16px;
  647. &.is-active {
  648. background-color: #FFFFFF;
  649. box-shadow: 0px 4px 10px 0px rgba(62,123,250,0.1);
  650. border-radius: 8px 8px 0px 0px;
  651. }
  652. }
  653. }
  654. }
  655. }
  656. }
  657. }
  658. .archive-list-block {
  659. flex: 1;
  660. background-color: #FFFFFF;
  661. padding: 16px;
  662. display: flex;
  663. flex-direction: column;
  664. gap: 15px;
  665. overflow: hidden;
  666. &:first-child {
  667. border-top-right-radius: 8px;
  668. }
  669. &:last-child {
  670. border-bottom-left-radius: 8px;
  671. border-bottom-right-radius: 8px;
  672. }
  673. .archive-list-block-title {
  674. display: flex;
  675. align-items: center;
  676. font-weight: bold;
  677. font-size: 16px;
  678. color: var(--cus-text-color-1);
  679. .svg-icon {
  680. margin-right: 8px;
  681. }
  682. }
  683. .archive-list-block-page {
  684. flex: 1;
  685. overflow: hidden;
  686. }
  687. }
  688. }
  689. }
  690. }
  691. </style>