index-worker.vue 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817
  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.indexNameShort || 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. <el-tooltip :content="state.chartData.showZero ? '点击切换隐藏空节点' : '点击切换显示空节点'" placement="top">
  56. <SvgIcon style="margin-left: 6px" class="__hover" :name="state.chartData.showZero ? 'show' : 'hide'" color="var(--cus-text-color-2)" size="14" @click="state.chartData.showZero = !state.chartData.showZero"/>
  57. </el-tooltip>
  58. <div class="layout-select">
  59. <CusFormColumn
  60. :span="24"
  61. v-model:param="state.chartData.layout"
  62. link="select"
  63. :options="DictionaryStore.relationChartLayoutList"
  64. />
  65. </div>
  66. </div>
  67. <div class="archive-list-block-page" v-loading="state.chartData.loading">
  68. <RelationChart :data="state.chartData.data" :layout="state.chartData.layout" :show-zero="state.chartData.showZero" @chartPage="onChartPage"/>
  69. </div>
  70. </div>
  71. <div class="archive-list-block" v-if="state.chartTable.chartId">
  72. <div class="archive-list-block-title">
  73. <SvgIcon class="flag" name="flag_1" color="var(--cus-main-color)"/>
  74. {{ themeConfigCpt.chart.nodes.filter(v => v.id == state.chartTable.chartId)?.[0]?.name }}
  75. </div>
  76. <div class="archive-list-block-page" v-loading="state.chartTable.loading">
  77. <CusTable
  78. :page-num="state.chartTable.page.pageNum"
  79. :page-size="state.chartTable.page.pageSize"
  80. :total="state.chartTable.result.total"
  81. :data="state.chartTable.result.data"
  82. :table-head="state.chartTable.tableHead"
  83. @handlePage="onPageChart"
  84. >
  85. </CusTable>
  86. </div>
  87. </div>
  88. </template>
  89. <template v-else>
  90. <div class="archive-list-block">
  91. <div class="archive-list-block-title">
  92. <SvgIcon class="flag" name="flag_1" color="var(--cus-main-color)"/>
  93. {{ themeConfigCpt.tabs.get(state.activeTab)?.indexNameShort || themeConfigCpt.tabs.get(state.activeTab)?.indexName }}
  94. </div>
  95. <div class="archive-list-block-page" v-loading="state.tabTable.loading">
  96. <CusTable
  97. :page-num="state.tabTable.page.pageNum"
  98. :page-size="state.tabTable.page.pageSize"
  99. :total="state.tabTable.result.total"
  100. :data="state.tabTable.result.data"
  101. :table-head="state.tabTable.tableHead"
  102. @handlePage="onPage"
  103. :page-sizes="[20, 40, 60, 80, 100]"
  104. >
  105. </CusTable>
  106. </div>
  107. </div>
  108. </template>
  109. </div>
  110. </div>
  111. </div>
  112. </template>
  113. <script setup lang="ts">
  114. import {computed, getCurrentInstance, onMounted, reactive, watch} from "vue";
  115. import RelationChart from "./relation-chart.vue";
  116. import {ElLoading, ElMessage} from "element-plus";
  117. import {useAppStore, useDictionaryStore, useThemeStore} from "@/stores";
  118. import {frontGetThemeByThemeId, searchLogsSaveSearchLog} from "@/api/modules/web/list";
  119. import {useRoute} from "vue-router";
  120. import WorkerWebsocket from "@/worker/worker-websocket?sharedworker";
  121. const ThemeStore = useThemeStore()
  122. const DictionaryStore = useDictionaryStore()
  123. const AppStore = useAppStore()
  124. const route = useRoute()
  125. const {proxy} = getCurrentInstance()
  126. const state: any = reactive({
  127. activeTab: '',
  128. tabTable: {
  129. loading: false,
  130. page: {
  131. pageNum: 1,
  132. pageSize: 20
  133. },
  134. tableHead: [],
  135. result: {
  136. total: 0,
  137. data: []
  138. },
  139. },
  140. chartTable: {
  141. chartId: '',
  142. loading: false,
  143. page: {
  144. pageNum: 1,
  145. pageSize: 10
  146. },
  147. tableHead: [],
  148. result: {
  149. total: 0,
  150. data: []
  151. },
  152. },
  153. chartData: {
  154. loading: false,
  155. data: {
  156. nodes: [],
  157. edges: []
  158. },
  159. layout: 'radial',
  160. showZero: false,
  161. },
  162. ws: {
  163. getMain: () => {},
  164. tabPage: () => {},
  165. getChart: () => {},
  166. chartPage: () => {},
  167. loading: null,
  168. loadingTimer: null
  169. },
  170. themeDetail: {},
  171. mainObj: {
  172. loading: false,
  173. data: {}
  174. }
  175. })
  176. const themeConfigCpt = computed(() => {
  177. const getSearchParam = (item) => {
  178. // es查询参数
  179. const param = {}
  180. if (item.type === 'inner_index') {
  181. param.indexCode = item.indexTableName
  182. if (item.indexOrder) {
  183. const obj = {}
  184. JSON.parse(item.indexOrder).forEach(s => {
  185. obj[s.sortParam] = s.sortType
  186. })
  187. param.orderBy = obj
  188. }
  189. if (item.indexCondition) {
  190. const arr: any = []
  191. JSON.parse(item.indexCondition).forEach(s => {
  192. arr.push({
  193. fieldCode: s.indexParam,
  194. fieldQueryType: s.condition,
  195. fieldQueryValue: route.query[s.themeParam]
  196. })
  197. })
  198. param.datas = arr
  199. }
  200. } else {
  201. param.serviceId = item.indexId
  202. if (item.indexCondition) {
  203. const obj: any = {}
  204. JSON.parse(item.indexCondition).forEach(s => {
  205. obj[s.indexParam] = route.query[s.themeParam] || ''
  206. })
  207. param.serviceMap = obj
  208. }
  209. }
  210. return param
  211. }
  212. const getTableHead = (item) => {
  213. const arr: any = []
  214. item.indexFields.forEach(v => {
  215. arr.push({value: v.fieldKey, label: v.fieldName})
  216. })
  217. return arr
  218. }
  219. const getSex = (idCard: string) => {
  220. if (idCard.length !== 18) {
  221. throw new Error("身份证号码长度不正确");
  222. }
  223. const genderDigit = parseInt(idCard.charAt(16), 10);
  224. return genderDigit % 2 === 1 ? "男" : "女";
  225. }
  226. const res = {
  227. main: <any>{
  228. indexStyle: {}
  229. },
  230. tabs: new Map(),
  231. chart: {
  232. nodes: <any>[],
  233. edges: <any>[],
  234. }
  235. }
  236. state.themeDetail.indexDtos?.forEach(v => {
  237. if (v.isMain == 1) {
  238. state.chartData.layout = v.chartLayout
  239. res.main = {
  240. __type: v.type,
  241. indexTableName: v.indexTableName,
  242. indexId: v.indexId,
  243. indexStyle: v.indexStyle ? JSON.parse(v.indexStyle) : '',
  244. searchParam: getSearchParam(v),
  245. tableHead: getTableHead(v),
  246. }
  247. const node = {
  248. id: String(v.id),
  249. iconSrc: v.url,
  250. iconSize: v.iconSize,
  251. weight: v.weight || 100,
  252. num: 0,
  253. name: v.indexNameShort,
  254. __type: v.type,
  255. indexTableName: v.indexTableName,
  256. indexId: v.indexId,
  257. searchParam: getSearchParam(v),
  258. tableHead: getTableHead(v),
  259. }
  260. try {
  261. JSON.parse(v.indexCondition).forEach(c => {
  262. if (c.paramType == 'sex') {
  263. const sex = getSex(route.query[c.themeParam])
  264. if (sex == '男') {
  265. node.iconSrc = c.sexImgs[0].url
  266. } else if (sex == '女') {
  267. node.iconSrc = c.sexImgs[1].url
  268. }
  269. }
  270. })
  271. } catch (e) {
  272. console.error(e)
  273. }
  274. res.chart.nodes.unshift(node)
  275. } else if (v.relateIndexId) {
  276. const node = {
  277. id: String(v.id),
  278. iconSrc: v.url,
  279. iconSize: v.iconSize,
  280. weight: v.weight,
  281. num: 0,
  282. name: v.indexNameShort,
  283. __type: v.type,
  284. indexTableName: v.indexTableName,
  285. indexId: v.indexId,
  286. searchParam: getSearchParam(v),
  287. tableHead: getTableHead(v),
  288. }
  289. try {
  290. JSON.parse(v.indexCondition).forEach(c => {
  291. if (c.paramType == 'sex') {
  292. const sex = getSex(route.query[c.themeParam])
  293. if (sex == '男') {
  294. node.iconSrc = c.sexImgs[0].url
  295. } else if (sex == '女') {
  296. node.iconSrc = c.sexImgs[1].url
  297. }
  298. }
  299. })
  300. } catch (e) {
  301. console.error(e)
  302. }
  303. res.chart.nodes.push(node)
  304. res.chart.edges.push({source: String(v.relateIndexId), target: String(v.id), labelText: v.description || '关系描述'})
  305. } else {
  306. res.tabs.set(v.id, {
  307. __type: v.type,
  308. indexName: v.indexName,
  309. indexNameShort: v.indexNameShort,
  310. indexTableName: v.indexTableName,
  311. indexId: v.indexId,
  312. searchParam: getSearchParam(v),
  313. tableHead: getTableHead(v),
  314. })
  315. }
  316. })
  317. console.log(res)
  318. return res
  319. })
  320. const onPage = (pageNum, pageSize) => {
  321. state.tabTable.page = {
  322. pageNum: pageNum,
  323. pageSize: pageSize
  324. }
  325. state.ws.tabPage()
  326. }
  327. const onPageChart = (pageNum, pageSize) => {
  328. state.chartTable.page = {
  329. pageNum: pageNum,
  330. pageSize: pageSize
  331. }
  332. state.ws.chartPage()
  333. }
  334. const onChartPage = (id) => {
  335. state.chartTable = {
  336. chartId: id,
  337. loading: false,
  338. page: {
  339. pageNum: 1,
  340. pageSize: 10
  341. },
  342. tableHead: themeConfigCpt.value.chart.nodes.filter(v => v.id == id)?.[0]?.tableHead,
  343. result: {
  344. total: 0,
  345. data: []
  346. },
  347. }
  348. state.ws.chartPage()
  349. }
  350. const initTheme = () => {
  351. const loading = ElLoading.service({
  352. lock: true,
  353. text: '获取主题中……',
  354. background: ThemeStore.loadingBg
  355. })
  356. frontGetThemeByThemeId(route.query.themeId).then(res => {
  357. state.themeDetail = res.data
  358. loading.close()
  359. initWorker()
  360. })
  361. }
  362. watch(() => state.activeTab, (n) => {
  363. if (n === '首页') {
  364. state.chartTable.chartId = ''
  365. state.ws.getChart()
  366. } else {
  367. state.tabTable = {
  368. loading: false,
  369. page: {
  370. pageNum: 1,
  371. pageSize: 20
  372. },
  373. tableHead: themeConfigCpt.value.tabs.get(n).tableHead,
  374. result: {
  375. total: 0,
  376. data: []
  377. },
  378. }
  379. state.ws.tabPage()
  380. }
  381. })
  382. const initWorker = () => {
  383. const loading = ElLoading.service({
  384. lock: true,
  385. text: '搜索服务连接中……',
  386. background: ThemeStore.loadingBg
  387. })
  388. let sessionId = ''
  389. let lastTabParams = ''
  390. let lastMainParams = ''
  391. let lastChartParams = ''
  392. let lastChartPageParams = ''
  393. const worker = new WorkerWebsocket()
  394. worker.port.onmessage = (e) => {
  395. const data = JSON.parse(e.data)
  396. switch (data.type) {
  397. case 'load': {
  398. worker.port.postMessage(JSON.stringify({type: 'init', userId: AppStore.userInfo.id}))
  399. } break
  400. case 'onopen': {
  401. loading.close()
  402. state.ws.loading?.close()
  403. state.ws.loading = null
  404. state.ws.getMain = () => {
  405. // 如果有的话,终止上一次请求
  406. if (lastMainParams) {
  407. const p = JSON.parse(lastMainParams)
  408. p.flag = 'stop'
  409. worker.port.postMessage(JSON.stringify({type: 'send', data: p}))
  410. lastMainParams = ''
  411. }
  412. const params = {
  413. __type: themeConfigCpt.value.main.__type,
  414. pageNumber: 1,
  415. pageSize: 1,
  416. type: 'list-fix',
  417. sessionId: sessionId,
  418. timestamp: new Date().getTime(),
  419. flag: 'run',
  420. orderBy: {},
  421. builder:[
  422. {
  423. tagCode: '',
  424. datas:[
  425. {
  426. typeCode: '',
  427. datas: [themeConfigCpt.value.main.searchParam]
  428. }
  429. ]
  430. }
  431. ],
  432. }
  433. lastMainParams = JSON.stringify(params)
  434. state.mainObj.loading = true
  435. worker.port.postMessage(JSON.stringify({type: 'send', data: lastMainParams}))
  436. }
  437. state.ws.tabPage = () => {
  438. // 如果有的话,终止上一次请求
  439. if (lastTabParams) {
  440. const p = JSON.parse(lastTabParams)
  441. p.flag = 'stop'
  442. worker.port.postMessage(JSON.stringify({type: 'send', data: p}))
  443. lastTabParams = ''
  444. }
  445. const params = {
  446. __type: themeConfigCpt.value.tabs.get(state.activeTab).__type,
  447. pageNumber: state.tabTable.page.pageNum,
  448. pageSize: state.tabTable.page.pageSize,
  449. type: 'list-fix',
  450. sessionId: sessionId,
  451. timestamp: new Date().getTime(),
  452. flag: 'run',
  453. builder:[
  454. {
  455. tagCode: '',
  456. datas:[
  457. {
  458. typeCode: '',
  459. datas: [themeConfigCpt.value.tabs.get(state.activeTab).searchParam]
  460. }
  461. ]
  462. }
  463. ],
  464. }
  465. lastTabParams = JSON.stringify(params)
  466. state.tabTable.loading = true
  467. worker.port.postMessage(JSON.stringify({type: 'send', data: lastTabParams}))
  468. }
  469. state.ws.getChart = () => {
  470. // 如果有的话,终止上一次请求
  471. if (lastChartParams) {
  472. const p = JSON.parse(lastChartParams)
  473. p.flag = 'stop'
  474. worker.port.postMessage(JSON.stringify({type: 'send', data: p}))
  475. lastChartParams = ''
  476. }
  477. const params = {
  478. pageNumber: state.tabTable.page.pageNum,
  479. pageSize: state.tabTable.page.pageSize,
  480. type: 'count-fix',
  481. sessionId: sessionId,
  482. timestamp: new Date().getTime(),
  483. flag: 'run',
  484. builder:[
  485. {
  486. tagCode: '',
  487. datas:[
  488. {
  489. typeCode: '',
  490. datas: themeConfigCpt.value.chart.nodes.map(v => {
  491. const obj = {...v.searchParam}
  492. if (obj.indexCode) {
  493. obj.indexCode += '|' + v.id
  494. } else if (obj.serviceId) {
  495. obj.serviceId += '|' + v.id
  496. }
  497. return obj
  498. })
  499. }
  500. ]
  501. }
  502. ]
  503. }
  504. lastChartParams = JSON.stringify(params)
  505. state.chartData.loading = true
  506. worker.port.postMessage(JSON.stringify({type: 'send', data: lastChartParams}))
  507. }
  508. state.ws.chartPage = () => {
  509. // 如果有的话,终止上一次请求
  510. if (lastChartPageParams) {
  511. const p = JSON.parse(lastChartPageParams)
  512. p.flag = 'stop'
  513. worker.port.postMessage(JSON.stringify({type: 'send', data: p}))
  514. lastChartPageParams = ''
  515. }
  516. const params = {
  517. __type: themeConfigCpt.value.chart.nodes.filter(v => v.id == state.chartTable.chartId)[0].__type,
  518. pageNumber: state.chartTable.page.pageNum,
  519. pageSize: state.chartTable.page.pageSize,
  520. type: 'list-fix',
  521. sessionId: sessionId,
  522. timestamp: new Date().getTime(),
  523. flag: 'run',
  524. builder:[
  525. {
  526. tagCode: '',
  527. datas:[
  528. {
  529. typeCode: '',
  530. datas: [themeConfigCpt.value.chart.nodes.filter(v => v.id == state.chartTable.chartId)[0].searchParam]
  531. }
  532. ]
  533. }
  534. ],
  535. }
  536. lastChartPageParams = JSON.stringify(params)
  537. state.chartTable.loading = true
  538. worker.port.postMessage(JSON.stringify({type: 'send', data: lastChartPageParams}))
  539. }
  540. } break
  541. case 'onmessage': {
  542. try {
  543. const json = JSON.parse(data.data)
  544. console.log(json)
  545. if (json.respCode == -1) {
  546. ElMessage.error(json.respMsg)
  547. return
  548. }
  549. if (json.type === 'session') {
  550. sessionId = json.sessionId
  551. // 左侧
  552. state.ws.getMain()
  553. state.activeTab = '首页'
  554. } else {
  555. switch (json.type) {
  556. case 'list-fix': {
  557. if (lastMainParams) {
  558. const pMain = JSON.parse(lastMainParams)
  559. // 返回为最新批次的
  560. if (json.timestamp == pMain.timestamp) {
  561. if (pMain.__type === 'inner_index') {
  562. state.mainObj.data = json.datas?.[0] || {}
  563. } else {
  564. state.mainObj.data = json.datas.data?.[0] || {}
  565. }
  566. state.mainObj.loading = false
  567. lastMainParams = ''
  568. }
  569. }
  570. if (lastTabParams) {
  571. const pTab = JSON.parse(lastTabParams)
  572. // 返回为最新批次的
  573. if (json.timestamp == pTab.timestamp) {
  574. if (pTab.__type === 'inner_index') {
  575. state.tabTable.result.data = json.datas
  576. state.tabTable.result.total = json.records
  577. } else {
  578. state.tabTable.result.data = json.datas.data
  579. state.tabTable.result.total = json.datas.pathForm.total
  580. }
  581. state.tabTable.loading = false
  582. lastTabParams = ''
  583. }
  584. }
  585. if (lastChartPageParams) {
  586. const pChartPage = JSON.parse(lastChartPageParams)
  587. // 返回为最新批次的
  588. if (json.timestamp == pChartPage.timestamp) {
  589. if (pChartPage.__type === 'inner_index') {
  590. state.chartTable.result.data = json.datas
  591. state.chartTable.result.total = json.records
  592. } else {
  593. state.chartTable.result.data = json.datas.data
  594. state.chartTable.result.total = json.datas.pathForm.total
  595. }
  596. state.chartTable.loading = false
  597. lastChartPageParams = ''
  598. }
  599. }
  600. } break
  601. case 'count-fix': {
  602. if (lastChartParams) {
  603. const pChart = JSON.parse(lastChartParams)
  604. // 返回为最新批次的
  605. if (json.timestamp == pChart.timestamp) {
  606. const obj = {
  607. edges: [...themeConfigCpt.value.chart.edges],
  608. nodes: themeConfigCpt.value.chart.nodes.map(v => {
  609. if (v.__type === 'inner_index') {
  610. v.num = json.datas.filter(s => s.indexCode == (v.indexTableName + '|' + v.id))?.[0]?.data || 0
  611. } else {
  612. v.num = json.datas.filter(s => s.serviceId == (v.indexId + '|' + v.id))?.[0]?.data || 0
  613. }
  614. return v
  615. })
  616. }
  617. state.chartData.data = obj
  618. state.chartData.loading = false
  619. lastChartParams = ''
  620. }
  621. }
  622. } break
  623. }
  624. }
  625. } catch (e) {
  626. console.log(e)
  627. }
  628. } break
  629. case 'onclose': {
  630. state.ws.loading = ElLoading.service({
  631. lock: true,
  632. text: '智搜服务已断开,请刷新页面……',
  633. background: ThemeStore.loadingBg,
  634. })
  635. } break
  636. }
  637. }
  638. }
  639. onMounted(() => {
  640. initTheme()
  641. DictionaryStore.initDict('relation_chart_layout')
  642. })
  643. </script>
  644. <style lang="scss" scoped>
  645. .archive-main {
  646. width: 100%;
  647. height: 100%;
  648. background-color: #F6F7FB;
  649. display: flex;
  650. flex-direction: column;
  651. .archive-content {
  652. flex: 1;
  653. padding: 24px;
  654. display: flex;
  655. gap: 16px;
  656. overflow: hidden;
  657. .archive-basic {
  658. width: 362px;
  659. height: 100%;
  660. background: linear-gradient( 180deg, rgba(var(--cus-main-color-rgb), 0.2) 0%, rgba(var(--cus-main-color-rgb),0) 100%), #FFFFFF;
  661. box-shadow: 0px 2px 4px 0px rgba(46,129,255,0.2);
  662. border-radius: 4px;
  663. padding: 16px;
  664. display: flex;
  665. flex-direction: column;
  666. gap: 10px;
  667. .archive-basic-title {
  668. display: flex;
  669. align-items: center;
  670. gap: 8px;
  671. .main {
  672. font-weight: bold;
  673. font-size: 21px;
  674. color: var(--cus-main-color);
  675. }
  676. .tag {
  677. display: flex;
  678. align-items: center;
  679. justify-content: center;
  680. padding: 0 8px;
  681. background: #f8e5e5;
  682. border-radius: 2px;
  683. border: 1px solid #FF5454;
  684. font-size: 12px;
  685. color: #FF5454;
  686. }
  687. }
  688. .archive-basic-sub {
  689. display: flex;
  690. align-items: center;
  691. font-size: 14px;
  692. color: var(--cus-text-color-2);
  693. line-height: 16px;
  694. gap: 10px;
  695. .line {
  696. width: 1px;
  697. height: 10px;
  698. background-color: var(--cus-text-color-4);
  699. }
  700. }
  701. .archive-basic-avatar {
  702. width: 100%;
  703. .el-image {
  704. width: 100%;
  705. .image-slot {
  706. width: 100%;
  707. >img {
  708. width: 100%;
  709. }
  710. }
  711. }
  712. }
  713. .archive-basic-cols {
  714. display: flex;
  715. flex-wrap: wrap;
  716. row-gap: 12px;
  717. .cols-item {
  718. font-size: 14px;
  719. min-width: 50%;
  720. .label {
  721. font-weight: bold;
  722. color: var(--cus-text-color-1);
  723. }
  724. .value {
  725. margin-top: 2px;
  726. color: var(--cus-text-color-2);
  727. text-align: justify;
  728. }
  729. }
  730. }
  731. }
  732. .archive-list {
  733. flex: 1;
  734. overflow: hidden;
  735. display: flex;
  736. flex-direction: column;
  737. :deep(.archive-list-tabs) {
  738. width: 100%;
  739. .el-tabs__header {
  740. margin: 0;
  741. height: 45px;
  742. .el-tabs__nav-wrap {
  743. height: 100%;
  744. &:after {
  745. display: none;
  746. }
  747. .el-tabs__nav-prev, .el-tabs__nav-next {
  748. height: 100%;
  749. display: flex;
  750. justify-content: center;
  751. align-items: center;
  752. }
  753. .el-tabs__nav-scroll {
  754. height: 100%;
  755. .el-tabs__nav {
  756. height: 100%;
  757. .el-tabs__item {
  758. height: 100%;
  759. font-size: 16px;
  760. color: var(--cus-text-color-2);
  761. padding: 0 16px;
  762. &.is-active {
  763. background-color: #FFFFFF;
  764. box-shadow: 0px 4px 10px 0px rgba(62,123,250,0.1);
  765. border-radius: 8px 8px 0px 0px;
  766. }
  767. }
  768. }
  769. }
  770. }
  771. }
  772. }
  773. .archive-list-block {
  774. flex: 1;
  775. background-color: #FFFFFF;
  776. padding: 16px;
  777. display: flex;
  778. flex-direction: column;
  779. gap: 15px;
  780. overflow: hidden;
  781. &:first-child {
  782. border-top-right-radius: 8px;
  783. }
  784. &:last-child {
  785. border-bottom-left-radius: 8px;
  786. border-bottom-right-radius: 8px;
  787. }
  788. .archive-list-block-title {
  789. display: flex;
  790. align-items: center;
  791. font-weight: bold;
  792. font-size: 16px;
  793. color: var(--cus-text-color-1);
  794. .svg-icon {
  795. margin-right: 8px;
  796. }
  797. :deep(.layout-select) {
  798. margin-left: auto;
  799. .el-form-item {
  800. width: 150px;
  801. margin-bottom: 0;
  802. }
  803. }
  804. }
  805. .archive-list-block-page {
  806. flex: 1;
  807. overflow: hidden;
  808. }
  809. }
  810. }
  811. }
  812. }
  813. </style>