上下拉-抽屉菜单实现

在开发中一般我们都使用左右侧换菜单,而上下抽屉菜单却很少见,今天我们实现一个简单的上下拉抽屉菜单。

首先我们先直观的看下上下拉抽屉菜单的实现效果:

                                       

IOS vue向上滑动 vue滑动菜单_js

在学习抽屉菜单前我们先学习标签的touch事件。

一、事件:

1、标签所对应touch事件:

  • touchstart事件:当手指触摸屏幕时候触发,即使已经有一个手指放在屏幕上也会触发。
  • touchmove事件:当手指在屏幕上滑动的时候连续地触发。在这个事件发生期间,调用preventDefault()事件可以阻止滚动。
  • touchend事件:当手指从屏幕上离开的时候触发。
  • touchcancel事件:当系统停止跟踪触摸的时候触发。关于这个事件的确切出发时间,文档中并没有具体说明,咱们只能去猜测了。

2、而每种touch的event事件都包含如下属性:

  • clientX:触摸目标在视口中的x坐标。
  • clientY:触摸目标在视口中的y坐标。
  • identifier:标识触摸的唯一ID。
  • pageX:触摸目标在页面中的x坐标。
  • pageY:触摸目标在页面中的y坐标。
  • screenX:触摸目标在屏幕中的x坐标。
  • screenY:触摸目标在屏幕中的y坐标。
  • target:触目的DOM节点目标。

3、事件监听处理

a) 声明标签对,并设置ref属性

<div ref="drawer"> 
    ...
</div>

b) 通过ref获取当前div标签,对div标签进行监听

this.$refs.drawer.addEventListener('touchstart', (event) => {
        console.log('====touchstart======', event)
      })
      this.$refs.drawer.addEventListener('touchmove', (event) => {
        console.log('====touchmove======', event)
      })
      this.$refs.drawer.addEventListener('touchend', (event) => {
        console.log('====touchend======', event)
      })
      this.$refs.drawer.addEventListener('touchcancel', (event) => {
        console.log('=====touchcancel=====', event)
      })

学习完事件后,接下来就实现抽屉效果。首先我们将抽屉布局固定到底部,接下来我们只需要改变抽屉的高度即可实现打开抽屉和关闭抽屉效果。

4、实现步骤:

  • touchstart事件触发时记录用户触摸的位置
  • touchmove事件触发时计算用户移动的距离
  • touchend事件触发时根据用户移动的偏移量的正负值判断方向,并设置抽屉到打开回或关闭状态

a) touchstart事件触发处理:

this.$refs.drawer.addEventListener('touchstart', (event) => {
        //记录点击坐标
        this.startPos.y = event.targetTouches[0].pageY
        this.defaultHeight = this.currentHeight
      })

b) touchmove事件计算处理

this.$refs.drawer.addEventListener('touchmove', (event) => {
        //计算移动距离
        if (event.targetTouches.length > 1) {
          return
        }
        let touch = event.targetTouches[0]
        let dy = ((this.startPos.y - touch.pageY) * 100) / 667

        if (this.isToped && !this.isScollToTop && dy < 0) {
          this.endPos.y = touch.pageY
          this.currentHeight = Math.max(
            this.window.startHeight,
            this.defaultHeight + dy
          )
          console.log(this.currentHeight)
          event.preventDefault()
        } else if (this.isToped && dy > 0 && this.isScollToTop) {
          //向上
          event.preventDefault()
          this.endPos.y = touch.pageY
          this.currentHeight = Math.min(
            this.window.endHeight,
            this.defaultHeight + dy
          )
          console.log(this.currentHeight)
        }
      })

c) touchend事件触发时处理:

this.$refs.drawer.addEventListener('touchend', (event) => {
        console.log('====touchend======', event)
        // 阈值 = 差值的20%
        const threshold =
          Math.abs(this.defaultHeight - this.currentHeight) * 0.2
        if (!this.isToped || threshold === 0) {
          return
        }
        const scrollDy = Math.abs(this.endPos.y - this.startPos.y)
        const isDirectorTop = this.endPos.y - this.startPos.y < 0
        const isToTop =
          (isDirectorTop && scrollDy > threshold) ||
          (!isDirectorTop && scrollDy < threshold)

        // 60Hz, 16.6ms
        let id = setInterval(() => {
          if (isToTop) {
            this.isScollToTop = false
            this.currentHeight = this.window.endHeight
            clearInterval(id)
          } else {
            this.currentHeight = this.window.startHeight
            this.isScollToTop = true
            clearInterval(id)
          }
        }, 10)
      })

d)用户点击头部打开或关闭抽屉事件处理:

openDrawable() {
      // 60Hz, 16.6ms
      let id = setInterval(() => {
        if (this.isScollToTop) {
          this.isScollToTop = false
          this.currentHeight = this.window.endHeight
          clearInterval(id)
        } else {
          this.currentHeight = this.window.startHeight
          this.isScollToTop = true
          clearInterval(id)
        }
      }, 10)
    },

最后附上完整抽屉组件分装代码:

<template>
  <div class="VerticalDrawable" :style="heightStyle" ref="drawer">
    <div class="title" @click="openDrawable">{{ title }}</div>
    <div class="listview" @scroll.passive="onListScrolled($event)">
      <div v-for="(item, index) in list" :key="index">
        <div class="listItem">{{ item }}</div>
      </div>
    </div>
  </div>
