supplyDetail.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422
  1. <template>
  2. <custom-layout class="main">
  3. <el-col :span="24" class="one">
  4. <div class="w_1300">
  5. <div class="info_1"></div>
  6. <div class="info_2">
  7. <div class="area">
  8. <!-- <el-icon color="#0085f5"><Location /></el-icon>
  9. <span>{{ info.area || '暂无' }}</span> -->
  10. </div>
  11. <div class="info_title">
  12. <div class="name">{{ info.name || '暂无' }}</div>
  13. <div class="collect iscollect" @click="toCollect" v-if="info.is_collection">
  14. <el-icon :size="24" color="#1073ff"><StarFilled /></el-icon>
  15. <span>已收藏</span>
  16. </div>
  17. <div class="collect" v-else @click="toCollect">
  18. <el-icon :size="24"><Star /></el-icon>
  19. <span>收藏</span>
  20. </div>
  21. </div>
  22. <div class="other">
  23. <span class="other_1">{{ getDict(info.urgent, 'urgent') || '暂无' }}</span>
  24. <div class="other_2">
  25. <p>
  26. 所属产业:<span class="moeny">{{ info.industry || '暂无' }}</span>
  27. </p>
  28. <p>
  29. 应用行业:<span>{{ info.field || '暂无' }} </span>
  30. </p>
  31. <p>
  32. 所属地区 :<span> {{ getArea(info.area) || '暂无' }}</span>
  33. </p>
  34. <p>
  35. 截止时间:<span class="time">{{ info.time || '暂无' }}</span>
  36. </p>
  37. <p>
  38. 来源 :<span>{{ info.source || '暂无' }}</span>
  39. </p>
  40. </div>
  41. </div>
  42. <!-- <div class="state">待解决</div> -->
  43. </div>
  44. <div class="title"><i></i>预约对接</div>
  45. <div class="infoButton">
  46. <div class="button" @click="toChat">预约对接</div>
  47. </div>
  48. <div class="title"><i></i>供给描述</div>
  49. <div class="info_3">
  50. <p>{{ info.brief || '暂无' }}</p>
  51. </div>
  52. <div class="title"><i></i>相关需求</div>
  53. <div class="info_4">
  54. <div class="list" v-for="(item, index) in list" :key="index" @click="toView(item)">
  55. <h2 class="name textMore">
  56. <span>{{ item.name || '暂无' }}</span>
  57. </h2>
  58. <div class="other">
  59. <span class="other_1">{{ getDict(item.urgent, 'urgent') || '暂无' }}</span>
  60. <div class="other_2">
  61. <span>应用行业:</span>
  62. {{ item.field || '暂无' }}
  63. </div>
  64. <div class="other_2">
  65. <span>所属产业:</span>
  66. {{ item.industry || '暂无' }}
  67. </div>
  68. <div class="other_2">
  69. <span>来源:</span>
  70. {{ item.source || '暂无' }}
  71. </div>
  72. <!-- <div class="other_2">
  73. <el-icon color="#0085f5"><Location /></el-icon>
  74. {{ item.area || '暂无' }}
  75. <span class="state">{{ item.status || '未解决' }}</span>
  76. </div> -->
  77. </div>
  78. </div>
  79. </div>
  80. </div>
  81. </el-col>
  82. </custom-layout>
  83. </template>
  84. <script setup>
  85. import { get } from 'lodash-es'
  86. import { DictDataStore } from '@/store/api/system/dictData'
  87. import { SupplyStore } from '@/store/api/platform/supply'
  88. import { ContactApplyStore } from '@/store/api/user/contactApply'
  89. const contactApplyStore = ContactApplyStore()
  90. const store = SupplyStore()
  91. const dictDataStore = DictDataStore()
  92. import { UserStore } from '@/store/user'
  93. const userStore = UserStore()
  94. const user = computed(() => userStore.user)
  95. const $checkRes = inject('$checkRes')
  96. // 收藏
  97. import moment from 'moment'
  98. import { CollectionStore } from '@/store/api/platform/collection'
  99. const collectionStore = CollectionStore()
  100. // 加载中
  101. const loading = ref(false)
  102. // 路由
  103. const route = useRoute()
  104. const router = useRouter()
  105. const info = ref({})
  106. const list = ref([])
  107. // 字典表
  108. const fieldList = ref([])
  109. const attributeList = ref([])
  110. const matureList = ref([])
  111. const sellList = ref([])
  112. const technologyList = ref([])
  113. const urgentList = ref([])
  114. // 请求
  115. onMounted(async () => {
  116. loading.value = true
  117. await searchOther()
  118. await search()
  119. await searchAchieve()
  120. loading.value = false
  121. })
  122. const search = async () => {
  123. let id = route.query.id
  124. if (id) {
  125. let res = await store.detail(id)
  126. if (res.errcode == '0') info.value = res.data
  127. }
  128. }
  129. const searchAchieve = async () => {
  130. const data = {
  131. skip: 0,
  132. limit: 3,
  133. is_use: '0',
  134. status: '1',
  135. field: info.value.field
  136. }
  137. const res = await store.list(data)
  138. if (res.errcode == '0') list.value = res.data
  139. }
  140. const searchOther = async () => {
  141. let result
  142. // // 成熟度
  143. // result = await dictDataStore.query({ code: 'mature', is_use: '0' })
  144. // if ($checkRes(result)) matureList.value = result.data
  145. // // 出让方式
  146. // result = await dictDataStore.query({ code: 'sell', is_use: '0' })
  147. // if ($checkRes(result)) sellList.value = result.data
  148. // // 技术领域
  149. // result = await dictDataStore.query({ code: 'field', is_use: '0' })
  150. // if ($checkRes(result)) fieldList.value = result.data
  151. // // 属性
  152. // result = await dictDataStore.query({ code: 'attribute', is_use: '0' })
  153. // if ($checkRes(result)) attributeList.value = result.data
  154. // // 技术分类
  155. // result = await dictDataStore.query({ code: 'technology', is_use: '0' })
  156. // if ($checkRes(result)) technologyList.value = result.data
  157. // 需求紧急度
  158. result = await dictDataStore.query({ code: 'urgent', is_use: '0' })
  159. if ($checkRes(result)) urgentList.value = result.data
  160. }
  161. // 字典数据转换
  162. const getDict = (data, model) => {
  163. let res
  164. if (model == 'mature') res = matureList.value.find((f) => f.value == data)
  165. else if (model == 'sell') res = sellList.value.find((f) => f.value == data)
  166. else if (model == 'field') res = fieldList.value.find((f) => f.value == data)
  167. else if (model == 'attribute') res = attributeList.value.find((f) => f.value == data)
  168. else if (model == 'technology') res = technologyList.value.find((f) => f.value == data)
  169. else if (model == 'urgent') res = urgentList.value.find((f) => f.value == data)
  170. return get(res, 'label')
  171. }
  172. // 地区
  173. const getArea = (data) => {
  174. if (data) return data.join('-')
  175. else return '暂无地区'
  176. }
  177. // 查看详情
  178. const toView = (item) => {
  179. router.push({ path: '/supply/detail', query: { id: item.id || item._id } }).then(() => {
  180. // 重新刷新页面
  181. location.reload()
  182. })
  183. }
  184. const toCollect = async () => {
  185. if (user.value.id) {
  186. info.value.is_collection = !info.value.is_collection
  187. let res
  188. let message
  189. const data = {
  190. user: user.value.id,
  191. source: info.value.id,
  192. type: 'supply',
  193. time: moment().format('YYYY-MM-DD')
  194. }
  195. if (info.value.is_collection) {
  196. message = '收藏成功'
  197. res = await collectionStore.create(data)
  198. } else {
  199. message = '取消收藏成功'
  200. res = await collectionStore.cancel(data)
  201. }
  202. if (res.errcode === 0) {
  203. ElMessage({ message, type: 'success' })
  204. await search()
  205. }
  206. } else ElMessage({ message: '未登录!', type: 'error' })
  207. }
  208. const toChat = () => {
  209. if (user.value.id) {
  210. ElMessageBox.confirm(`您确认要预约对接?`, '提示', { confirmButtonText: '确定', cancelButtonText: '取消', type: 'warning' })
  211. .then(async () => {
  212. let source_id = route.query.id
  213. let source = 'supply'
  214. let apply_user = user.value.id
  215. const obj = { source_id, source, apply_user }
  216. const res = await contactApplyStore.create(obj)
  217. $checkRes(res,`预约对接成功等待消息通知`,res.errmsg)
  218. // ElMessage({ message: `预约对接成功等待消息通知`, type: 'success' })
  219. })
  220. .catch(() => {})
  221. } else ElMessage({ message: '未登录!', type: 'error' })
  222. }
  223. </script>
  224. <style scoped lang="scss">
  225. .main {
  226. .one {
  227. background: url(/images/bg-need-detail.jpg) no-repeat;
  228. background-position: center top;
  229. padding-bottom: 30px;
  230. .info_1 {
  231. padding: 30px 0;
  232. }
  233. .info_2 {
  234. position: relative;
  235. padding: 0 40px;
  236. background-color: #fff;
  237. box-shadow: 0 0 13px 0 rgba(5, 88, 219, 0.18);
  238. .info_title {
  239. display: flex;
  240. justify-content: space-between;
  241. align-items: center;
  242. .name {
  243. margin-top: 15px;
  244. font-size: $global-font-size-24;
  245. width: 90%;
  246. line-height: 36px;
  247. color: #2b2b2b;
  248. }
  249. .collect {
  250. display: flex;
  251. align-items: center;
  252. justify-content: flex-end;
  253. width: 10%;
  254. font-size: $global-font-size-20;
  255. color: #929292;
  256. cursor: default;
  257. span {
  258. margin: 0 0 0 5px;
  259. }
  260. }
  261. .iscollect {
  262. color: #1073ff;
  263. }
  264. }
  265. .other {
  266. margin-top: 25px;
  267. .other_1 {
  268. padding: 0 10px;
  269. height: 25px;
  270. background-color: #e6f2fd;
  271. border-radius: 2px;
  272. border: solid 1px #cae0f5;
  273. font-size: $global-font-size-14;
  274. line-height: 25px;
  275. color: #0085f5;
  276. }
  277. .other_2 {
  278. margin: 20px 0 0 0;
  279. display: flex;
  280. align-items: center;
  281. flex-wrap: wrap;
  282. p {
  283. margin-right: 20px;
  284. width: 45%;
  285. font-size: $global-font-size-18;
  286. color: #666;
  287. .moeny {
  288. font-family: keyifont;
  289. color: #f60;
  290. }
  291. span {
  292. color: #2b2b2b;
  293. }
  294. }
  295. }
  296. }
  297. .state {
  298. background: url(/images/icon-hzxq-state2.png);
  299. position: absolute;
  300. left: -36px;
  301. top: 0;
  302. padding: 17px 10px 0;
  303. width: 36px;
  304. height: 94px;
  305. color: #fff;
  306. font-size: $global-font-size-16;
  307. }
  308. }
  309. .infoButton {
  310. margin: 10px 0 0 0;
  311. .button {
  312. cursor: default;
  313. justify-content: center;
  314. display: flex;
  315. align-items: center;
  316. width: 136px;
  317. height: 44px;
  318. color: #fff;
  319. font-size: $global-font-size-16;
  320. background-image: linear-gradient(90deg, #0455da 0%, #378cff 100%), linear-gradient(#0455da, #0455da);
  321. background-blend-mode: normal, normal;
  322. border-radius: 4px;
  323. }
  324. }
  325. .title {
  326. margin-top: 30px;
  327. font-size: $global-font-size-20;
  328. line-height: 26px;
  329. color: #333;
  330. i {
  331. display: inline-block;
  332. vertical-align: middle;
  333. margin-right: 20px;
  334. width: 7px;
  335. height: 22px;
  336. background-color: #0085f5;
  337. border-radius: 3px;
  338. }
  339. }
  340. .info_3 {
  341. margin-top: 20px;
  342. padding: 40px;
  343. background-color: #fff;
  344. border: solid 1px #dedede;
  345. font-size: $global-font-size-16;
  346. line-height: 30px;
  347. color: #333;
  348. }
  349. .info_4 {
  350. display: flex;
  351. margin-top: 20px;
  352. .list {
  353. position: relative;
  354. margin-right: 30px;
  355. width: 380px;
  356. height: 233px;
  357. box-shadow: 0 0 13px 0 rgba(5, 88, 219, 0.18);
  358. .name {
  359. padding: 10px 20px;
  360. width: 380px;
  361. height: 71px;
  362. background-color: #dce5ff;
  363. font-size: 16px;
  364. line-height: 24px;
  365. color: #0d0d0d;
  366. display: flex;
  367. align-items: center;
  368. }
  369. .other {
  370. height: 162px;
  371. padding: 10px 20px 0 20px;
  372. .other_1 {
  373. padding: 0 10px;
  374. height: 25px;
  375. background-color: #e6f2fd;
  376. border-radius: 2px;
  377. border: solid 1px #cae0f5;
  378. font-size: $global-font-size-14;
  379. line-height: 25px;
  380. color: #0085f5;
  381. }
  382. .other_2 {
  383. display: flex;
  384. align-items: center;
  385. margin-top: 15px;
  386. font-size: $global-font-size-16;
  387. span {
  388. color: #909090;
  389. }
  390. .state {
  391. position: absolute;
  392. right: 20px;
  393. bottom: 15px;
  394. display: inline-block;
  395. vertical-align: middle;
  396. padding: 0 20px;
  397. height: 30px;
  398. line-height: 30px;
  399. background-image: linear-gradient(90deg, #ff8a00 0, #ff5a00 100%), linear-gradient(#ff7800, #ff7800);
  400. background-blend-mode: normal, normal;
  401. border-radius: 14px;
  402. border: solid 1px #e5e5e5;
  403. color: #fff;
  404. }
  405. }
  406. }
  407. }
  408. .list:hover {
  409. box-shadow: 0 0 5px 0 $global-color-107;
  410. .name {
  411. background-color: $global-color-107;
  412. color: $global-color-fff;
  413. }
  414. }
  415. }
  416. }
  417. }
  418. </style>