index.vue 15 KB

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