Vue 实现滚动楼层导航定位(可实现监听页面滚动和监听某元素滚动)

实现思路:

在vue中使用v-for动态渲染楼层和楼层导航,然后使用原生JavaScript实现楼层导航动态更新与楼层滚动定位

1、页面初始化时定位首个楼层

2、页面滚动楼层自动定位到相应的楼层导航并高亮显示

3、鼠标点击楼层导航时,当前楼层导航高亮显示,并且页面自动滚动到相应的楼层

效果

页面布局  和 style

 .floor-nav {
    // position: fixed;
    // top: 180px;
    // left: 350px;
    // z-index: 10;
    float: left
  }
  .nav-list {
    width: 80px;
    padding: 0;
    display: inline-block;
    text-align: center;
    background-color: #f8f8f8;
    margin-top: 0;
    margin-right: 5px
  }
  .nav-list-item {
    display: inline-block;
    width: 100%;
    height: 100%;
    line-height: 48px;
    vertical-align: middle;
    align-self: center;
    border-bottom: 1px solid #fff;
    cursor: pointer;
    background: #067749 !important;
    color: #FFF;
    font-size: 14px;
  }
  .nav-list-item.active,
  .nav-list-item:hover {
    color: rgb(112, 243, 255);
  }
  .floor-item {
    width: 100%;
    margin: 60px auto;
    min-height: 300px;
    text-align: center;
    color: #FFF;
    background-color: #404040;
  }

 

数据 放在data中

 element: {
          nav_item: null,
          floor_item: null
        },
        timer: null,
        floorNavList: [{
          id: 1,
          name: '主诉现病史'
        }, {
          id: 2,
          name: '既往史'
        }, {
          id: 3,
          name: '过敏史'
        }, {
          id: 4,
          name: '诊断'
        }, {
          id: 5,
          name: '处方类型'
        }, {
          id: 6,
          name: '检验'
        }, {
          id: 7,
          name: '检查'
        }, {
          id: 8,
          name: '手术'
        }, {
          id: 9,
          name: '治疗'
        }, {
          id: 10,
          name: '用血'
        }, {
          id: 11,
          name: '会诊'
        }, {
          id: 12,
          name: '病历文书'
        }],

滚动事件监听 

 (1)监听页面滚动

    mounted () {
        this.element = {
          nav_item: document.getElementsByClassName('nav-list-item'),
          floor_item: document.getElementsByClassName('floor-item')
        }
        this.element.nav_item[0].classList.add('active')
        window.addEventListener('scroll', this.floorSrcollEventListener)
      },
      beforeDestroy () { 
        window.removeEventListener('scroll', this.floorSrcollEventListener)
      },

 

(2)监听元素滚动

   tips:需要监听的这个元素需要拥有固定的高度

  <div style="overflow-y:auto;width: calc(100% - 85px);float: left;" :style="{'height': height + 'px'}" id="floorWarp"   @scroll="floorSrcollEventListener">     

    mounted () {
        this.element = {
          nav_item: document.getElementsByClassName('nav-list-item'),
          floor_item: document.getElementsByClassName('floor-item')
        }
        this.element.nav_item[0].classList.add('active')
      },
      beforeDestroy () { 
        window.removeEventListener('scroll', this.floorSrcollEventListener)
      },

     tips:如果页面进行了keep-alive  beforeDestroy 就要换成 deactivated

原生js部分

