info-detail.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593
  1. <template>
  2. <view class="info-detail">
  3. <u-navbar :title="title" :autoBack="true" placeholder>
  4. <view class="u-nav-slot" slot="right">
  5. <u-icon name="more-dot-fill" size="22" @click="open"></u-icon>
  6. </view>
  7. </u-navbar>
  8. <view class="cover">
  9. <image :src="detail.image" mode="aspectFill" v-if="type == 'info' || type == 'article'"></image>
  10. <view class="video hflex acenter jbetween" v-if="type == 'video'">
  11. <image :src="detail.image" mode="aspectFill" class="img"></image>
  12. <text class="text_hide2">{{detail.video_title}}</text>
  13. <image src="../static/images/arrow-right.png" mode="aspectFill" class="icon"></image>
  14. </view>
  15. <view v-if="type == 'post'">
  16. <u-swiper :indicator="true" indicatorMode="dot" :list="detail.images" height="452rpx" radius="12rpx"
  17. v-if="detail.images && detail.images.length > 0">
  18. </u-swiper>
  19. </view>
  20. <view class="title" v-if="detail.title">{{detail.title}}</view>
  21. <view class="user hflex acenter jbetween" v-if="detail.user">
  22. <view class="hflex acenter">
  23. <image :src="detail.user.avatar" mode="aspectFill" @click="touser"></image>
  24. <view class="user-name">{{detail.user.username}}</view>
  25. </view>
  26. <view class="user-btn hflex acenter jcenter" v-if="detail.user.id != userid" @click="toFollow">
  27. <image src="./static/like.png" mode="aspectFill"></image>
  28. <text>{{detail.is_follow == 0 ? '关注' : '已关注'}}</text>
  29. </view>
  30. </view>
  31. <view class="detail">
  32. <u-parse :content="detail.content"></u-parse>
  33. </view>
  34. <view class="time hflex jend">发布于{{detail.created_at}}</view>
  35. <view class="circle hflex acenter" v-if="detail.topic_id != 0 && detail.topic">
  36. <text>{{detail.topic.title}}</text>
  37. </view>
  38. <comment-list :show_jianpan="show_jianpan" :type="type" :id="detail.id" :user="detail.user"
  39. v-if="detail.user"></comment-list>
  40. </view>
  41. <view style="height: 166rpx;"></view>
  42. <view class="bottom hflex acneter jbetween">
  43. <u-input v-model="comment" border="none" placeholder="添加新评论"></u-input>
  44. <view class="hflex acenter jbetween bottom-right">
  45. <view class="vflex acenter jbetween bottom-item" @click="tolike">
  46. <image src="/static/images/dianzan.png" mode="aspectFill" v-if="detail.is_like == 0"></image>
  47. <image src="/static/images/dianzan2.png" mode="aspectFill" v-else></image>
  48. <text>{{detail.like_count || 0}}</text>
  49. </view>
  50. <view class="vflex acenter jbetween bottom-item">
  51. <image src="/static/images/pinglun.png" mode="aspectFill"></image>
  52. <text>{{detail.comment_count || 0}}</text>
  53. </view>
  54. <view class="vflex acenter jbetween bottom-item" @click="tocollect">
  55. <image src="/static/images/shoucang.png" mode="aspectFill" v-if="detail.is_collect == 0"></image>
  56. <image src="/static/images/shoucang2.png" mode="aspectFill" v-else></image>
  57. <text>{{detail.collect_count || 0}}</text>
  58. </view>
  59. </view>
  60. </view>
  61. <u-popup :show="show" mode="bottom" :round="10" @close="toclose">
  62. <view class="popu">
  63. <view class="top hflex acenter">
  64. <view class="vflex acenter jcenter top-item" @click="copy">
  65. <view class="img-bg vflex acenter jcenter">
  66. <image src="@/static/images/lianjie.png" mode="aspectFill"></image>
  67. </view>
  68. <text>复制链接</text>
  69. </view>
  70. <view class="vflex acenter jcenter top-item" @click="show_jubao = true">
  71. <view class="img-bg vflex acenter jcenter" style="background: #E9E9E9;">
  72. <image src="/pageA/static/warn.png" mode="aspectFill"></image>
  73. </view>
  74. <text>举报</text>
  75. </view>
  76. </view>
  77. <view class="btn hflex acenter jcenter" @click="toclose">取消</view>
  78. </view>
  79. </u-popup>
  80. <u-popup :show="show_jubao" @close="toclose" mode="bottom" :round="10">
  81. <view class="popu1">
  82. <view class="title hflex acenter jbetween">
  83. <text>内容描述</text>
  84. <u-icon name="close" color="#000000" size="14" @click="toclose"></u-icon>
  85. </view>
  86. <u--textarea v-model="jubao" placeholder="请对您举报的内容进行文字描述"></u--textarea>
  87. <view class="title">
  88. <text>辅助图片</text>
  89. </view>
  90. <u-upload :fileList="fileList1" @afterRead="afterRead" @delete="deletePic" name="1" :maxCount="9"
  91. width="160rpx" height="160rpx">
  92. <view class="upload-bg hflex acenter jcenter">
  93. <u-icon name="plus" color="#CCCCCC" size="20"></u-icon>
  94. </view>
  95. </u-upload>
  96. <view class="btn" :class="jubao != '' ? 'btn2' : ''" @click="tojubao">确定</view>
  97. </view>
  98. </u-popup>
  99. </view>
  100. </template>
  101. <script>
  102. import commentList from '@/components/comment/index.vue'
  103. import $api from '@/static/js/api.js'
  104. export default {
  105. components: {
  106. commentList
  107. },
  108. data() {
  109. return {
  110. id: '',
  111. detail: {},
  112. title: '资讯',
  113. comment: '',
  114. show_jianpan: false,
  115. type: '',
  116. userid: '',
  117. show: false,
  118. fileList1: [],
  119. jubao: '',
  120. show_jubao: false,
  121. login: false,
  122. }
  123. },
  124. onLoad(option) {
  125. this.userid = uni.getStorageSync('userid')
  126. this.login = uni.getStorageSync('token') ? true: false
  127. console.log(this.userid);
  128. if (option.id) {
  129. this.id = option.id
  130. }
  131. if (option.type) {
  132. this.type = option.type
  133. if (this.type == 'info') {
  134. this.title = '资讯'
  135. }
  136. if (this.type == 'article') {
  137. this.title = '文章'
  138. }
  139. if (this.type == 'video') {
  140. this.title = '视频'
  141. }
  142. if (this.type == 'post') {
  143. this.title = '论坛'
  144. }
  145. }
  146. this.getdata()
  147. uni.onKeyboardHeightChange(this.onKeyboardHeightChange);
  148. },
  149. onUnload() {
  150. uni.offKeyboardHeightChange(this.onKeyboardHeightChange);
  151. },
  152. methods: {
  153. touser() {
  154. uni.navigateTo({
  155. url: '/pageC/otherInfo?id=' + this.detail.user.id
  156. })
  157. },
  158. toFollow() {
  159. if(!this.login) {
  160. $api.info('请先登录')
  161. return
  162. }
  163. var that = this
  164. $api.req({
  165. url: 'user/' + that.detail.user.id + '/follow',
  166. method: 'post'
  167. }, function(res) {
  168. $api.info(res.msg)
  169. that.getdata()
  170. })
  171. },
  172. tojubao() {
  173. if(!this.login) {
  174. $api.info('请先登录')
  175. return
  176. }
  177. var that = this
  178. let image = []
  179. if(that.fileList1.length>0) {
  180. for(var i=0;i<that.fileList1.length;i++) {
  181. image.push(that.fileList1[i].url)
  182. }
  183. }
  184. $api.req({
  185. url: 'report',
  186. method: 'post',
  187. data: {
  188. source_type: that.type,
  189. source_id: that.id,
  190. content: that.jubao,
  191. images: image
  192. }
  193. }, function(res) {
  194. $api.info('举报成功')
  195. that.toclose()
  196. })
  197. },
  198. // 删除图片
  199. deletePic(event) {
  200. this[`fileList${event.name}`].splice(event.index, 1)
  201. },
  202. // 新增图片
  203. async afterRead(event) {
  204. // 当设置 multiple 为 true 时, file 为数组格式,否则为对象格式
  205. let lists = [].concat(event.file)
  206. let fileListLen = this[`fileList${event.name}`].length
  207. lists.map((item) => {
  208. this[`fileList${event.name}`].push({
  209. ...item,
  210. status: 'uploading',
  211. message: '上传中'
  212. })
  213. })
  214. for (let i = 0; i < lists.length; i++) {
  215. const result = await this.uploadFilePromise(lists[i].url)
  216. let item = this[`fileList${event.name}`][fileListLen]
  217. this[`fileList${event.name}`].splice(fileListLen, 1, Object.assign(item, {
  218. status: 'success',
  219. message: '',
  220. url: result
  221. }))
  222. fileListLen++
  223. }
  224. },
  225. uploadFilePromise(url) {
  226. return new Promise((resolve, reject) => {
  227. let a = uni.uploadFile({
  228. url: $api.config.baseUrl + 'upload/image', // 仅为示例,非真实的接口地址
  229. filePath: url,
  230. name: 'image',
  231. formData: {
  232. user: 'test'
  233. },
  234. success: (res) => {
  235. let data = JSON.parse(res.data)
  236. setTimeout(() => {
  237. resolve(data.data.url)
  238. }, 1000)
  239. }
  240. });
  241. })
  242. },
  243. copy() {
  244. let text = $api.config.luck
  245. if(this.type == 'info') {
  246. text += 'news?id=' + this.id
  247. } else if(this.type == 'articles') {
  248. text += 'Articles?id=' + this.id
  249. } else if(this.type == 'video') {
  250. text += 'videos?id=' + this.id + '&type=common'
  251. }
  252. uni.setClipboardData({
  253. data: text,
  254. success() {
  255. uni.$u.toast('复制成功')
  256. }
  257. })
  258. },
  259. open() {
  260. this.show = true
  261. },
  262. toclose() {
  263. this.show = false
  264. this.show_jubao = false
  265. },
  266. tocollect() {
  267. if(!this.login) {
  268. $api.info('请先登录')
  269. return
  270. }
  271. var _this = this
  272. $api.req({
  273. url: 'collect',
  274. method: 'POST',
  275. data: {
  276. source_type: _this.type,
  277. source_id: _this.id
  278. }
  279. }, function(res) {
  280. if (res.code == 10000) {
  281. uni.$u.toast(res.msg)
  282. setTimeout(() => {
  283. _this.getdata()
  284. }, 800)
  285. }
  286. })
  287. },
  288. tolike() {
  289. var _this = this
  290. if(!this.login) {
  291. $api.info('请先登录')
  292. return
  293. }
  294. $api.req({
  295. url: 'like',
  296. method: 'POST',
  297. data: {
  298. source_type: _this.type,
  299. source_id: _this.id
  300. }
  301. }, function(res) {
  302. if (res.code == 10000) {
  303. uni.$u.toast(res.msg)
  304. setTimeout(() => {
  305. _this.getdata()
  306. }, 800)
  307. }
  308. })
  309. },
  310. onKeyboardHeightChange(e) {
  311. const {
  312. height,
  313. duration
  314. } = e;
  315. if (height > 0) {
  316. this.show_jianpan = true
  317. } else {
  318. this.show_jianpan = false
  319. }
  320. },
  321. getdata() {
  322. var _this = this
  323. $api.req({
  324. url: _this.type + '/' + _this.id,
  325. method: 'GET',
  326. /* data: {
  327. id: this.id
  328. } */
  329. }, function(res) {
  330. _this.detail = res.data
  331. })
  332. }
  333. }
  334. }
  335. </script>
  336. <style lang="scss">
  337. .info-detail::v-deep {
  338. background: #FFFFFF;
  339. min-height: 100vh;
  340. .popu1 {
  341. background: #FFFFFF;
  342. border-radius: 20rpx 20rpx 0rpx 0rpx;
  343. box-sizing: border-box;
  344. padding: 0 28rpx;
  345. .title {
  346. padding: 36rpx 0 28rpx;
  347. width: 100%;
  348. text {
  349. font-size: 32rpx;
  350. font-weight: 500;
  351. color: #333333;
  352. }
  353. }
  354. .u-textarea {
  355. width: 694rpx;
  356. height: 236rpx;
  357. background: #F5F5F5;
  358. border-radius: 12rpx;
  359. padding: 28rpx 20rpx !important;
  360. box-sizing: border-box;
  361. border: none;
  362. }
  363. .upload-bg {
  364. width: 160rpx;
  365. height: 160rpx;
  366. background: #F5F5F5;
  367. border-radius: 12rpx;
  368. }
  369. .btn2 {
  370. opacity: 1 !important;
  371. }
  372. .btn {
  373. margin: 156rpx 0 54rpx;
  374. width: 694rpx;
  375. height: 88rpx;
  376. background: #00B0B0;
  377. border-radius: 44rpx;
  378. opacity: 0.2;
  379. font-size: 32rpx;
  380. font-family: PingFangSC, PingFang SC;
  381. font-weight: 500;
  382. color: #FFFFFF;
  383. line-height: 88rpx;
  384. text-align: center;
  385. }
  386. }
  387. .popu {
  388. background: #FFFFFF;
  389. border-radius: 20rpx 20rpx 0rpx 0rpx;
  390. padding: 82rpx 30rpx;
  391. .btn {
  392. padding: 24rpx 0 0;
  393. font-size: 32rpx;
  394. font-family: PingFangSC, PingFang SC;
  395. font-weight: 400;
  396. color: #444444;
  397. }
  398. .top {
  399. padding-bottom: 40rpx;
  400. border-bottom: 1px solid #F7F7F7;
  401. .top-item {
  402. padding: 0 0 0 108rpx;
  403. .img-bg {
  404. width: 96rpx;
  405. height: 96rpx;
  406. background: #00B0B0;
  407. border-radius: 50%;
  408. image {
  409. width: 36rpx;
  410. height: 36rpx;
  411. }
  412. }
  413. text {
  414. padding: 16rpx 0 0;
  415. font-size: 22rpx;
  416. font-family: PingFangSC, PingFang SC;
  417. font-weight: 400;
  418. color: #444444;
  419. }
  420. }
  421. }
  422. }
  423. .bottom {
  424. width: 100%;
  425. position: fixed;
  426. bottom: 0;
  427. left: 0;
  428. background: #FFFFFF;
  429. box-shadow: 0rpx 0rpx 0rpx 0rpx rgba(0, 0, 0, 0.1);
  430. padding: 14rpx 24rpx 80rpx;
  431. box-sizing: border-box;
  432. .bottom-right {
  433. .bottom-item {
  434. margin-left: 38rpx;
  435. height: 72rpx;
  436. image {
  437. width: 40rpx;
  438. height: 38rpx;
  439. }
  440. text {
  441. padding-top: 8rpx;
  442. font-size: 20rpx;
  443. font-family: SFPro, SFPro;
  444. font-weight: 400;
  445. color: #7C7C7C;
  446. }
  447. }
  448. }
  449. .u-input {
  450. background: #F5F5F5;
  451. border-radius: 40rpx;
  452. width: 446rpx !important;
  453. height: 72rpx;
  454. padding: 0 32rpx !important;
  455. }
  456. }
  457. .cover {
  458. width: 100%;
  459. .video {
  460. width: 100%;
  461. box-sizing: border-box;
  462. padding: 28rpx;
  463. // height: 182rpx;
  464. background: #F5F5F5;
  465. .img {
  466. width: 224rpx;
  467. height: 126rpx;
  468. }
  469. text {
  470. font-size: 28rpx;
  471. font-family: SFPro, SFPro;
  472. font-weight: 500;
  473. color: #333333;
  474. width: 354rpx;
  475. }
  476. .icon {
  477. width: 24rpx;
  478. height: 24rpx;
  479. }
  480. }
  481. image {
  482. width: 100%;
  483. }
  484. .title {
  485. padding: 28rpx 32rpx;
  486. font-size: 36rpx;
  487. font-family: PingFangSC, PingFang SC;
  488. font-weight: 600;
  489. color: #333333;
  490. }
  491. .user {
  492. padding: 30rpx 28rpx 30rpx;
  493. image {
  494. width: 68rpx;
  495. height: 68rpx;
  496. border-radius: 50%;
  497. }
  498. .user-name {
  499. font-size: 28rpx;
  500. font-family: PingFangSC, PingFang SC;
  501. font-weight: 600;
  502. color: #333333;
  503. padding-left: 20rpx;
  504. }
  505. .user-btn {
  506. width: 140rpx;
  507. height: 60rpx;
  508. background: #00B0B0;
  509. border-radius: 30rpx;
  510. image {
  511. width: 36rpx;
  512. height: 36rpx;
  513. }
  514. text {
  515. font-size: 28rpx;
  516. font-family: PingFangSC, PingFang SC;
  517. font-weight: 500;
  518. color: #FFFFFF;
  519. padding-left: 8rpx;
  520. }
  521. }
  522. }
  523. .detail {
  524. padding: 0 28rpx;
  525. }
  526. .time {
  527. padding: 32rpx 28rpx;
  528. font-size: 28rpx;
  529. font-family: SFPro, SFPro;
  530. font-weight: 400;
  531. color: #999999;
  532. }
  533. .circle {
  534. margin: 0 28rpx 48rpx;
  535. background: rgba(0, 176, 176, .1);
  536. border-radius: 34rpx;
  537. padding: 14rpx 20rpx;
  538. max-width: max-content;
  539. image {
  540. width: 40rpx;
  541. height: 40rpx;
  542. }
  543. text {
  544. font-size: 28rpx;
  545. font-family: AppleColorEmoji;
  546. color: #00B0B0;
  547. }
  548. }
  549. }
  550. }
  551. </style>