项目心得二

本文介绍了如何在Vue项目中使用vant组件库,包括页面布局、样式覆盖方法,以及List组件和PullRefresh的使用技巧。详细阐述了在处理列表滚动时的加载状态控制,以及如何在切换标签时保持滚动位置独立。文章还提供了具体的代码示例,展示了异步加载数据和上拉刷新功能的实现,并讨论了解决滚动条位置问题的解决方案。
摘要由CSDN通过智能技术生成

首先页面布局,vant组件库有丰富的组件可以使用,先选择好对应的组件引入项目,再根据自己具体的要求进行修改:

如果遇到设置样式不生效的情况,可以使用/deep/加权重,

/deep/ .van-tabs__wrap {
    position: fixed;
    top: 92px;
    z-index: 1;
    left: 0;
    right: 0;
    height: 82px;
  }

同时在使用list组件和上拉刷新功能,要注意控制好加载状态:

List 组件通过 loading 和 finished 两个变量控制加载状态, 当组件初始化或滚动到到底部时,会触发 load 事件并将 loading 设置成 true,此时可以发起异步操作并更新数据,数据更新完毕后,将 loading 设置成 false 即可。 若数据已全部加载完毕,则直接将 finished 设置成 true 即可。

  • load 事件
    • List 初始化后会触发一次 load 事件,用于加载第一屏的数据。
    • 如果一次请求加载的数据条数较少,导致列表内容无法铺满当前屏幕,List 会继续触发 load 事件,直到内容铺满屏幕或数据全部加载完成。
  • loading 属性:控制加载中的 loading 状态
    • 非加载中,loading 为 false,此时会根据列表滚动位置判断是否触发 load 事件(列表内容不足一屏幕时,会直接触发)
    • 加载中,loading 为 true,表示正在发送异步请求,此时不会触发 load 事件
  • finished 属性:控制加载结束的状态
    • 在每次请求完毕后,需要手动将 loading 设置为 false,表示本次加载结束
    • 所有数据加载结束,finished 为 true,此时不会触发 load 事件
<template>
  <div class="article-list">
    <van-pull-refresh
      v-model="isLoading"
      @refresh="onRefresh"
      :success-text="successText"
    >
      <van-list
        :error.sync="error"
        error-text="请求失败,点击重新加载"
        v-model="loading"
        :finished="finished"
        finished-text="没有更多了"
        @load="onLoad"
      >
        <van-cell v-for="(art, ind) in list" :key="ind">
          <div slot="title">
              <articleItem :article='art'></articleItem>
          </div>
        </van-cell>
      </van-list>
    </van-pull-refresh>
  </div>
</template>

<script>
import { getArticle } from '@/api/article'
import articleItem from '@/components/article-item'
export default {
  name: 'articleList',
  components: {
    articleItem
  },
  props: {
    lists: {
      type: Object,
      required: true
    }
  },
  data () {
    return {
      list: [],
      loading: false,
      finished: false,
      pre_timestamp: null,
      error: false,
      isLoading: false,
      successText: '刷新成功'
    }
  },
  computed: {},
  watch: {},
  created () {},
  mounted () {},
  methods: {
    async onLoad () {
      // 异步更新数据
      // setTimeout 仅做示例,真实场景中一般为 ajax 请求
      //   setTimeout(() => {
      //     for (let i = 0; i < 10; i++) {
      //       this.list.push(this.list.length + 1);
      //     }
      //     // console.log(this.loading);
      //     // 加载状态结束
      //     this.loading = false;

      //     // 数据全部加载完成
      //     if (this.list.length >= 40) {
      //       this.finished = true;
      //     }
      //   }, 3000);
      // 1、发送请求拿数据
      try {
        const { data } = await getArticle({
          channel_id: this.lists.id,
          timestamp: this.pre_timestamp || Date.now(), // 事件戳
          with_top: 0
        })
        // if(Math.random() > 0.5) {
        //     throw new Error('错误')
        // }测试用抛错
        console.log(data)
        const { results } = data.data
        // 2、将数据追加到数组
        this.list = [...this.list, ...results]
        // 3、将loading设置成false
        this.loading = false
        // 4、判断数据是否加载完
        // 将时间戳更新,实现翻页
        this.pre_timestamp = data.data.pre_timestamp
        if (data.data.pre_timestamp === null) {
          this.finished = true
        }
      } catch (error) {
        this.error = true
        this.loading = false
        this.$toast('获取频道信息失败!')
      }
    },
    // 上拉刷新
    async onRefresh () {
      //   setTimeout(() => {
      //     this.$toast('刷新成功');
      //     this.isLoading = false;
      //     this.count++;
      //   }, 1000);
      try {
        const { data } = await getArticle({
          channel_id: this.lists.id,
          timestamp: Date.now(), // 事件戳
          with_top: 0
        })
        // if(Math.random() > 0.5) {
        //     throw new Error('错误')
        // }测试用抛错
        console.log(data)
        const { results } = data.data
        // 2、将数据追加到数组
        this.list = [...results, ...this.list]
        this.successText = `刷新了${results.length}条数据`
        this.isLoading = false
      } catch (error) {
        this.$toast('刷新失败')
        this.isLoading = false
      }
    }
  }
}
</script>
<style lang="less" scoped>
.article-list {
  // 百分比单位是相对于父元素的
  // height: 100%;

  // 视口(在移动端是布局视口)单位:vw 和 vh,不受父元素影响
  // 1vw = 视口宽度的百分之一
  // 1vh = 视口高度的百分之一
  height: 79vh;
  overflow-y: auto;
}
</style>

灵活运用vue组件,可以复用的组件封装到components中,方便复用。

同时,关于记住滚动条的位置:

在控制台中, 使用如下的代码检测滚动的元素:

function findScroller(element) {
  element.onscroll = function () {
     console.log(element)
  }
  Array.from(element.children).forEach(findScroller)
}

findScroller(document.body)

问题: 控制台显示每个列表内容滚动的都是同一个元素 body, 因而切换标签时, 无法记录列表上一次的滚动条位置

解决方法: 我们通过为每个标签的列表容器设置单独的滚动条, 这样切换不同标签时, 就不会影响相互的滚动位置

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值