index.vue 19 KB

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