notice.vue 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. <template>
  2. <view>
  3. <view class="cu-load padding-top-10" :class="ajax.loading?'loading':''" @tap="getHistoryMsg"></view>
  4. <navigator :url="'./detail?id='+item.id" class="cart-view-box" v-for="(item,index) in msgList" :key="index" :id="`msg-${item.id}`">
  5. <view class="text-gray text-center">{{item.time}}</view>
  6. <view class="bg-white margin-top radius card-view">
  7. <view class="content-view">
  8. <view class="text-black text-bold text-lg text-cut">{{item.title}}</view>
  9. <view class=" text-sm text-cut-2 subtitle">{{item.subtitle}}</view>
  10. <view class="flex justify-between padding-top-20" style=" border-top: 1rpx solid #f2f2f2;box-sizing: border-box;">
  11. <view class="text-df text-black">
  12. <text>查看详情</text>
  13. </view>
  14. <view style="padding-top: 8rpx;">
  15. <text class=" cuIcon-right"></text>
  16. </view>
  17. </view>
  18. </view>
  19. </view>
  20. </navigator>
  21. </view>
  22. </template>
  23. <script>
  24. import DtCustomBar from '@/comps/dt_custom_bar.vue'
  25. export default {
  26. components:{
  27. DtCustomBar
  28. },
  29. data() {
  30. return {
  31. msgList:[],
  32. ajax:{
  33. rows:10, //每页数量
  34. page:1, //页码
  35. flag:true, // 请求开关
  36. loading:true, // 加载中
  37. },
  38. content:''
  39. }
  40. },
  41. mounted() {
  42. this.$nextTick(()=>{
  43. this.getHistoryMsg();
  44. });
  45. },
  46. onPageScroll(e){
  47. if(e.scrollTop<5){
  48. this.getHistoryMsg();
  49. }
  50. },
  51. onLoad() {
  52. },
  53. methods: {
  54. // 获取历史消息
  55. getHistoryMsg(){
  56. if(!this.ajax.flag){
  57. return; //
  58. }
  59. // 此处用到 ES7 的 async/await 知识,为使代码更加优美。不懂的请自行学习。
  60. let get = async ()=>{
  61. this.hideLoadTips();
  62. this.ajax.flag = false;
  63. let data = await this.joinHistoryMsg();
  64. console.log('----- 模拟数据格式,供参考 -----');
  65. console.log(data); // 查看请求返回的数据结构
  66. // 获取待滚动元素选择器,解决插入数据后,滚动条定位时使用
  67. let selector = '';
  68. if(this.ajax.page>1){
  69. // 非第一页,则取历史消息数据的第一条信息元素
  70. selector = `#msg-${this.msgList[0].id}`;
  71. }else{
  72. // 第一页,则取当前消息数据的最后一条信息元素
  73. selector = `#msg-${data[data.length-1].id}`;
  74. }
  75. // 将获取到的消息数据合并到消息数组中
  76. this.msgList = [...data,...this.msgList];
  77. // 数据挂载后执行,不懂的请自行阅读 Vue.js 文档对 Vue.nextTick 函数说明。
  78. this.$nextTick(()=>{
  79. // 设置当前滚动的位置
  80. this.setPageScrollTo(selector);
  81. this.hideLoadTips(true);
  82. if(data.length < this.ajax.rows){
  83. // 当前消息数据条数小于请求要求条数时,则无更多消息,不再允许请求。
  84. // 可在此处编写无更多消息数据时的逻辑
  85. }else{
  86. this.ajax.page ++;
  87. // 延迟 200ms ,以保证设置窗口滚动已完成
  88. setTimeout(()=>{
  89. this.ajax.flag = true;
  90. },200)
  91. }
  92. })
  93. }
  94. get();
  95. },
  96. // 拼接历史记录消息,正式项目可替换为请求历史记录接口
  97. joinHistoryMsg(){
  98. let join = ()=>{
  99. let arr = [];
  100. //通过当前页码及页数,模拟数据内容
  101. let startIndex = (this.ajax.page-1) * this.ajax.rows;
  102. let endIndex = startIndex + this.ajax.rows;
  103. for(let i = startIndex; i < endIndex; i++){
  104. arr.push({
  105. "id":i, // 消息的ID
  106. "time":'2020年11月18日 11:00:18',
  107. "title":`${i+1}放假通知`, // 消息内容
  108. "subtitle":'2020年10月1日-2020年10月8日,中秋 国庆节放假,假期间需注意,祝国庆节玩的开心。' ,
  109. })
  110. }
  111. /*
  112. 颠倒数组中元素的顺序。将最新的数据排在本次接口返回数据的最后面。
  113. 后端接口按 消息的时间降序查找出当前页的数据后,再将本页数据按消息时间降序排序返回。
  114. 这是数据的重点,因为页面滚动条和上拉加载历史的问题。
  115. */
  116. arr.reverse();
  117. return arr;
  118. }
  119. // 此处用到 ES6 的 Promise 知识,不懂的请自行学习。
  120. return new Promise((done,fail)=>{
  121. // 无数据请求接口,由 setTimeout 模拟,正式项目替换为 ajax 即可。
  122. setTimeout(()=>{
  123. let data = join();
  124. done(data);
  125. },800);
  126. })
  127. },
  128. // 设置页面滚动位置
  129. setPageScrollTo(selector){
  130. let view = uni.createSelectorQuery().in(this).select(selector);
  131. view.boundingClientRect((res) => {
  132. uni.pageScrollTo({
  133. scrollTop:res.top - 30, // -30 为多显示出大半个消息的高度,示意上面还有信息。
  134. duration: 0
  135. });
  136. }).exec();
  137. },
  138. // 隐藏加载提示
  139. hideLoadTips(flag){
  140. if(flag){
  141. console.log("333");
  142. setTimeout(()=>{
  143. this.ajax.loading = false;
  144. },300);
  145. }else{
  146. this.ajax.loading = true;
  147. }
  148. },
  149. },
  150. }
  151. </script>
  152. <style lang="scss">
  153. page{
  154. background-color: #F3F3F3;
  155. font-size: 28rpx;
  156. }
  157. .subtitle{
  158. color: #7e7e7e;
  159. font-size: 28rpx;
  160. margin: 50rpx 0 10rpx 0;
  161. line-height: 52rpx;
  162. }
  163. .cart-view-box {
  164. padding: 36.36rpx 27.27rpx;
  165. .card-view {
  166. background-color: #FFFFFF;
  167. position: relative;
  168. border-radius: 10rpx;
  169. .head-img {
  170. width: 100%;
  171. height: 236.36rpx;
  172. border-radius: 9.09rpx 9.09rpx 0 0;
  173. }
  174. .content-view {
  175. padding: 27.27rpx;
  176. }
  177. }
  178. }
  179. </style>