export default {
  methods: {
     /**
     * 监听窗口滚动楼层导航动态定位
     */
      floorSrcollEventListener() {
        const { nav_item, floor_item } = this.element
        const window_scrollTop = this.$el.querySelector('#floorWarp').scrollTop
        for (let i = 0, len = floor_item.length; i < len; i++) {
          const floor_offsetTop = floor_item[i].offsetTop - floor_item[0].offsetTop
          if (window_scrollTop >= floor_offsetTop) {
            for (let n = 0, len = nav_item.length; n < len; n++) {
              const current_classList = nav_item[n].classList
              i === n
                ? current_classList.add('active')
                : current_classList.remove('active')
            }
          }
        }
      },
      /**
     * 设置楼层导航事件驱动方法
     * @param {Number} index  楼层下标
     */
      setFloorNavMountClick(index) {
        const { floor_item } = this.element
        const floor_offsetTop = floor_item[index].offsetTop
        const window_scrollTop = this.$el.querySelector('#floorWarp').scrollTop
        const timer = {
          step: 60,
          times: 16,
          FLOOR_OFFSETTOP: floor_offsetTop
        }
        if (window_scrollTop > floor_offsetTop) {
          this.setFloorScrollArrowUp(timer)
        } else if (window_scrollTop === floor_offsetTop) {
          return false
        } else {
          this.setFloorScrollArrowDown(timer)
        }
      },
      /**
     * 设置楼层向上滚动
     * @param {Object} timer 定时器配置
     */
      setFloorScrollArrowUp(timer) {
        clearInterval(this.timer)
        this.timer = setInterval(() => {
          const window_scrollTop = this.$el.querySelector('#floorWarp').scrollTop
          if (window_scrollTop <= timer.FLOOR_OFFSETTOP) {
            this.$el.querySelector('#floorWarp').scrollTop = timer.FLOOR_OFFSETTOP
            clearInterval(this.timer)
          } else {
            this.$el.querySelector('#floorWarp').scrollTop = window_scrollTop - timer.step
          }
        }, timer.times)
      },
      /**
     * 设置楼层向下滚动
     * @param {Object} timer 定时器配置
     */
      setFloorScrollArrowDown(timer) {
        clearInterval(this.timer)
        this.timer = setInterval(() => {
          const window_scrollTop = this.$el.querySelector('#floorWarp').scrollTop
  
          if (window_scrollTop >= timer.FLOOR_OFFSETTOP) {
            this.$el.querySelector('#floorWarp').scrollTop = timer.FLOOR_OFFSETTOP
            clearInterval(this.timer)
          } else {
            if (window_scrollTop === 2095) {
              clearInterval(this.timer)
              return false
            }
            this.$el.querySelector('#floorWarp').scrollTop = window_scrollTop + timer.step
          
          }
        }, timer.times)
      },
  }
}

 

  • 2
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
关于Vue· 美籍华人尤雨溪所作,文档健全,生态完整,与脸谱的React各领前端半边天下;· 相对于React,Vue相对于初学者更容易上手;· 目前统治C端市场的跨平台框架uni-app也是基于Vue的基础技术栈进行开发;· Vue与React在框架设计上有极高的相似度,上手Vue以后,我们可以轻松过渡到React学习;课程特色· 零基础教学,由浅入深,轻松诙谐,物超所值;· 基于官方文档教学,在学习Vue的同时学习阅读文档的方式,授人以鱼更授人以渔;· 课程包含Vuex和Vue-Router最新版内容,即所谓【Vue全家桶开发】;· 完整包含Vue3全部内容;· 含京东商城完整项目从零到一开发过程;· 含最新版@vue/cli的开发和生产环境打包流程;· 含Vite两套脚手架的开发和生产环境打包流程;· 含项目部署上线全流程;· 含自定义组件库高阶课程,带你从零到一完成电梯组件开发与发布;· 附Vue高频面试题50余道,OFFER轻松斩获;课程收益· 掌握Vue2+Vue3的框架基础知识;· 掌握Vue全家桶项目开发+打包+上线全流程;· 掌握自定义组件库技巧;· 掌握Vue高频面试题回答技巧;课程关键词Vue2 Vue3 Vuex Vue-RouterVue项目 Vue全家桶@vue/cli Vite 项目打包 项目上线 自定义组件库 Vue面试题写在最后就连太阳光照到地球都需要8分钟,所以,你也需要时间,耐心一点,持续修炼。 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值