</template>

<script>
export default {
  components: {},
  props: {
    title: {
      type: String,
      default: '图书馆借阅记录',
    },
    list: {
      type: Array,
      default: () => {
        return []
      },
    },
  },
  data() {
    return {
      isToped: true, //判断列表的第一个元素是否位于顶部
      isScollToTop: true, //true 向顶部滑动
      defaultHeight: 46, //记录默认高度 %
      currentHeight: 46, //记录当前窗体高度%
      window: {
        startHeight: 46,
        endHeight: 82,
      },
      startPos: {
        y: 0,
      },
      endPos: {
        y: 0,
      },
    }
  },
  computed: {
    heightStyle() {
      return {
        height: this.currentHeight + '%',
      }
    },
  },
  created() {},
  mounted() {
    this.handleTounchListener()
  },
  methods: {
    openDrawable() {
      // 60Hz, 16.6ms
      let id = setInterval(() => {
        if (this.isScollToTop) {
          this.isScollToTop = false
          this.currentHeight = this.window.endHeight
          clearInterval(id)
        } else {
          this.currentHeight = this.window.startHeight
          this.isScollToTop = true
          clearInterval(id)
        }
      }, 10)
    },
    onListScrolled(event) {
      let top = event.target.scrollTop
      this.isToped = top === 0
    },
    handleTounchListener() {
      this.$refs.drawer.addEventListener('touchstart', (event) => {
        //记录点击坐标
        this.startPos.y = event.targetTouches[0].pageY
        this.defaultHeight = this.currentHeight
      })
      this.$refs.drawer.addEventListener('touchmove', (event) => {
        //计算移动距离
        if (event.targetTouches.length > 1) {
          return
        }
        let touch = event.targetTouches[0]
        let dy = ((this.startPos.y - touch.pageY) * 100) / 667

        if (this.isToped && !this.isScollToTop && dy < 0) {
          this.endPos.y = touch.pageY
          this.currentHeight = Math.max(
            this.window.startHeight,
            this.defaultHeight + dy
          )
          console.log(this.currentHeight)
          event.preventDefault()
        } else if (this.isToped && dy > 0 && this.isScollToTop) {
          //向上
          event.preventDefault()
          this.endPos.y = touch.pageY
          this.currentHeight = Math.min(
            this.window.endHeight,
            this.defaultHeight + dy
          )
          console.log(this.currentHeight)
        }
      })
      this.$refs.drawer.addEventListener('touchend', (event) => {
        console.log('====touchend======', event)
        // 阈值 = 差值的20%
        const threshold =
          Math.abs(this.defaultHeight - this.currentHeight) * 0.2
        if (!this.isToped || threshold === 0) {
          return
        }
        const scrollDy = Math.abs(this.endPos.y - this.startPos.y)
        const isDirectorTop = this.endPos.y - this.startPos.y < 0
        const isToTop =
          (isDirectorTop && scrollDy > threshold) ||
          (!isDirectorTop && scrollDy < threshold)

        // 60Hz, 16.6ms
        let id = setInterval(() => {
          if (isToTop) {
            this.isScollToTop = false
            this.currentHeight = this.window.endHeight
            clearInterval(id)
          } else {
            this.currentHeight = this.window.startHeight
            this.isScollToTop = true
            clearInterval(id)
          }
        }, 10)
      })
      this.$refs.drawer.addEventListener('touchcancel', (event) => {
        console.log('=====touchcancel=====', event)
      })
    },
    onClickMore() {
      this.$emit('onClickMore')
    },
    btnClick(item) {
      debugger
      this.$emit('returnBookClick', item)
    },
    cellClick(item) {
      debugger
      this.$emit('bookCellClick', item)
    },
  },
}
</script>

<style lang="scss" scoped>
.VerticalDrawable {
  position: fixed;
  left: 0px;
  right: 0px;
  bottom: 0px;
  overflow-y: hidden;
  background: #ffffff;

  .title {
    line-height: 50px;
    text-align: center;
    background: chocolate;
  }
  .listview {
    display: flex;
    height: 100%;
    flex-direction: column;
    overflow-y: auto;
    .listItem {
      line-height: 50px;
      text-align: center;
      border-bottom: 1px solid #999999;
      margin-bottom: -1px;
    }
  }
}
</style>

在父组件中使用抽屉组件如下:

<template>
  <div class="BookHome">
    <div class="toplayout">
      <img :src="srcUrl" alt="">
      <div>顶部样式布局</div>
    </div>
    <div class="drawer">
      <VerticalDrawer 
      :list='list' />
    </div>
  </div>
</template>

<script>
  import VerticalDrawer from "./VerticalDrawer";
  export default {
    components: {
      VerticalDrawer,
    },
    data() {
      return {
        list: [],
        srcUrl:require('../assets/logo.png'),
      }
    },
    created () {
      for(var i=0;i<100;i++){
        this.list[i] = '第'+i+'条数据';
      }
    },
  }
</script>

<style lang="scss" scoped>
.BookHome{
  display: flex;
  height: 100%;
  flex-direction: column;
  .toplayout{
    height: 50%;
    text-align: center;
    border: 1px solid chocolate;
  }
}

</style>