comment.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. <template>
  2. <view :style="viewColor">
  3. <view class="container" :class="isShow==true?'on':''" @click.stop="loseFocus">
  4. <view class="main_content">
  5. <view class="header">
  6. <text class="title">评论 {{all}}</text>
  7. <text class="iconfont icon-guanbi5" @click="close"></text>
  8. </view>
  9. <view class="main">
  10. <scroll-view scroll-y="true">
  11. <view v-if="list.length > 0" @touchmove="onTouchmove" id="reply">
  12. <view class="common_list" v-for="(item, index) in list" :key="index">
  13. <view class="commen_one">
  14. <image :src="(item.author&&item.author.avatar) || '/static/images/f.png'" class="image"></image>
  15. </view>
  16. <view class="info_count">
  17. <view class="info">
  18. <view class="message" @click.stop="toReply(item,index)">
  19. <view v-if="item.author" class="name">{{item.author.nickname}}</view>
  20. <view class="desc">{{item.content}}</view>
  21. <view class="time">{{item.create_time}}</view>
  22. </view>
  23. <view class="like" @click.stop="starComment(item)">
  24. <view class="iconfont":class="item.relevance_id ? 'icon-yidianzan' : 'icon-dianzan1'"></view>
  25. {{item.count_start}}
  26. </view>
  27. </view>
  28. <view v-if="item.children && item.children.length > 0" class="reply_count">
  29. <view v-for="(itemn,indexn) in item.children" :key="indexn" class="reply_list">
  30. <view class="item">
  31. <view class="item_count" @click.stop="toReply(itemn,index)">
  32. <image class="image" :src="itemn.author && itemn.author.avatar || '/static/images/f.png'"></image>
  33. <view v-if="itemn.author" class="name_two">{{itemn.author.nickname}}</view>
  34. <view class="desc_two">
  35. <text class="reply_user" v-if="itemn.reply">回复 @{{itemn.reply.nickname}} </text> {{itemn.content}}
  36. </view>
  37. <view class="time_two">{{itemn.create_time}}</view>
  38. </view>
  39. <view class="like_two" @click.stop="starComment(itemn)">
  40. <view class="iconfont":class="itemn.relevance_id ? 'icon-yidianzan' : 'icon-dianzan1'"></view>
  41. {{itemn.count_start}}
  42. </view>
  43. </view>
  44. </view>
  45. </view>
  46. </view>
  47. </view>
  48. <view class="end"><text>到底了</text></view>
  49. </view>
  50. <Loading :loaded="loaded" :loading="loading"></Loading>
  51. <view v-if="list.length == 0 && !loading" class="empty">
  52. <image src="/static/images/no_commen.png"></image>
  53. <text>暂无评论,快去抢沙发吧~</text>
  54. </view>
  55. </scroll-view>
  56. </view>
  57. </view>
  58. <view class="release_bar" :style="'bottom:'+bottom+'rpx;'">
  59. <image class="image" :src="userInfo.avatar || '/static/images/f.png'"></image>
  60. <view class="input_count">
  61. <input type="text" :placeholder="placeholder" placeholder-style="color: #999999; font-size: 26rpx;" v-model="content" :focus="focus" confirm-type="search">
  62. </view>
  63. <button class="send" @click.stop="submitComment">发送</button>
  64. </view>
  65. </view>
  66. <!-- 绑定手机号 -->
  67. <uni-popup ref="bindmobile" type="bottom">
  68. <bindmobile @close="closepoup" :isCommuity="true"></bindmobile>
  69. </uni-popup>
  70. <view class='mask' catchtouchmove="true" :hidden='isShow==false' @tap="close"></view>
  71. <authorize @onLoadFun="onLoadFun" :isAuto="isAuto" :isShowAuth="isShowAuth" @authColse="authColse"></authorize>
  72. </view>
  73. </template>
  74. <script>
  75. import { replyLstApi, starCommentApi, replyCreateApi } from '@/api/community.js';
  76. import Loading from '@/components/Loading/index.vue';
  77. import bindmobile from '@/components/bindmobile.vue';
  78. import { getUserInfo } from '@/api/user.js';
  79. import { mapGetters } from "vuex";
  80. import { configMap } from '@/utils';
  81. import authorize from '@/components/Authorize';
  82. export default {
  83. props:{
  84. isShow: {
  85. type: Boolean,
  86. default: false
  87. },
  88. bottom: {
  89. type: Number,
  90. default: 0
  91. },
  92. userInfo: {
  93. type: Object,
  94. default: () => {uid:0;avatar:""}
  95. }
  96. },
  97. components: {
  98. Loading,bindmobile,authorize
  99. },
  100. computed: configMap({community_reply_auth:0},mapGetters(['isLogin', 'viewColor'])),
  101. data() {
  102. return {
  103. content: '',
  104. id: "",
  105. list: [],
  106. loaded: false,
  107. loading: false,
  108. where: {
  109. page: 1,
  110. limit: 10
  111. },
  112. reply_id: "",
  113. placeholder: "快来说点儿什么吧...",
  114. isChild: false,
  115. index: 0,
  116. listIndex: 0,
  117. focus: false,
  118. all: 0,
  119. isShowAuth: false, //是否隐藏授权
  120. isAuto: false, //没有授权的不会自动授权
  121. };
  122. },
  123. methods: {
  124. // 点击关闭按钮
  125. close() {
  126. this.$emit('close');
  127. },
  128. onTouchmove(e){
  129. if (this.loadend || this.loading) return;
  130. const query = uni.createSelectorQuery().in(this);
  131. query.select('#reply').boundingClientRect(data => {
  132. console.log(data)
  133. if(data.bottom < 1500 && data.top < 0) {
  134. this.getList();
  135. }
  136. }).exec();
  137. // 模拟触底刷新
  138. },
  139. // 授权回调
  140. onLoadFun() {
  141. this.isShowAuth = false
  142. },
  143. // 授权关闭
  144. authColse: function(e) {
  145. this.isShowAuth = e
  146. },
  147. getData(item,index){
  148. this.id = item.community_id
  149. this.loading = this.loaded = false
  150. this.where.page = 1
  151. this.list = []
  152. this.getList()
  153. this.listIndex = index
  154. },
  155. getList(){
  156. let that = this;
  157. if(that.loading || that.loaded) return;
  158. that.loading = true;
  159. replyLstApi(that.id,that.where).then(res => {
  160. that.loading = false;
  161. that.all = res.data.all;
  162. that.loaded = res.data.list.length < that.where.limit;
  163. that.list.push.apply(that.list, res.data.list);
  164. that.where.page = that.where.page + 1;
  165. },
  166. error => {
  167. that.$util.Tips({
  168. title: error.msg
  169. })
  170. }
  171. );
  172. },
  173. /*发表评论*/
  174. submitComment(){
  175. let that = this;
  176. if (that.isLogin === false) {
  177. that.isAuto = true;
  178. that.isShowAuth = true;
  179. }else{
  180. that.getUserInfo();
  181. }
  182. },
  183. /**
  184. * 获取个人用户信息
  185. */
  186. getUserInfo: function() {
  187. let that = this;
  188. getUserInfo().then(res => {
  189. that.userInfo = res.data;
  190. /*判断是否绑定手机号*/
  191. if(res.data.phone || that.community_reply_auth == 0){
  192. that.createReply()
  193. }else{
  194. that.$refs.bindmobile.open()
  195. }
  196. });
  197. },
  198. createReply() {
  199. let that = this;
  200. let reply_id = that.reply_id ? that.reply_id : 0
  201. replyCreateApi(that.id,{content: that.content,reply_id: reply_id}).then(res => {
  202. that.$util.Tips({
  203. title: res.message
  204. });
  205. if(that.isChild){
  206. if(that.list[that.index]['children']){
  207. that.list[that.index]['children'].push(res.data)
  208. }else{
  209. that.list[that.index]['children'] = [res.data]
  210. }
  211. }else{
  212. that.list.unshift(res.data)
  213. }
  214. that.all++
  215. that.content = ""
  216. that.$util.Tips({
  217. title: res.message
  218. });
  219. that.$emit('successFul',that.listIndex);
  220. that.loseFocus()
  221. }).catch(err => {
  222. uni.showToast({
  223. title: err,
  224. icon: 'none'
  225. })
  226. });
  227. },
  228. toReply(item,index){
  229. this.content = ""
  230. this.placeholder = '回复:'+item.author.nickname
  231. this.reply_id = item.reply_id
  232. this.isChild = true
  233. this.index = index;
  234. this.focus = true;
  235. },
  236. loseFocus(){
  237. this.focus = false;
  238. this.reply_id = 0;
  239. this.placeholder = "快来说点儿什么吧..."
  240. this.isChild = false
  241. // this.content = ""
  242. },
  243. /*点赞评论*/
  244. starComment(item){
  245. let that = this;
  246. let status = item.relevance_id ? 0 : 1
  247. starCommentApi(item.reply_id,{status: status}).then(res => {
  248. if (res.status === 200) {
  249. if(item.relevance_id){
  250. item.count_start--;
  251. item.count_start = item.count_start == 0 ? 0 : item.count_start
  252. item.relevance_id = false
  253. }else{
  254. item.count_start++;
  255. item.relevance_id = true
  256. }
  257. }
  258. that.$util.Tips({
  259. title: res.message
  260. });
  261. }).catch(err => {
  262. uni.showToast({
  263. title: err,
  264. icon: 'none'
  265. })
  266. });
  267. },
  268. closepoup(){
  269. this.$refs.bindmobile.close()
  270. },
  271. }
  272. }
  273. </script>
  274. <style lang="scss" scoped>
  275. .container{
  276. display: block;
  277. background: #ffffff;
  278. border-radius: 16rpx 16rpx 0 0;
  279. position: relative;
  280. position: fixed;
  281. bottom: 0;
  282. padding-bottom: calc(0rpx+ constant(safe-area-inset-bottom)); ///兼容 IOS<11.2/
  283. padding-bottom: calc(0rpx + env(safe-area-inset-bottom)); ///兼容 IOS>11.2/
  284. width: 100%;
  285. left: 0;
  286. background-color: #f5f5f5;
  287. z-index: 40;
  288. border-radius: 16rpx 16rpx 0 0;
  289. transform: translate3d(0, 100%, 0);
  290. transition: all .3s cubic-bezier(.25, .5, .5, .9);
  291. &.on {
  292. transform: translate3d(0, 0, 0);
  293. }
  294. .main_content{
  295. padding: 24rpx 30rpx;
  296. border-bottom: 1px solid #F5F5F5;
  297. position: relative;
  298. }
  299. .header{
  300. position: relative;
  301. text-align: center;
  302. .title{
  303. color: #282828;
  304. font-size: 36rpx;
  305. font-weight: bold;
  306. }
  307. .iconfont{
  308. color: #8A8A8A;
  309. font-size: 36rpx;
  310. position: absolute;
  311. top: 4rpx;
  312. right: 0;
  313. }
  314. }
  315. }
  316. scroll-view{
  317. max-height: 60vh;
  318. }
  319. .main{
  320. margin-top: 60rpx;
  321. padding-bottom: 20rpx;
  322. position: static;
  323. .common_list{
  324. position: relative;
  325. padding-left: 94rpx;
  326. margin-bottom: 20rpx;
  327. .commen_one{
  328. position: absolute;
  329. top: 0;
  330. left: 0;
  331. .image,uni-image{
  332. width: 74rpx;
  333. height: 74rpx;
  334. border-radius: 100%;
  335. }
  336. }
  337. .info{
  338. position: relative;
  339. padding-right: 90rpx;
  340. }
  341. .name,.name_two{
  342. color: #999999;
  343. font-size: 26rpx;
  344. }
  345. .desc,.desc_two{
  346. color: #282828;
  347. font-size: 28rpx;
  348. margin-top: 10rpx;
  349. }
  350. .desc_two{
  351. font-size: 26rpx;
  352. .reply_user{
  353. font-size: 24rpx;
  354. color: #4A8AC9;
  355. margin: 0 6rpx;
  356. }
  357. }
  358. .time,.time_two{
  359. color: #BBBBBB;
  360. font-size: 22rpx;
  361. margin-top: 15rpx;
  362. }
  363. .like,.like_two{
  364. color: #999999;
  365. font-size: 26rpx;
  366. text-align: center;
  367. position: absolute;
  368. top: 0;
  369. right: 0;
  370. width: 75rpx;
  371. .icon-yidianzan{
  372. color: var(--view-theme);
  373. }
  374. }
  375. .reply_list{
  376. margin-top: 20rpx;
  377. .item{
  378. padding-right: 140rpx;
  379. position: relative;
  380. }
  381. .item_count{
  382. position: relative;
  383. padding-left: 56rpx;
  384. .image,un-image{
  385. width: 36rpx;
  386. height: 36rpx;
  387. border-radius: 100%;
  388. position: absolute;
  389. top: 0;
  390. left: 0;
  391. }
  392. }
  393. }
  394. }
  395. .end{
  396. margin-top: 50rpx;
  397. text-align: center;
  398. text{
  399. color: #999999;
  400. font-size: 22rpx;
  401. position: relative;
  402. &::before,&::after{
  403. content: "";
  404. display: inline-block;
  405. width: 22rpx;
  406. height: 1rpx;
  407. background: #999999;
  408. position: absolute;
  409. top: 18rpx;
  410. opacity: .5;
  411. }
  412. &::before{
  413. left: -30rpx;
  414. }
  415. &::after{
  416. right: -30rpx;
  417. }
  418. }
  419. }
  420. }
  421. .release_bar{
  422. // position: absolute;
  423. width: 100%;
  424. left: 0;
  425. /*#ifndef MP*/
  426. bottom: 0;
  427. /*#endif*/
  428. /*#ifdef MP*/
  429. bottom: 10rpx;
  430. /*#endif*/
  431. background: #ffffff;
  432. display: flex;
  433. align-items: center;
  434. justify-content: space-between;
  435. padding: 15rpx 30rpx;
  436. border-top: 1px solid #F5F5F5;
  437. flex-direction: row;
  438. .avatar,image,uni-image{
  439. width: 54rpx;
  440. height: 54rpx;
  441. border-radius: 100%;
  442. }
  443. .input_count{
  444. width: 480rpx;
  445. background: #F7F7F7;
  446. border-radius: 31rpx;
  447. padding: 12rpx 30rpx;
  448. }
  449. .send{
  450. font-size: 26rpx;
  451. color: #ffffff;
  452. padding: 12rpx 30rpx;
  453. background-image: linear-gradient(126deg, var(--view-bntColor21) 0%, var(--view-bntColor22) 100%);
  454. border-radius: 30rpx;
  455. text-align: center;
  456. }
  457. }
  458. .empty{
  459. display: block;
  460. margin: 130rpx 0 150rpx;
  461. text-align: center;
  462. image,uni-image{
  463. display: inline-block;
  464. width: 414rpx;
  465. height: 305rpx;
  466. }
  467. text{
  468. display: block;
  469. color: #999999;
  470. font-size: 26rpx;
  471. }
  472. }
  473. </style>