page copy.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657
  1. <template>
  2. <div class="page">
  3. <div class="one" data-aos="fade-up" data-aos-duration="1000">
  4. <div class="one_left">
  5. <el-carousel height="680px">
  6. <el-carousel-item v-for="(item, index) in carouselList" :key="index">
  7. <el-image class="image" :src="getUrl(item)" fit="fill" @click="toRedirect(item)" />
  8. </el-carousel-item>
  9. </el-carousel>
  10. </div>
  11. <div class="one_right">
  12. <div class="right_1">
  13. <div class="right_left">
  14. <span class="tab" :class="[item.value == active ? 'active' : '']" v-for="(item, index) in newList" :key="index" @click="toActive(item)">{{ item.label }}</span>
  15. </div>
  16. <div class="right_right">
  17. <div class="more_title" @click="toMore(0)">查看更多</div>
  18. <el-icon size="24"><ArrowRight /></el-icon>
  19. </div>
  20. </div>
  21. <div class="right_2">
  22. <vue3-seamless-scroll :list="list" :hover="true" :step="0.5" :limit-scroll-num="3" :wheel="true" :isWatch="true">
  23. <div class="list" v-for="(item, index) in list" :key="index" @click="toView(item, '0')">
  24. <div class="left">
  25. <el-image class="image" v-if="item.logo && item.logo.length > 0" :src="getFile(item.logo)" fit="fill">
  26. <template v-slot:error>
  27. <el-image v-if="item.type == '0'" class="image" :src="new_1" fit="fill" />
  28. <el-image v-else-if="item.type == '1'" class="image" :src="new_2" fit="fill" />
  29. <el-image v-else class="image" :src="new_2" fit="fill" />
  30. </template>
  31. </el-image>
  32. <div v-else>
  33. <el-image v-if="item.type == '0'" class="image" :src="new_1" fit="fill" />
  34. <el-image v-else-if="item.type == '1'" class="image" :src="new_2" fit="fill" />
  35. <el-image class="image" v-else :src="new_3" fit="fill" />
  36. </div>
  37. </div>
  38. <div class="right">
  39. <div class="new_title textOne">
  40. {{ item.title || '暂无标题' }}
  41. </div>
  42. <div class="new_content textMore">{{ removeHtmlStyle(item.content) || '暂无内容' }}</div>
  43. </div>
  44. </div>
  45. </vue3-seamless-scroll>
  46. </div>
  47. </div>
  48. </div>
  49. <div class="thr" data-aos="fade-up" data-aos-duration="1000">
  50. <div class="title">
  51. <!-- <el-image class="image" :src="left" fit="fill" /> -->
  52. <div class="title_center">成果展示</div>
  53. <div class="title_brief">产学研用协同创新数字化平台</div>
  54. <!-- <el-image class="image" :src="right" fit="fill" /> -->
  55. </div>
  56. <div class="thr_1">
  57. <div class="w_1700">
  58. <div class="list" v-for="(item, index) in achieveList" :key="index" @click="toView(item, '2')">
  59. <div class="list_left">
  60. <el-image class="image" :src="icon" fit="fill" />
  61. </div>
  62. <div class="list_right">
  63. <div class="other_1">
  64. <div class="name">{{ item.name || '暂无成果名称' }}</div>
  65. </div>
  66. <div class="other_2" v-if="user && user.id"><span>技术领域:</span>{{ item.field || '暂无' }}</div>
  67. <div class="other_2" v-if="user && user.id"><span>负责人:</span>{{ item.person || '暂无' }}</div>
  68. <div class="other_2" v-if="user && user.id"><span>来源:</span>{{ item.source || '暂无' }}</div>
  69. </div>
  70. </div>
  71. </div>
  72. </div>
  73. <div class="more">
  74. <div class="more_title" @click="toMore(2)">查看更多</div>
  75. <el-icon size="24"><ArrowRight /></el-icon>
  76. </div>
  77. </div>
  78. <div class="four" data-aos="fade-up" data-aos-duration="1000">
  79. <div class="title">
  80. <!-- <el-image class="image" :src="left" fit="fill" /> -->
  81. <div class="title_center">孵化体系</div>
  82. <div class="title_brief">产学研用协同创新数字化平台</div>
  83. <!-- <el-image class="image" :src="right" fit="fill" /> -->
  84. </div>
  85. <div class="four_1">
  86. <div class="list" v-for="(item, index) in incubatorList" :key="index">
  87. <el-image class="image" :src="getFile(item.logo)" fit="fill">
  88. <template v-slot:error>
  89. <el-image class="image" :src="baseLogo" fit="fill" />
  90. </template>
  91. </el-image>
  92. <div class="content">
  93. <div class="name">{{ item.name || '暂无孵化基地名称' }}</div>
  94. <div class="other_1"><span>负责人姓名:</span>{{ item.person || '暂无' }}</div>
  95. <div class="other_1"><span>负责人联系电话:</span>{{ item.person_phone || '暂无' }}</div>
  96. <div class="other_1"><span>地址:</span>{{ getArea(item.area) || '暂无' }}</div>
  97. <div class="button">
  98. <button @click="toView(item, '1')">查看详情</button>
  99. </div>
  100. </div>
  101. </div>
  102. </div>
  103. <div class="more">
  104. <div class="more_title" @click="toMore(1)">查看更多</div>
  105. <el-icon size="24"><ArrowRight /></el-icon>
  106. </div>
  107. <div class="join">
  108. <div class="join_bg"></div>
  109. <div class="join_scroll">
  110. <div class="join_shadow">
  111. <div class="join_shadow_left"></div>
  112. <div class="join_shadow_right"></div>
  113. </div>
  114. <div class="join_top">
  115. <div class="join_item" v-for="(item, index) in friendList" :key="index" @click="toFriend(item)">
  116. <el-image class="image" :src="getFile(item.file)" fit="fill">
  117. <template v-slot:error>
  118. <el-image class="image" :src="friend" fit="fill" />
  119. </template>
  120. </el-image>
  121. <div class="name">{{ item.name }}</div>
  122. </div>
  123. </div>
  124. </div>
  125. </div>
  126. </div>
  127. </div>
  128. </template>
  129. <script setup>
  130. import { get } from 'lodash-es'
  131. import { useNumberAnimation } from '@/utils/animation'
  132. // 图片引入
  133. import icon from '/images/icon.png'
  134. import new_1 from '/images/new_1.png'
  135. import new_2 from '/images/new_2.png'
  136. import new_3 from '/images/new_3.png'
  137. import friend from '/images/friend.jpeg'
  138. import baseLogo from '/images/base.jpg'
  139. // 用户信息
  140. import { UserStore } from '@/store/user'
  141. const userStore = UserStore()
  142. const user = computed(() => userStore.user)
  143. // 路由
  144. const router = useRouter()
  145. const props = defineProps({
  146. carouselList: { type: Array, default: () => [] },
  147. friendList: { type: Array, default: () => [] },
  148. incubatorList: { type: Array, default: () => [] },
  149. achieveList: { type: Array, default: () => [] },
  150. recordList: { type: Array, default: () => [] },
  151. list: { type: Array, default: () => [] }
  152. })
  153. const { carouselList, incubatorList, recordList } = toRefs(props)
  154. const emits = defineEmits(['toActive'])
  155. const active = ref('0')
  156. const recordInfo = ref({})
  157. const newList = ref([
  158. { value: '0', label: '政策信息' },
  159. { value: '1', label: '新闻通知' },
  160. { value: '2', label: '行业动态' }
  161. ])
  162. const toActive = async (item) => {
  163. active.value = item.value
  164. emits('toActive', item.value)
  165. }
  166. const toRedirect = (data) => {
  167. const to = get(data, 'to')
  168. if (to) window.open(to)
  169. }
  170. const getUrl = (item) => {
  171. if (item) return `${import.meta.env.VITE_APP_HOST}${item.uri}`
  172. }
  173. const getFile = (item) => {
  174. if (item && item.length > 0) return `${import.meta.env.VITE_APP_HOST}${item[0].uri}`
  175. }
  176. // 查看详情
  177. const toView = (item, type) => {
  178. if (type == '0') {
  179. if (item.jump_type == '1') window.open(item.route, '_blank')
  180. else router.push({ path: '/news/detail', query: { id: item.id || item._id } })
  181. } else {
  182. if (user.value.id) {
  183. if (type == '1') router.push({ path: '/base/detail', query: { id: item.id || item._id } })
  184. else router.push({ path: '/achievement/detail', query: { id: item.id || item._id } })
  185. } else ElMessage({ message: '未登录!', type: 'error' })
  186. }
  187. }
  188. // 查看更多
  189. const toMore = (type) => {
  190. if (type == '0') router.push({ path: '/news' })
  191. if (type == '1') router.push({ path: '/base' })
  192. if (type == '2') router.push({ path: '/nine' })
  193. }
  194. // 地区
  195. const getArea = (data) => {
  196. if (data) return data.join('-')
  197. else return '暂无地区'
  198. }
  199. // 富文本处理
  200. const removeHtmlStyle = (html) => {
  201. let relStyle = /style\s*?=\s*?([‘"])[\s\S]*?\1/g //去除样式
  202. let relTag = /<.+?>/g //去除标签
  203. let relClass = /class\s*?=\s*?([‘"])[\s\S]*?\1/g // 清除类名
  204. let newHtml = ''
  205. if (html) {
  206. newHtml = html.replace(relStyle, '')
  207. newHtml = newHtml.replace(relTag, '')
  208. newHtml = newHtml.replace(relClass, '')
  209. }
  210. return newHtml
  211. }
  212. // 查看合作伙伴下级
  213. const toFriend = (item) => {
  214. router.push({ path: '/friend/detail', query: { code: item.code } })
  215. }
  216. // 请求
  217. onMounted(() => {
  218. setTimeout(async () => {
  219. await Animation()
  220. }, 2000)
  221. })
  222. // 选择数据分析
  223. const Animation = () => {
  224. if (recordList.value && recordList.value.length > 0) {
  225. for (const val of recordList.value) {
  226. useNumberAnimation({
  227. from: 0,
  228. to: val.num,
  229. duration: 3000,
  230. onProgress: (v) => {
  231. if (v) val.num = v.toFixed(0)
  232. }
  233. })
  234. }
  235. }
  236. }
  237. watch(
  238. recordList,
  239. (item) => {
  240. recordInfo.value = item[0]
  241. },
  242. {
  243. deep: true
  244. }
  245. )
  246. </script>
  247. <style scoped lang="scss">
  248. .page {
  249. .title {
  250. text-align: center;
  251. margin: 30px 0 50px;
  252. .image {
  253. margin: 0 10px;
  254. vertical-align: middle;
  255. border-style: none;
  256. }
  257. .title_center {
  258. font-size: $global-font-size-40;
  259. font-weight: 600;
  260. }
  261. .title_brief {
  262. margin: 15px 0 0 0;
  263. font-size: $global-font-size-24;
  264. color: #999;
  265. }
  266. }
  267. .one {
  268. max-width: 1700px;
  269. margin: 30px auto 0;
  270. height: 680px;
  271. display: flex;
  272. justify-content: space-between;
  273. .one_left {
  274. width: 50%;
  275. .image {
  276. height: 100%;
  277. width: 100%;
  278. }
  279. }
  280. .one_right {
  281. width: 48%;
  282. padding: 0 0 0 10px;
  283. .right_1 {
  284. display: flex;
  285. align-items: center;
  286. justify-content: space-between;
  287. margin: 0 0 10px 0;
  288. .right_left {
  289. font-size: $global-font-size-28;
  290. .tab {
  291. margin: 0 30px 0 0;
  292. cursor: pointer;
  293. }
  294. .active {
  295. font-size: 40px;
  296. }
  297. }
  298. .right_right {
  299. display: flex;
  300. align-items: center;
  301. font-size: 24px;
  302. cursor: default;
  303. }
  304. }
  305. .right_2 {
  306. height: 630px;
  307. overflow: hidden;
  308. .list {
  309. display: flex;
  310. border-bottom: 1px solid #979797;
  311. padding: 30px 0;
  312. .left {
  313. width: 200px;
  314. margin: 0 10px 0 0;
  315. .image {
  316. width: 200px;
  317. height: 150px;
  318. }
  319. }
  320. .right {
  321. width: 700px;
  322. .new_title {
  323. font-size: $global-font-size-24;
  324. font-weight: 600;
  325. color: #5f5f5f;
  326. cursor: default;
  327. margin: 0 0 10px 0;
  328. }
  329. .new_title:hover {
  330. color: #006cff;
  331. }
  332. .new_content {
  333. margin: 14px 0 0 0;
  334. color: #9ea6b3;
  335. font-size: $global-font-size-20;
  336. }
  337. }
  338. }
  339. }
  340. }
  341. }
  342. .two {
  343. max-width: 1700px;
  344. margin: 0 auto 10px;
  345. .two_1 {
  346. display: flex;
  347. align-items: center;
  348. justify-content: space-between;
  349. height: 111px;
  350. background-image: linear-gradient(0deg, #eeeffb 0, #fff 100%), linear-gradient(#fff, #fff);
  351. background-blend-mode: normal, normal;
  352. box-shadow: 0 0 16px 0 rgba(72, 117, 229, 0.5);
  353. border-radius: 6px;
  354. .twoList {
  355. text-align: center;
  356. width: 340px;
  357. border-right: 1px solid #d6d9e2;
  358. cursor: default;
  359. .two_num {
  360. color: #1054ee;
  361. span:first-child {
  362. font-size: 28px;
  363. font-family: '微软雅黑';
  364. line-height: 28px;
  365. height: 28px;
  366. overflow: hidden;
  367. display: inline-block;
  368. position: relative;
  369. font-weight: bold;
  370. }
  371. span:last-child {
  372. font-size: 20px;
  373. }
  374. }
  375. .two_title {
  376. margin: 10px 0 0 0;
  377. font-size: 21px;
  378. color: #676767;
  379. }
  380. }
  381. }
  382. .two_2 {
  383. margin: 10px 0 0 0;
  384. .two_title {
  385. height: 90px;
  386. line-height: 90px;
  387. text-align: center;
  388. font-size: 30px;
  389. color: #323232;
  390. }
  391. .two_content {
  392. margin: 20px 0 0 0;
  393. }
  394. }
  395. }
  396. .thr {
  397. margin: 20px 0;
  398. background: #f1f6f9;
  399. background-image: url(/images/con1-bg.png);
  400. background-size: 100% 100%;
  401. padding: 10px 0 0 0;
  402. .thr_1 {
  403. padding: 10px 0 10px;
  404. .w_1700 {
  405. max-width: 1700px;
  406. margin: 0 auto;
  407. display: flex;
  408. justify-content: space-between;
  409. flex-wrap: wrap;
  410. .list {
  411. margin-bottom: 15px;
  412. padding-top: 18px;
  413. width: 413px;
  414. padding: 20px;
  415. background-color: #ffffff;
  416. border-radius: 10px;
  417. display: flex;
  418. .list_left {
  419. margin: 0 10px 0 0;
  420. }
  421. .list_right {
  422. width: 320px;
  423. .other_1 {
  424. display: flex;
  425. align-items: center;
  426. margin: 0 0 20px 0;
  427. .name {
  428. font-family: PingFangSC-Medium;
  429. font-size: 20px;
  430. font-weight: 600;
  431. overflow: hidden;
  432. text-overflow: ellipsis;
  433. display: -webkit-box;
  434. -webkit-line-clamp: 1;
  435. -webkit-box-orient: vertical;
  436. }
  437. }
  438. .other_2 {
  439. margin: 0 0 10px 0;
  440. font-family: PingFangSC-Medium;
  441. font-size: 18px;
  442. font-weight: 500;
  443. overflow: hidden;
  444. text-overflow: ellipsis;
  445. display: -webkit-box;
  446. -webkit-line-clamp: 1;
  447. -webkit-box-orient: vertical;
  448. span {
  449. color: #7e8288;
  450. }
  451. }
  452. }
  453. }
  454. .list:hover {
  455. box-shadow: 0px 10px 30px #d5eaf3;
  456. }
  457. }
  458. }
  459. .more {
  460. padding: 0 0 30px 0;
  461. max-width: 1700px;
  462. margin: 0 auto;
  463. text-align: center;
  464. display: flex;
  465. align-items: center;
  466. justify-content: center;
  467. .more_title {
  468. font-size: 24px;
  469. cursor: default;
  470. }
  471. }
  472. }
  473. .four {
  474. .four_1 {
  475. max-width: 1700px;
  476. margin: 0 auto 10px;
  477. display: flex;
  478. .list {
  479. width: 400px;
  480. margin: 0 30px 0 0;
  481. box-shadow: 0px 2px 10px 1px rgba(0, 0, 0, 0.12);
  482. .image {
  483. width: 100%;
  484. height: 280px;
  485. }
  486. .content {
  487. margin: 20px 30px 30px;
  488. .name {
  489. font-weight: bold;
  490. font-size: 22px;
  491. margin-bottom: 14px;
  492. overflow: hidden;
  493. text-overflow: ellipsis;
  494. display: -webkit-box;
  495. -webkit-line-clamp: 1;
  496. -webkit-box-orient: vertical;
  497. }
  498. .other_1 {
  499. overflow: hidden;
  500. text-overflow: ellipsis;
  501. display: -webkit-box;
  502. -webkit-line-clamp: 3;
  503. -webkit-box-orient: vertical;
  504. font-size: 16px;
  505. line-height: 21px;
  506. font-weight: 600;
  507. margin-bottom: 10px;
  508. span {
  509. font-weight: 400;
  510. }
  511. }
  512. .button {
  513. text-align: center;
  514. button {
  515. width: 200px;
  516. height: 40px;
  517. line-height: 36px;
  518. text-align: center;
  519. color: #fff;
  520. font-size: 16px;
  521. background: linear-gradient(to bottom, #009fff, #0077ff);
  522. margin: 40px auto 0px;
  523. display: block;
  524. border-radius: 30px;
  525. border: none;
  526. }
  527. }
  528. }
  529. }
  530. .list:last-child {
  531. margin: 0;
  532. }
  533. }
  534. .more {
  535. padding: 30px 0;
  536. max-width: 1700px;
  537. margin: 0 auto;
  538. text-align: center;
  539. display: flex;
  540. align-items: center;
  541. justify-content: center;
  542. .more_title {
  543. font-size: 24px;
  544. cursor: default;
  545. }
  546. }
  547. .join {
  548. margin: auto;
  549. width: 100%;
  550. position: relative;
  551. overflow: hidden;
  552. display: flex;
  553. padding-top: 48px;
  554. justify-content: center;
  555. .join_bg {
  556. margin: 0 auto;
  557. max-width: 1700px;
  558. position: absolute;
  559. z-index: -100;
  560. top: 0;
  561. height: 300px;
  562. background: linear-gradient(270deg, #7aa4ff 39.53%, #b7fff5 81.34%);
  563. mix-blend-mode: normal;
  564. opacity: 0.1;
  565. filter: blur(67.957px);
  566. }
  567. .join_scroll {
  568. margin: 0 auto;
  569. max-width: 1920px;
  570. min-width: 1700px;
  571. position: relative;
  572. overflow: hidden;
  573. .join_shadow {
  574. margin: 0 auto;
  575. width: 100%;
  576. min-width: 1700px;
  577. position: absolute;
  578. z-index: 1;
  579. display: flex;
  580. justify-content: space-between;
  581. .join_shadow_left {
  582. background: linear-gradient(180deg, rgba(255, 255, 255, 0) 0, #fff 43.52%);
  583. position: absolute;
  584. z-index: 1;
  585. width: 552px;
  586. height: 400px;
  587. top: 0;
  588. right: -150px;
  589. transform: matrix(0, 1, 1, 0, 0, 0);
  590. }
  591. .join_shadow_right {
  592. background: linear-gradient(180deg, rgba(255, 255, 255, 0) 0, #fff 43.52%);
  593. position: absolute;
  594. z-index: 1;
  595. width: 552px;
  596. height: 400px;
  597. top: 0;
  598. transform: rotate(90deg);
  599. left: -150px;
  600. }
  601. }
  602. .join_top {
  603. display: flex;
  604. transform: translate3d(-50%, 0, 0);
  605. width: -webkit-max-content;
  606. width: max-content;
  607. animation: masked-animation 120s linear 1s infinite;
  608. margin-left: 280px;
  609. .join_item {
  610. margin-right: 16px;
  611. background: #fff;
  612. opacity: 0.9;
  613. border: 1px solid #fff;
  614. border-radius: 4px;
  615. padding: 20px;
  616. width: 430px;
  617. .image {
  618. margin-bottom: 10px;
  619. width: 200px;
  620. height: 140px;
  621. border-radius: 4px;
  622. }
  623. .name {
  624. font-weight: 600;
  625. font-size: 16px;
  626. line-height: 26px;
  627. overflow: hidden;
  628. text-overflow: ellipsis;
  629. white-space: nowrap;
  630. cursor: default;
  631. }
  632. }
  633. }
  634. @keyframes masked-animation {
  635. 0% {
  636. transform: translateZ(0);
  637. }
  638. 100% {
  639. transform: translate3d(-50%, 0, 0);
  640. }
  641. }
  642. }
  643. }
  644. }
  645. }
  646. </style>